523 lines
19 KiB
C
523 lines
19 KiB
C
/*
|
|
* Copyright 2010-2013 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.
|
|
*/
|
|
|
|
#define PY_SSIZE_T_CLEAN
|
|
#include <Python.h>
|
|
#include <structmember.h>
|
|
#include "error.h"
|
|
#include "types.h"
|
|
#include "utils.h"
|
|
#include "diff.h"
|
|
|
|
extern PyObject *GitError;
|
|
|
|
extern PyTypeObject TreeType;
|
|
extern PyTypeObject IndexType;
|
|
extern PyTypeObject DiffType;
|
|
extern PyTypeObject HunkType;
|
|
|
|
PyTypeObject PatchType;
|
|
|
|
PyObject*
|
|
wrap_diff(git_diff_list *diff, Repository *repo)
|
|
{
|
|
Diff *py_diff;
|
|
|
|
py_diff = PyObject_New(Diff, &DiffType);
|
|
if (py_diff) {
|
|
Py_INCREF(repo);
|
|
py_diff->repo = repo;
|
|
py_diff->list = diff;
|
|
}
|
|
|
|
return (PyObject*) py_diff;
|
|
}
|
|
|
|
PyObject*
|
|
diff_get_patch_byindex(git_diff_list* list, size_t idx)
|
|
{
|
|
const git_diff_delta* delta;
|
|
const git_diff_range* range;
|
|
git_diff_patch* patch = NULL;
|
|
size_t i, j, hunk_amounts, lines_in_hunk, line_len, header_len, additions, deletions;
|
|
const char* line, *header;
|
|
char line_origin;
|
|
int err;
|
|
Hunk *py_hunk = NULL;
|
|
Patch *py_patch = NULL;
|
|
PyObject *py_line_origin=NULL, *py_line=NULL;
|
|
|
|
err = git_diff_get_patch(&patch, &delta, list, idx);
|
|
if (err < 0)
|
|
return Error_set(err);
|
|
|
|
py_patch = PyObject_New(Patch, &PatchType);
|
|
if (py_patch != NULL) {
|
|
py_patch->old_file_path = delta->old_file.path;
|
|
py_patch->new_file_path = delta->new_file.path;
|
|
py_patch->status = git_diff_status_char(delta->status);
|
|
py_patch->similarity = delta->similarity;
|
|
py_patch->flags = delta->flags;
|
|
py_patch->old_oid = git_oid_allocfmt(&delta->old_file.oid);
|
|
py_patch->new_oid = git_oid_allocfmt(&delta->new_file.oid);
|
|
|
|
git_diff_patch_line_stats(NULL, &additions, &deletions, patch);
|
|
py_patch->additions = additions;
|
|
py_patch->deletions = deletions;
|
|
|
|
hunk_amounts = git_diff_patch_num_hunks(patch);
|
|
py_patch->hunks = PyList_New(hunk_amounts);
|
|
for (i=0; i < hunk_amounts; ++i) {
|
|
err = git_diff_patch_get_hunk(&range, &header, &header_len,
|
|
&lines_in_hunk, patch, i);
|
|
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
py_hunk = PyObject_New(Hunk, &HunkType);
|
|
if (py_hunk != NULL) {
|
|
py_hunk->old_start = range->old_start;
|
|
py_hunk->old_lines = range->old_lines;
|
|
py_hunk->new_start = range->new_start;
|
|
py_hunk->new_lines = range->new_lines;
|
|
|
|
py_hunk->lines = PyList_New(lines_in_hunk);
|
|
for (j=0; j < lines_in_hunk; ++j) {
|
|
err = git_diff_patch_get_line_in_hunk(&line_origin,
|
|
&line, &line_len, NULL, NULL, patch, i, j);
|
|
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
py_line_origin = to_unicode_n(&line_origin, 1, NULL, NULL);
|
|
py_line = to_unicode_n(line, line_len, NULL, NULL);
|
|
PyList_SetItem(py_hunk->lines, j,
|
|
Py_BuildValue("OO",
|
|
py_line_origin,
|
|
py_line
|
|
)
|
|
);
|
|
Py_DECREF(py_line_origin);
|
|
Py_DECREF(py_line);
|
|
}
|
|
|
|
PyList_SetItem((PyObject*) py_patch->hunks, i,
|
|
(PyObject*) py_hunk);
|
|
}
|
|
}
|
|
}
|
|
|
|
cleanup:
|
|
git_diff_patch_free(patch);
|
|
|
|
return (err < 0) ? Error_set(err) : (PyObject*) py_patch;
|
|
}
|
|
|
|
static void
|
|
Patch_dealloc(Patch *self)
|
|
{
|
|
Py_CLEAR(self->hunks);
|
|
free(self->old_oid);
|
|
free(self->new_oid);
|
|
// we do not have to free old_file_path and new_file_path, they will
|
|
// be freed by git_diff_list_free in Diff_dealloc
|
|
PyObject_Del(self);
|
|
}
|
|
|
|
PyMemberDef Patch_members[] = {
|
|
MEMBER(Patch, old_file_path, T_STRING, "old file path"),
|
|
MEMBER(Patch, new_file_path, T_STRING, "new file path"),
|
|
MEMBER(Patch, old_oid, T_STRING, "old oid"),
|
|
MEMBER(Patch, new_oid, T_STRING, "new oid"),
|
|
MEMBER(Patch, status, T_CHAR, "status"),
|
|
MEMBER(Patch, similarity, T_INT, "similarity"),
|
|
MEMBER(Patch, hunks, T_OBJECT, "hunks"),
|
|
MEMBER(Patch, additions, T_INT, "additions"),
|
|
MEMBER(Patch, deletions, T_INT, "deletions"),
|
|
{NULL}
|
|
};
|
|
|
|
PyDoc_STRVAR(Patch_is_binary__doc__, "True if binary data, False if not.");
|
|
|
|
PyObject *
|
|
Patch_is_binary__get__(Patch *self)
|
|
{
|
|
if (!(self->flags & GIT_DIFF_FLAG_NOT_BINARY) &&
|
|
(self->flags & GIT_DIFF_FLAG_BINARY))
|
|
Py_RETURN_TRUE;
|
|
Py_RETURN_FALSE;
|
|
}
|
|
|
|
PyGetSetDef Patch_getseters[] = {
|
|
GETTER(Patch, is_binary),
|
|
{NULL}
|
|
};
|
|
|
|
PyDoc_STRVAR(Patch__doc__, "Diff patch object.");
|
|
|
|
PyTypeObject PatchType = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
"_pygit2.Patch", /* tp_name */
|
|
sizeof(Patch), /* tp_basicsize */
|
|
0, /* tp_itemsize */
|
|
(destructor)Patch_dealloc, /* 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 | Py_TPFLAGS_BASETYPE, /* tp_flags */
|
|
Patch__doc__, /* tp_doc */
|
|
0, /* tp_traverse */
|
|
0, /* tp_clear */
|
|
0, /* tp_richcompare */
|
|
0, /* tp_weaklistoffset */
|
|
0, /* tp_iter */
|
|
0, /* tp_iternext */
|
|
0, /* tp_methods */
|
|
Patch_members, /* tp_members */
|
|
Patch_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 *
|
|
DiffIter_iternext(DiffIter *self)
|
|
{
|
|
if (self->i < self->n)
|
|
return diff_get_patch_byindex(self->diff->list, self->i++);
|
|
|
|
PyErr_SetNone(PyExc_StopIteration);
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
DiffIter_dealloc(DiffIter *self)
|
|
{
|
|
Py_CLEAR(self->diff);
|
|
PyObject_Del(self);
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(DiffIter__doc__, "Diff iterator object.");
|
|
|
|
PyTypeObject DiffIterType = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
"_pygit2.DiffIter", /* tp_name */
|
|
sizeof(DiffIter), /* tp_basicsize */
|
|
0, /* tp_itemsize */
|
|
(destructor)DiffIter_dealloc, /* 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 | Py_TPFLAGS_BASETYPE, /* tp_flags */
|
|
DiffIter__doc__, /* tp_doc */
|
|
0, /* tp_traverse */
|
|
0, /* tp_clear */
|
|
0, /* tp_richcompare */
|
|
0, /* tp_weaklistoffset */
|
|
PyObject_SelfIter, /* tp_iter */
|
|
(iternextfunc) DiffIter_iternext, /* tp_iternext */
|
|
};
|
|
|
|
|
|
PyDoc_STRVAR(Diff_patch__doc__, "Patch diff string.");
|
|
|
|
PyObject *
|
|
Diff_patch__get__(Diff *self)
|
|
{
|
|
const git_diff_delta* delta;
|
|
git_diff_patch* patch;
|
|
char **strings = NULL;
|
|
char *buffer = NULL;
|
|
int err = GIT_ERROR;
|
|
size_t i, len, num;
|
|
PyObject *py_patch = NULL;
|
|
|
|
num = git_diff_num_deltas(self->list);
|
|
MALLOC(strings, num * sizeof(char*), cleanup);
|
|
|
|
for (i = 0, len = 1; i < num ; ++i) {
|
|
err = git_diff_get_patch(&patch, &delta, self->list, i);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_diff_patch_to_str(&(strings[i]), patch);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
len += strlen(strings[i]);
|
|
git_diff_patch_free(patch);
|
|
}
|
|
|
|
CALLOC(buffer, (len + 1), sizeof(char), cleanup);
|
|
for (i = 0; i < num; ++i) {
|
|
strcat(buffer, strings[i]);
|
|
free(strings[i]);
|
|
}
|
|
free(strings);
|
|
|
|
py_patch = to_unicode(buffer, NULL, NULL);
|
|
free(buffer);
|
|
|
|
cleanup:
|
|
return (err < 0) ? Error_set(err) : py_patch;
|
|
}
|
|
|
|
|
|
static void
|
|
Hunk_dealloc(Hunk *self)
|
|
{
|
|
Py_CLEAR(self->lines);
|
|
PyObject_Del(self);
|
|
}
|
|
|
|
PyMemberDef Hunk_members[] = {
|
|
MEMBER(Hunk, old_start, T_INT, "Old start."),
|
|
MEMBER(Hunk, old_lines, T_INT, "Old lines."),
|
|
MEMBER(Hunk, new_start, T_INT, "New start."),
|
|
MEMBER(Hunk, new_lines, T_INT, "New lines."),
|
|
MEMBER(Hunk, lines, T_OBJECT, "Lines."),
|
|
{NULL}
|
|
};
|
|
|
|
|
|
PyDoc_STRVAR(Hunk__doc__, "Hunk object.");
|
|
|
|
PyTypeObject HunkType = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
"_pygit2.Hunk", /* tp_name */
|
|
sizeof(Hunk), /* tp_basicsize */
|
|
0, /* tp_itemsize */
|
|
(destructor)Hunk_dealloc, /* 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 | Py_TPFLAGS_BASETYPE, /* tp_flags */
|
|
Hunk__doc__, /* tp_doc */
|
|
0, /* tp_traverse */
|
|
0, /* tp_clear */
|
|
0, /* tp_richcompare */
|
|
0, /* tp_weaklistoffset */
|
|
0, /* tp_iter */
|
|
0, /* tp_iternext */
|
|
0, /* tp_methods */
|
|
Hunk_members, /* tp_members */
|
|
0, /* 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 */
|
|
};
|
|
|
|
|
|
PyDoc_STRVAR(Diff_merge__doc__,
|
|
"merge(diff)\n"
|
|
"\n"
|
|
"Merge one diff into another.");
|
|
|
|
PyObject *
|
|
Diff_merge(Diff *self, PyObject *args)
|
|
{
|
|
Diff *py_diff;
|
|
int err;
|
|
|
|
if (!PyArg_ParseTuple(args, "O!", &DiffType, &py_diff))
|
|
return NULL;
|
|
|
|
if (py_diff->repo->repo != self->repo->repo)
|
|
return Error_set(GIT_ERROR);
|
|
|
|
err = git_diff_merge(self->list, py_diff->list);
|
|
if (err < 0)
|
|
return Error_set(err);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Diff_find_similar__doc__,
|
|
"find_similar([flags])\n"
|
|
"\n"
|
|
"Find renamed files in diff and updates them in-place in the diff itself.");
|
|
|
|
PyObject *
|
|
Diff_find_similar(Diff *self, PyObject *args)
|
|
{
|
|
int err;
|
|
git_diff_find_options opts = GIT_DIFF_FIND_OPTIONS_INIT;
|
|
|
|
if (!PyArg_ParseTuple(args, "|i", &opts.flags))
|
|
return NULL;
|
|
|
|
err = git_diff_find_similar(self->list, &opts);
|
|
if (err < 0)
|
|
return Error_set(err);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyObject *
|
|
Diff_iter(Diff *self)
|
|
{
|
|
DiffIter *iter;
|
|
|
|
iter = PyObject_New(DiffIter, &DiffIterType);
|
|
if (iter != NULL) {
|
|
Py_INCREF(self);
|
|
iter->diff = self;
|
|
iter->i = 0;
|
|
iter->n = git_diff_num_deltas(self->list);
|
|
}
|
|
return (PyObject*)iter;
|
|
}
|
|
|
|
PyObject *
|
|
Diff_getitem(Diff *self, PyObject *value)
|
|
{
|
|
size_t i;
|
|
|
|
if (PyLong_Check(value) < 0)
|
|
return NULL;
|
|
|
|
i = PyLong_AsUnsignedLong(value);
|
|
|
|
return diff_get_patch_byindex(self->list, i);
|
|
}
|
|
|
|
|
|
static void
|
|
Diff_dealloc(Diff *self)
|
|
{
|
|
git_diff_list_free(self->list);
|
|
Py_CLEAR(self->repo);
|
|
PyObject_Del(self);
|
|
}
|
|
|
|
PyGetSetDef Diff_getseters[] = {
|
|
GETTER(Diff, patch),
|
|
{NULL}
|
|
};
|
|
|
|
PyMappingMethods Diff_as_mapping = {
|
|
0, /* mp_length */
|
|
(binaryfunc)Diff_getitem, /* mp_subscript */
|
|
0, /* mp_ass_subscript */
|
|
};
|
|
|
|
static PyMethodDef Diff_methods[] = {
|
|
METHOD(Diff, merge, METH_VARARGS),
|
|
METHOD(Diff, find_similar, METH_VARARGS),
|
|
{NULL}
|
|
};
|
|
|
|
|
|
PyDoc_STRVAR(Diff__doc__, "Diff objects.");
|
|
|
|
PyTypeObject DiffType = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
"_pygit2.Diff", /* tp_name */
|
|
sizeof(Diff), /* tp_basicsize */
|
|
0, /* tp_itemsize */
|
|
(destructor)Diff_dealloc, /* 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 */
|
|
&Diff_as_mapping, /* 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 | Py_TPFLAGS_BASETYPE, /* tp_flags */
|
|
Diff__doc__, /* tp_doc */
|
|
0, /* tp_traverse */
|
|
0, /* tp_clear */
|
|
0, /* tp_richcompare */
|
|
0, /* tp_weaklistoffset */
|
|
(getiterfunc)Diff_iter, /* tp_iter */
|
|
0, /* tp_iternext */
|
|
Diff_methods, /* tp_methods */
|
|
0, /* tp_members */
|
|
Diff_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 */
|
|
};
|