Files
deb-python-pygit2/src/branch.c
2014-02-12 23:05:51 +01:00

337 lines
9.7 KiB
C

/*
* Copyright 2010-2014 The pygit2 contributors
*
* This file is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License, version 2,
* as published by the Free Software Foundation.
*
* In addition to the permissions in the GNU General Public License,
* the authors give you unlimited permission to link the compiled
* version of this file into combinations with other programs,
* and to distribute those combinations without any restriction
* coming from the use of this file. (The General Public License
* restrictions do apply in other respects; for example, they cover
* modification of the file, and distribution when not linked into
* a combined executable.)
*
* This file is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; see the file COPYING. If not, write to
* the Free Software Foundation, 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "types.h"
#include "branch.h"
#include "error.h"
#include "reference.h"
#include "utils.h"
extern PyObject *GitError;
extern PyTypeObject ReferenceType;
PyDoc_STRVAR(Branch_delete__doc__,
"delete()\n"
"\n"
"Delete this branch. It will no longer be valid!");
PyObject *
Branch_delete(Branch *self, PyObject *args)
{
int err;
CHECK_REFERENCE(self);
/* Delete the branch */
err = git_branch_delete(self->reference);
if (err < 0)
return Error_set(err);
git_reference_free(self->reference);
self->reference = NULL; /* Invalidate the pointer */
Py_RETURN_NONE;
}
PyDoc_STRVAR(Branch_is_head__doc__,
"is_head()\n"
"\n"
"True if HEAD points at the branch, False otherwise.");
PyObject *
Branch_is_head(Branch *self)
{
int err;
CHECK_REFERENCE(self);
err = git_branch_is_head(self->reference);
if (err == 1)
Py_RETURN_TRUE;
else if (err == 0)
Py_RETURN_FALSE;
else
return Error_set(err);
}
PyDoc_STRVAR(Branch_rename__doc__,
"rename(name, force=False)\n"
"\n"
"Move/rename an existing local branch reference. The new branch name will be "
"checked for validity.\n"
"Returns the new branch.");
PyObject *
Branch_rename(Branch *self, PyObject *args)
{
int err, force = 0;
git_reference *c_out;
const char *c_name;
CHECK_REFERENCE(self);
if (!PyArg_ParseTuple(args, "s|i", &c_name, &force))
return NULL;
err = git_branch_move(&c_out, self->reference, c_name, force);
if (err == GIT_OK)
return wrap_branch(c_out, self->repo);
else
return Error_set(err);
}
PyDoc_STRVAR(Branch_branch_name__doc__,
"The name of the local or remote branch.");
PyObject *
Branch_branch_name__get__(Branch *self)
{
int err;
const char *c_name;
CHECK_REFERENCE(self);
err = git_branch_name(&c_name, self->reference);
if (err == GIT_OK)
return to_unicode(c_name, NULL, NULL);
else
return Error_set(err);
}
PyDoc_STRVAR(Branch_remote_name__doc__,
"The name of the remote that the remote tracking branch belongs to.");
PyObject *
Branch_remote_name__get__(Branch *self)
{
int err;
const char *branch_name;
char *c_name = NULL;
PyObject *py_name;
CHECK_REFERENCE(self);
branch_name = git_reference_name(self->reference);
/* Get the length of the remote name */
err = git_branch_remote_name(NULL, 0, self->repo->repo, branch_name);
if (err < GIT_OK)
return Error_set(err);
/* Get the actual remote name */
c_name = calloc(err, sizeof(char));
if (c_name == NULL)
return PyErr_NoMemory();
err = git_branch_remote_name(c_name,
err * sizeof(char),
self->repo->repo,
branch_name);
if (err < GIT_OK) {
free(c_name);
return Error_set(err);
}
py_name = to_unicode_n(c_name, err - 1, NULL, NULL);
free(c_name);
return py_name;
}
PyDoc_STRVAR(Branch_upstream__doc__,
"The branch supporting the remote tracking branch or None if this is not a "
"remote tracking branch. Set to None to unset.");
PyObject *
Branch_upstream__get__(Branch *self)
{
int err;
git_reference *c_reference;
CHECK_REFERENCE(self);
err = git_branch_upstream(&c_reference, self->reference);
if (err == GIT_ENOTFOUND)
Py_RETURN_NONE;
else if (err < GIT_OK)
return Error_set(err);
return wrap_branch(c_reference, self->repo);
}
int Branch_upstream__set__(Branch *self, Reference *py_ref)
{
int err;
const char *branch_name = NULL;
CHECK_REFERENCE_INT(self);
if ((PyObject *)py_ref != Py_None) {
if (!PyObject_TypeCheck(py_ref, (PyTypeObject *)&ReferenceType)) {
PyErr_SetObject(PyExc_TypeError, (PyObject *)py_ref);
return -1;
}
CHECK_REFERENCE_INT(py_ref);
err = git_branch_name(&branch_name, py_ref->reference);
if (err < GIT_OK) {
Error_set(err);
return -1;
}
}
err = git_branch_set_upstream(self->reference, branch_name);
if (err < GIT_OK) {
Error_set(err);
return -1;
}
return 0;
}
PyDoc_STRVAR(Branch_upstream_name__doc__,
"The name of the reference supporting the remote tracking branch.");
PyObject *
Branch_upstream_name__get__(Branch *self)
{
int err;
const char *branch_name;
char *c_name = NULL;
PyObject *py_name;
CHECK_REFERENCE(self);
branch_name = git_reference_name(self->reference);
/* Get the length of the upstream name */
err = git_branch_upstream_name(NULL, 0, self->repo->repo, branch_name);
if (err < GIT_OK)
return Error_set(err);
/* Get the actual upstream name */
c_name = calloc(err, sizeof(char));
if (c_name == NULL)
return PyErr_NoMemory();
err = git_branch_upstream_name(c_name,
err * sizeof(char),
self->repo->repo,
branch_name);
if (err < GIT_OK) {
free(c_name);
return Error_set(err);
}
py_name = to_unicode_n(c_name, err - 1, NULL, NULL);
free(c_name);
return py_name;
}
PyMethodDef Branch_methods[] = {
METHOD(Branch, delete, METH_NOARGS),
METHOD(Branch, is_head, METH_NOARGS),
METHOD(Branch, rename, METH_VARARGS),
{NULL}
};
PyGetSetDef Branch_getseters[] = {
GETTER(Branch, branch_name),
GETTER(Branch, remote_name),
GETSET(Branch, upstream),
GETTER(Branch, upstream_name),
{NULL}
};
PyDoc_STRVAR(Branch__doc__, "Branch.");
PyTypeObject BranchType = {
PyVarObject_HEAD_INIT(NULL, 0)
"_pygit2.Branch", /* tp_name */
sizeof(Branch), /* tp_basicsize */
0, /* tp_itemsize */
0, /* tp_dealloc */
0, /* tp_print */
0, /* tp_getattr */
0, /* tp_setattr */
0, /* tp_compare */
0, /* tp_repr */
0, /* tp_as_number */
0, /* tp_as_sequence */
0, /* tp_as_mapping */
0, /* tp_hash */
0, /* tp_call */
0, /* tp_str */
0, /* tp_getattro */
0, /* tp_setattro */
0, /* tp_as_buffer */
Py_TPFLAGS_DEFAULT, /* tp_flags */
Branch__doc__, /* tp_doc */
0, /* tp_traverse */
0, /* tp_clear */
0, /* tp_richcompare */
0, /* tp_weaklistoffset */
0, /* tp_iter */
0, /* tp_iternext */
Branch_methods, /* tp_methods */
0, /* tp_members */
Branch_getseters, /* tp_getset */
0, /* tp_base */
0, /* tp_dict */
0, /* tp_descr_get */
0, /* tp_descr_set */
0, /* tp_dictoffset */
0, /* tp_init */
0, /* tp_alloc */
0, /* tp_new */
};
PyObject *
wrap_branch(git_reference *c_reference, Repository *repo)
{
Branch *py_branch=NULL;
py_branch = PyObject_New(Branch, &BranchType);
if (py_branch) {
py_branch->reference = c_reference;
if (repo) {
py_branch->repo = repo;
Py_INCREF(repo);
}
}
return (PyObject *)py_branch;
}