mirror of
https://git.proxmox.com/git/libgit2
synced 2025-05-05 15:33:51 +00:00
847 lines
22 KiB
C
847 lines
22 KiB
C
/*
|
|
* Copyright (C) 2012 the libgit2 contributors
|
|
*
|
|
* This file is part of libgit2, distributed under the GNU GPL v2 with
|
|
* a Linking Exception. For full terms see the included COPYING file.
|
|
*/
|
|
#include "common.h"
|
|
#include "diff.h"
|
|
#include "fileops.h"
|
|
#include "config.h"
|
|
#include "attr_file.h"
|
|
#include "filter.h"
|
|
#include "pathspec.h"
|
|
|
|
static git_diff_delta *diff_delta__alloc(
|
|
git_diff_list *diff,
|
|
git_delta_t status,
|
|
const char *path)
|
|
{
|
|
git_diff_delta *delta = git__calloc(1, sizeof(git_diff_delta));
|
|
if (!delta)
|
|
return NULL;
|
|
|
|
delta->old_file.path = git_pool_strdup(&diff->pool, path);
|
|
if (delta->old_file.path == NULL) {
|
|
git__free(delta);
|
|
return NULL;
|
|
}
|
|
|
|
delta->new_file.path = delta->old_file.path;
|
|
|
|
if (diff->opts.flags & GIT_DIFF_REVERSE) {
|
|
switch (status) {
|
|
case GIT_DELTA_ADDED: status = GIT_DELTA_DELETED; break;
|
|
case GIT_DELTA_DELETED: status = GIT_DELTA_ADDED; break;
|
|
default: break; /* leave other status values alone */
|
|
}
|
|
}
|
|
delta->status = status;
|
|
|
|
return delta;
|
|
}
|
|
|
|
static int diff_delta__from_one(
|
|
git_diff_list *diff,
|
|
git_delta_t status,
|
|
const git_index_entry *entry)
|
|
{
|
|
git_diff_delta *delta;
|
|
|
|
if (status == GIT_DELTA_IGNORED &&
|
|
(diff->opts.flags & GIT_DIFF_INCLUDE_IGNORED) == 0)
|
|
return 0;
|
|
|
|
if (status == GIT_DELTA_UNTRACKED &&
|
|
(diff->opts.flags & GIT_DIFF_INCLUDE_UNTRACKED) == 0)
|
|
return 0;
|
|
|
|
if (!git_pathspec_match_path(
|
|
&diff->pathspec, entry->path,
|
|
(diff->opts.flags & GIT_DIFF_DISABLE_PATHSPEC_MATCH) != 0,
|
|
(diff->opts.flags & GIT_DIFF_DELTAS_ARE_ICASE) != 0))
|
|
return 0;
|
|
|
|
delta = diff_delta__alloc(diff, status, entry->path);
|
|
GITERR_CHECK_ALLOC(delta);
|
|
|
|
/* This fn is just for single-sided diffs */
|
|
assert(status != GIT_DELTA_MODIFIED);
|
|
|
|
if (delta->status == GIT_DELTA_DELETED) {
|
|
delta->old_file.mode = entry->mode;
|
|
delta->old_file.size = entry->file_size;
|
|
git_oid_cpy(&delta->old_file.oid, &entry->oid);
|
|
} else /* ADDED, IGNORED, UNTRACKED */ {
|
|
delta->new_file.mode = entry->mode;
|
|
delta->new_file.size = entry->file_size;
|
|
git_oid_cpy(&delta->new_file.oid, &entry->oid);
|
|
}
|
|
|
|
delta->old_file.flags |= GIT_DIFF_FILE_VALID_OID;
|
|
|
|
if (delta->status == GIT_DELTA_DELETED ||
|
|
!git_oid_iszero(&delta->new_file.oid))
|
|
delta->new_file.flags |= GIT_DIFF_FILE_VALID_OID;
|
|
|
|
if (git_vector_insert(&diff->deltas, delta) < 0) {
|
|
git__free(delta);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int diff_delta__from_two(
|
|
git_diff_list *diff,
|
|
git_delta_t status,
|
|
const git_index_entry *old_entry,
|
|
uint32_t old_mode,
|
|
const git_index_entry *new_entry,
|
|
uint32_t new_mode,
|
|
git_oid *new_oid)
|
|
{
|
|
git_diff_delta *delta;
|
|
|
|
if (status == GIT_DELTA_UNMODIFIED &&
|
|
(diff->opts.flags & GIT_DIFF_INCLUDE_UNMODIFIED) == 0)
|
|
return 0;
|
|
|
|
if ((diff->opts.flags & GIT_DIFF_REVERSE) != 0) {
|
|
uint32_t temp_mode = old_mode;
|
|
const git_index_entry *temp_entry = old_entry;
|
|
old_entry = new_entry;
|
|
new_entry = temp_entry;
|
|
old_mode = new_mode;
|
|
new_mode = temp_mode;
|
|
}
|
|
|
|
delta = diff_delta__alloc(diff, status, old_entry->path);
|
|
GITERR_CHECK_ALLOC(delta);
|
|
|
|
git_oid_cpy(&delta->old_file.oid, &old_entry->oid);
|
|
delta->old_file.size = old_entry->file_size;
|
|
delta->old_file.mode = old_mode;
|
|
delta->old_file.flags |= GIT_DIFF_FILE_VALID_OID;
|
|
|
|
git_oid_cpy(&delta->new_file.oid, &new_entry->oid);
|
|
delta->new_file.size = new_entry->file_size;
|
|
delta->new_file.mode = new_mode;
|
|
|
|
if (new_oid) {
|
|
if ((diff->opts.flags & GIT_DIFF_REVERSE) != 0)
|
|
git_oid_cpy(&delta->old_file.oid, new_oid);
|
|
else
|
|
git_oid_cpy(&delta->new_file.oid, new_oid);
|
|
}
|
|
|
|
if (new_oid || !git_oid_iszero(&new_entry->oid))
|
|
delta->new_file.flags |= GIT_DIFF_FILE_VALID_OID;
|
|
|
|
if (git_vector_insert(&diff->deltas, delta) < 0) {
|
|
git__free(delta);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static git_diff_delta *diff_delta__last_for_item(
|
|
git_diff_list *diff,
|
|
const git_index_entry *item)
|
|
{
|
|
git_diff_delta *delta = git_vector_last(&diff->deltas);
|
|
if (!delta)
|
|
return NULL;
|
|
|
|
switch (delta->status) {
|
|
case GIT_DELTA_UNMODIFIED:
|
|
case GIT_DELTA_DELETED:
|
|
if (git_oid_cmp(&delta->old_file.oid, &item->oid) == 0)
|
|
return delta;
|
|
break;
|
|
case GIT_DELTA_ADDED:
|
|
if (git_oid_cmp(&delta->new_file.oid, &item->oid) == 0)
|
|
return delta;
|
|
break;
|
|
case GIT_DELTA_MODIFIED:
|
|
if (git_oid_cmp(&delta->old_file.oid, &item->oid) == 0 ||
|
|
git_oid_cmp(&delta->new_file.oid, &item->oid) == 0)
|
|
return delta;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static char *diff_strdup_prefix(git_pool *pool, const char *prefix)
|
|
{
|
|
size_t len = strlen(prefix);
|
|
|
|
/* append '/' at end if needed */
|
|
if (len > 0 && prefix[len - 1] != '/')
|
|
return git_pool_strcat(pool, prefix, "/");
|
|
else
|
|
return git_pool_strndup(pool, prefix, len + 1);
|
|
}
|
|
|
|
int git_diff_delta__cmp(const void *a, const void *b)
|
|
{
|
|
const git_diff_delta *da = a, *db = b;
|
|
int val = strcmp(da->old_file.path, db->old_file.path);
|
|
return val ? val : ((int)da->status - (int)db->status);
|
|
}
|
|
|
|
bool git_diff_delta__should_skip(
|
|
const git_diff_options *opts, const git_diff_delta *delta)
|
|
{
|
|
uint32_t flags = opts ? opts->flags : 0;
|
|
|
|
if (delta->status == GIT_DELTA_UNMODIFIED &&
|
|
(flags & GIT_DIFF_INCLUDE_UNMODIFIED) == 0)
|
|
return true;
|
|
|
|
if (delta->status == GIT_DELTA_IGNORED &&
|
|
(flags & GIT_DIFF_INCLUDE_IGNORED) == 0)
|
|
return true;
|
|
|
|
if (delta->status == GIT_DELTA_UNTRACKED &&
|
|
(flags & GIT_DIFF_INCLUDE_UNTRACKED) == 0)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
static int config_bool(git_config *cfg, const char *name, int defvalue)
|
|
{
|
|
int val = defvalue;
|
|
|
|
if (git_config_get_bool(&val, cfg, name) < 0)
|
|
giterr_clear();
|
|
|
|
return val;
|
|
}
|
|
|
|
static git_diff_list *git_diff_list_alloc(
|
|
git_repository *repo, const git_diff_options *opts)
|
|
{
|
|
git_config *cfg;
|
|
git_diff_list *diff = git__calloc(1, sizeof(git_diff_list));
|
|
if (diff == NULL)
|
|
return NULL;
|
|
|
|
GIT_REFCOUNT_INC(diff);
|
|
diff->repo = repo;
|
|
|
|
if (git_vector_init(&diff->deltas, 0, git_diff_delta__cmp) < 0 ||
|
|
git_pool_init(&diff->pool, 1, 0) < 0)
|
|
goto fail;
|
|
|
|
/* load config values that affect diff behavior */
|
|
if (git_repository_config__weakptr(&cfg, repo) < 0)
|
|
goto fail;
|
|
if (config_bool(cfg, "core.symlinks", 1))
|
|
diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_HAS_SYMLINKS;
|
|
if (config_bool(cfg, "core.ignorestat", 0))
|
|
diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_ASSUME_UNCHANGED;
|
|
if (config_bool(cfg, "core.filemode", 1))
|
|
diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_TRUST_MODE_BITS;
|
|
if (config_bool(cfg, "core.trustctime", 1))
|
|
diff->diffcaps = diff->diffcaps | GIT_DIFFCAPS_TRUST_CTIME;
|
|
/* Don't set GIT_DIFFCAPS_USE_DEV - compile time option in core git */
|
|
|
|
/* TODO: there are certain config settings where even if we were
|
|
* not given an options structure, we need the diff list to have one
|
|
* so that we can store the altered default values.
|
|
*
|
|
* - diff.ignoreSubmodules
|
|
* - diff.mnemonicprefix
|
|
* - diff.noprefix
|
|
*/
|
|
|
|
if (opts == NULL)
|
|
return diff;
|
|
|
|
memcpy(&diff->opts, opts, sizeof(git_diff_options));
|
|
|
|
if(opts->flags & GIT_DIFF_IGNORE_FILEMODE)
|
|
diff->diffcaps = diff->diffcaps & ~GIT_DIFFCAPS_TRUST_MODE_BITS;
|
|
|
|
/* pathspec init will do nothing for empty pathspec */
|
|
if (git_pathspec_init(&diff->pathspec, &opts->pathspec, &diff->pool) < 0)
|
|
goto fail;
|
|
|
|
/* TODO: handle config diff.mnemonicprefix, diff.noprefix */
|
|
|
|
diff->opts.old_prefix = diff_strdup_prefix(&diff->pool,
|
|
opts->old_prefix ? opts->old_prefix : DIFF_OLD_PREFIX_DEFAULT);
|
|
diff->opts.new_prefix = diff_strdup_prefix(&diff->pool,
|
|
opts->new_prefix ? opts->new_prefix : DIFF_NEW_PREFIX_DEFAULT);
|
|
|
|
if (!diff->opts.old_prefix || !diff->opts.new_prefix)
|
|
goto fail;
|
|
|
|
if (diff->opts.flags & GIT_DIFF_REVERSE) {
|
|
char *swap = diff->opts.old_prefix;
|
|
diff->opts.old_prefix = diff->opts.new_prefix;
|
|
diff->opts.new_prefix = swap;
|
|
}
|
|
|
|
/* INCLUDE_TYPECHANGE_TREES implies INCLUDE_TYPECHANGE */
|
|
if (diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE_TREES)
|
|
diff->opts.flags |= GIT_DIFF_INCLUDE_TYPECHANGE;
|
|
|
|
return diff;
|
|
|
|
fail:
|
|
git_diff_list_free(diff);
|
|
return NULL;
|
|
}
|
|
|
|
static void diff_list_free(git_diff_list *diff)
|
|
{
|
|
git_diff_delta *delta;
|
|
unsigned int i;
|
|
|
|
git_vector_foreach(&diff->deltas, i, delta) {
|
|
git__free(delta);
|
|
diff->deltas.contents[i] = NULL;
|
|
}
|
|
git_vector_free(&diff->deltas);
|
|
|
|
git_pathspec_free(&diff->pathspec);
|
|
git_pool_clear(&diff->pool);
|
|
git__free(diff);
|
|
}
|
|
|
|
void git_diff_list_free(git_diff_list *diff)
|
|
{
|
|
if (!diff)
|
|
return;
|
|
|
|
GIT_REFCOUNT_DEC(diff, diff_list_free);
|
|
}
|
|
|
|
void git_diff_list_addref(git_diff_list *diff)
|
|
{
|
|
GIT_REFCOUNT_INC(diff);
|
|
}
|
|
|
|
int git_diff__oid_for_file(
|
|
git_repository *repo,
|
|
const char *path,
|
|
uint16_t mode,
|
|
git_off_t size,
|
|
git_oid *oid)
|
|
{
|
|
int result = 0;
|
|
git_buf full_path = GIT_BUF_INIT;
|
|
|
|
if (git_buf_joinpath(
|
|
&full_path, git_repository_workdir(repo), path) < 0)
|
|
return -1;
|
|
|
|
if (!mode) {
|
|
struct stat st;
|
|
|
|
if (p_stat(path, &st) < 0) {
|
|
giterr_set(GITERR_OS, "Could not stat '%s'", path);
|
|
result = -1;
|
|
goto cleanup;
|
|
}
|
|
|
|
mode = st.st_mode;
|
|
size = st.st_size;
|
|
}
|
|
|
|
/* calculate OID for file if possible */
|
|
if (S_ISGITLINK(mode)) {
|
|
git_submodule *sm;
|
|
const git_oid *sm_oid;
|
|
|
|
if (!git_submodule_lookup(&sm, repo, path) &&
|
|
(sm_oid = git_submodule_wd_oid(sm)) != NULL)
|
|
git_oid_cpy(oid, sm_oid);
|
|
else {
|
|
/* if submodule lookup failed probably just in an intermediate
|
|
* state where some init hasn't happened, so ignore the error
|
|
*/
|
|
giterr_clear();
|
|
memset(oid, 0, sizeof(*oid));
|
|
}
|
|
} else if (S_ISLNK(mode)) {
|
|
result = git_odb__hashlink(oid, full_path.ptr);
|
|
} else if (!git__is_sizet(size)) {
|
|
giterr_set(GITERR_OS, "File size overflow (for 32-bits) on '%s'", path);
|
|
result = -1;
|
|
} else {
|
|
git_vector filters = GIT_VECTOR_INIT;
|
|
|
|
result = git_filters_load(&filters, repo, path, GIT_FILTER_TO_ODB);
|
|
if (result >= 0) {
|
|
int fd = git_futils_open_ro(full_path.ptr);
|
|
if (fd < 0)
|
|
result = fd;
|
|
else {
|
|
result = git_odb__hashfd_filtered(
|
|
oid, fd, (size_t)size, GIT_OBJ_BLOB, &filters);
|
|
p_close(fd);
|
|
}
|
|
}
|
|
|
|
git_filters_free(&filters);
|
|
}
|
|
|
|
cleanup:
|
|
git_buf_free(&full_path);
|
|
return result;
|
|
}
|
|
|
|
#define MODE_BITS_MASK 0000777
|
|
|
|
static int maybe_modified(
|
|
git_iterator *old_iter,
|
|
const git_index_entry *oitem,
|
|
git_iterator *new_iter,
|
|
const git_index_entry *nitem,
|
|
git_diff_list *diff)
|
|
{
|
|
git_oid noid, *use_noid = NULL;
|
|
git_delta_t status = GIT_DELTA_MODIFIED;
|
|
unsigned int omode = oitem->mode;
|
|
unsigned int nmode = nitem->mode;
|
|
bool new_is_workdir = (new_iter->type == GIT_ITERATOR_WORKDIR);
|
|
|
|
GIT_UNUSED(old_iter);
|
|
|
|
if (!git_pathspec_match_path(
|
|
&diff->pathspec, oitem->path,
|
|
(diff->opts.flags & GIT_DIFF_DISABLE_PATHSPEC_MATCH) != 0,
|
|
(diff->opts.flags & GIT_DIFF_DELTAS_ARE_ICASE) != 0))
|
|
return 0;
|
|
|
|
/* on platforms with no symlinks, preserve mode of existing symlinks */
|
|
if (S_ISLNK(omode) && S_ISREG(nmode) && new_is_workdir &&
|
|
!(diff->diffcaps & GIT_DIFFCAPS_HAS_SYMLINKS))
|
|
nmode = omode;
|
|
|
|
/* on platforms with no execmode, just preserve old mode */
|
|
if (!(diff->diffcaps & GIT_DIFFCAPS_TRUST_MODE_BITS) &&
|
|
(nmode & MODE_BITS_MASK) != (omode & MODE_BITS_MASK) &&
|
|
new_is_workdir)
|
|
nmode = (nmode & ~MODE_BITS_MASK) | (omode & MODE_BITS_MASK);
|
|
|
|
/* support "assume unchanged" (poorly, b/c we still stat everything) */
|
|
if ((diff->diffcaps & GIT_DIFFCAPS_ASSUME_UNCHANGED) != 0)
|
|
status = (oitem->flags_extended & GIT_IDXENTRY_INTENT_TO_ADD) ?
|
|
GIT_DELTA_MODIFIED : GIT_DELTA_UNMODIFIED;
|
|
|
|
/* support "skip worktree" index bit */
|
|
else if ((oitem->flags_extended & GIT_IDXENTRY_SKIP_WORKTREE) != 0)
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
|
|
/* if basic type of file changed, then split into delete and add */
|
|
else if (GIT_MODE_TYPE(omode) != GIT_MODE_TYPE(nmode)) {
|
|
if ((diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE) != 0)
|
|
status = GIT_DELTA_TYPECHANGE;
|
|
else {
|
|
if (diff_delta__from_one(diff, GIT_DELTA_DELETED, oitem) < 0 ||
|
|
diff_delta__from_one(diff, GIT_DELTA_ADDED, nitem) < 0)
|
|
return -1;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/* if oids and modes match, then file is unmodified */
|
|
else if (git_oid_equal(&oitem->oid, &nitem->oid) && omode == nmode)
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
|
|
/* if we have an unknown OID and a workdir iterator, then check some
|
|
* circumstances that can accelerate things or need special handling
|
|
*/
|
|
else if (git_oid_iszero(&nitem->oid) && new_is_workdir) {
|
|
/* TODO: add check against index file st_mtime to avoid racy-git */
|
|
|
|
/* if the stat data looks exactly alike, then assume the same */
|
|
if (omode == nmode &&
|
|
oitem->file_size == nitem->file_size &&
|
|
(!(diff->diffcaps & GIT_DIFFCAPS_TRUST_CTIME) ||
|
|
(oitem->ctime.seconds == nitem->ctime.seconds)) &&
|
|
oitem->mtime.seconds == nitem->mtime.seconds &&
|
|
(!(diff->diffcaps & GIT_DIFFCAPS_USE_DEV) ||
|
|
(oitem->dev == nitem->dev)) &&
|
|
oitem->ino == nitem->ino &&
|
|
oitem->uid == nitem->uid &&
|
|
oitem->gid == nitem->gid)
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
|
|
else if (S_ISGITLINK(nmode)) {
|
|
git_submodule *sub;
|
|
|
|
if ((diff->opts.flags & GIT_DIFF_IGNORE_SUBMODULES) != 0)
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
else if (git_submodule_lookup(&sub, diff->repo, nitem->path) < 0)
|
|
return -1;
|
|
else if (git_submodule_ignore(sub) == GIT_SUBMODULE_IGNORE_ALL)
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
else {
|
|
unsigned int sm_status = 0;
|
|
if (git_submodule_status(&sm_status, sub) < 0)
|
|
return -1;
|
|
status = GIT_SUBMODULE_STATUS_IS_UNMODIFIED(sm_status)
|
|
? GIT_DELTA_UNMODIFIED : GIT_DELTA_MODIFIED;
|
|
|
|
/* grab OID while we are here */
|
|
if (git_oid_iszero(&nitem->oid)) {
|
|
const git_oid *sm_oid = git_submodule_wd_oid(sub);
|
|
if (sm_oid != NULL) {
|
|
git_oid_cpy(&noid, sm_oid);
|
|
use_noid = &noid;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* if we got here and decided that the files are modified, but we
|
|
* haven't calculated the OID of the new item, then calculate it now
|
|
*/
|
|
if (status != GIT_DELTA_UNMODIFIED && git_oid_iszero(&nitem->oid)) {
|
|
if (!use_noid) {
|
|
if (git_diff__oid_for_file(diff->repo,
|
|
nitem->path, nitem->mode, nitem->file_size, &noid) < 0)
|
|
return -1;
|
|
use_noid = &noid;
|
|
}
|
|
if (omode == nmode && git_oid_equal(&oitem->oid, use_noid))
|
|
status = GIT_DELTA_UNMODIFIED;
|
|
}
|
|
|
|
return diff_delta__from_two(
|
|
diff, status, oitem, omode, nitem, nmode, use_noid);
|
|
}
|
|
|
|
static bool entry_is_prefixed(
|
|
git_diff_list *diff,
|
|
const git_index_entry *item,
|
|
const git_index_entry *prefix_item)
|
|
{
|
|
size_t pathlen;
|
|
|
|
if (!prefix_item || diff->pfxcomp(prefix_item->path, item->path))
|
|
return false;
|
|
|
|
pathlen = strlen(item->path);
|
|
|
|
return (item->path[pathlen - 1] == '/' ||
|
|
prefix_item->path[pathlen] == '\0' ||
|
|
prefix_item->path[pathlen] == '/');
|
|
}
|
|
|
|
static int diff_list_init_from_iterators(
|
|
git_diff_list *diff,
|
|
git_iterator *old_iter,
|
|
git_iterator *new_iter)
|
|
{
|
|
diff->old_src = old_iter->type;
|
|
diff->new_src = new_iter->type;
|
|
|
|
/* Use case-insensitive compare if either iterator has
|
|
* the ignore_case bit set */
|
|
if (!old_iter->ignore_case && !new_iter->ignore_case) {
|
|
diff->opts.flags &= ~GIT_DIFF_DELTAS_ARE_ICASE;
|
|
|
|
diff->strcomp = strcmp;
|
|
diff->strncomp = strncmp;
|
|
diff->pfxcomp = git__prefixcmp;
|
|
diff->entrycomp = git_index_entry__cmp;
|
|
} else {
|
|
diff->opts.flags |= GIT_DIFF_DELTAS_ARE_ICASE;
|
|
|
|
diff->strcomp = strcasecmp;
|
|
diff->strncomp = strncasecmp;
|
|
diff->pfxcomp = git__prefixcmp_icase;
|
|
diff->entrycomp = git_index_entry__cmp_icase;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int diff_from_iterators(
|
|
git_diff_list **diff_ptr,
|
|
git_repository *repo,
|
|
git_iterator *old_iter,
|
|
git_iterator *new_iter,
|
|
const git_diff_options *opts)
|
|
{
|
|
int error = 0;
|
|
const git_index_entry *oitem, *nitem;
|
|
git_buf ignore_prefix = GIT_BUF_INIT;
|
|
git_diff_list *diff = git_diff_list_alloc(repo, opts);
|
|
|
|
*diff_ptr = NULL;
|
|
|
|
if (!diff ||
|
|
diff_list_init_from_iterators(diff, old_iter, new_iter) < 0)
|
|
goto fail;
|
|
|
|
if (diff->opts.flags & GIT_DIFF_DELTAS_ARE_ICASE) {
|
|
/* If one of the iterators doesn't have ignore_case set,
|
|
* then that's unfortunate because we'll have to spool
|
|
* its data, sort it icase, and then use that for our
|
|
* merge join to the other iterator that is icase sorted */
|
|
if (!old_iter->ignore_case &&
|
|
git_iterator_spoolandsort(
|
|
&old_iter, old_iter, diff->entrycomp, true) < 0)
|
|
goto fail;
|
|
|
|
if (!new_iter->ignore_case &&
|
|
git_iterator_spoolandsort(
|
|
&new_iter, new_iter, diff->entrycomp, true) < 0)
|
|
goto fail;
|
|
}
|
|
|
|
if (git_iterator_current(old_iter, &oitem) < 0 ||
|
|
git_iterator_current(new_iter, &nitem) < 0)
|
|
goto fail;
|
|
|
|
/* run iterators building diffs */
|
|
while (oitem || nitem) {
|
|
|
|
/* create DELETED records for old items not matched in new */
|
|
if (oitem && (!nitem || diff->entrycomp(oitem, nitem) < 0)) {
|
|
if (diff_delta__from_one(diff, GIT_DELTA_DELETED, oitem) < 0)
|
|
goto fail;
|
|
|
|
/* if we are generating TYPECHANGE records then check for that
|
|
* instead of just generating a DELETE record
|
|
*/
|
|
if ((diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE_TREES) != 0 &&
|
|
entry_is_prefixed(diff, oitem, nitem))
|
|
{
|
|
/* this entry has become a tree! convert to TYPECHANGE */
|
|
git_diff_delta *last = diff_delta__last_for_item(diff, oitem);
|
|
if (last) {
|
|
last->status = GIT_DELTA_TYPECHANGE;
|
|
last->new_file.mode = GIT_FILEMODE_TREE;
|
|
}
|
|
}
|
|
|
|
if (git_iterator_advance(old_iter, &oitem) < 0)
|
|
goto fail;
|
|
}
|
|
|
|
/* create ADDED, TRACKED, or IGNORED records for new items not
|
|
* matched in old (and/or descend into directories as needed)
|
|
*/
|
|
else if (nitem && (!oitem || diff->entrycomp(oitem, nitem) > 0)) {
|
|
git_delta_t delta_type = GIT_DELTA_UNTRACKED;
|
|
|
|
/* check if contained in ignored parent directory */
|
|
if (git_buf_len(&ignore_prefix) &&
|
|
diff->pfxcomp(nitem->path, git_buf_cstr(&ignore_prefix)) == 0)
|
|
delta_type = GIT_DELTA_IGNORED;
|
|
|
|
if (S_ISDIR(nitem->mode)) {
|
|
/* recurse into directory only if there are tracked items in
|
|
* it or if the user requested the contents of untracked
|
|
* directories and it is not under an ignored directory.
|
|
*/
|
|
bool contains_tracked =
|
|
entry_is_prefixed(diff, nitem, oitem);
|
|
bool recurse_untracked =
|
|
(delta_type == GIT_DELTA_UNTRACKED &&
|
|
(diff->opts.flags & GIT_DIFF_RECURSE_UNTRACKED_DIRS) != 0);
|
|
|
|
/* do not advance into directories that contain a .git file */
|
|
if (!contains_tracked && recurse_untracked) {
|
|
git_buf *full = NULL;
|
|
if (git_iterator_current_workdir_path(new_iter, &full) < 0)
|
|
goto fail;
|
|
if (git_path_contains_dir(full, DOT_GIT))
|
|
recurse_untracked = false;
|
|
}
|
|
|
|
if (contains_tracked || recurse_untracked) {
|
|
/* if this directory is ignored, remember it as the
|
|
* "ignore_prefix" for processing contained items
|
|
*/
|
|
if (delta_type == GIT_DELTA_UNTRACKED &&
|
|
git_iterator_current_is_ignored(new_iter))
|
|
git_buf_sets(&ignore_prefix, nitem->path);
|
|
|
|
if (git_iterator_advance_into_directory(new_iter, &nitem) < 0)
|
|
goto fail;
|
|
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* In core git, the next two "else if" clauses are effectively
|
|
* reversed -- i.e. when an untracked file contained in an
|
|
* ignored directory is individually ignored, it shows up as an
|
|
* ignored file in the diff list, even though other untracked
|
|
* files in the same directory are skipped completely.
|
|
*
|
|
* To me, this is odd. If the directory is ignored and the file
|
|
* is untracked, we should skip it consistently, regardless of
|
|
* whether it happens to match a pattern in the ignore file.
|
|
*
|
|
* To match the core git behavior, just reverse the following
|
|
* two "else if" cases so that individual file ignores are
|
|
* checked before container directory exclusions are used to
|
|
* skip the file.
|
|
*/
|
|
else if (delta_type == GIT_DELTA_IGNORED) {
|
|
if (git_iterator_advance(new_iter, &nitem) < 0)
|
|
goto fail;
|
|
continue; /* ignored parent directory, so skip completely */
|
|
}
|
|
|
|
else if (git_iterator_current_is_ignored(new_iter))
|
|
delta_type = GIT_DELTA_IGNORED;
|
|
|
|
else if (new_iter->type != GIT_ITERATOR_WORKDIR)
|
|
delta_type = GIT_DELTA_ADDED;
|
|
|
|
if (diff_delta__from_one(diff, delta_type, nitem) < 0)
|
|
goto fail;
|
|
|
|
/* if we are generating TYPECHANGE records then check for that
|
|
* instead of just generating an ADD/UNTRACKED record
|
|
*/
|
|
if (delta_type != GIT_DELTA_IGNORED &&
|
|
(diff->opts.flags & GIT_DIFF_INCLUDE_TYPECHANGE_TREES) != 0 &&
|
|
entry_is_prefixed(diff, nitem, oitem))
|
|
{
|
|
/* this entry was a tree! convert to TYPECHANGE */
|
|
git_diff_delta *last = diff_delta__last_for_item(diff, oitem);
|
|
if (last) {
|
|
last->status = GIT_DELTA_TYPECHANGE;
|
|
last->old_file.mode = GIT_FILEMODE_TREE;
|
|
}
|
|
}
|
|
|
|
if (git_iterator_advance(new_iter, &nitem) < 0)
|
|
goto fail;
|
|
}
|
|
|
|
/* otherwise item paths match, so create MODIFIED record
|
|
* (or ADDED and DELETED pair if type changed)
|
|
*/
|
|
else {
|
|
assert(oitem && nitem && diff->entrycomp(oitem, nitem) == 0);
|
|
|
|
if (maybe_modified(old_iter, oitem, new_iter, nitem, diff) < 0 ||
|
|
git_iterator_advance(old_iter, &oitem) < 0 ||
|
|
git_iterator_advance(new_iter, &nitem) < 0)
|
|
goto fail;
|
|
}
|
|
}
|
|
|
|
*diff_ptr = diff;
|
|
|
|
fail:
|
|
if (!*diff_ptr) {
|
|
git_diff_list_free(diff);
|
|
error = -1;
|
|
}
|
|
|
|
git_buf_free(&ignore_prefix);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
#define DIFF_FROM_ITERATORS(MAKE_FIRST, MAKE_SECOND) do { \
|
|
git_iterator *a = NULL, *b = NULL; \
|
|
char *pfx = opts ? git_pathspec_prefix(&opts->pathspec) : NULL; \
|
|
if (!(error = MAKE_FIRST) && !(error = MAKE_SECOND)) \
|
|
error = diff_from_iterators(diff, repo, a, b, opts); \
|
|
git__free(pfx); git_iterator_free(a); git_iterator_free(b); \
|
|
} while (0)
|
|
|
|
int git_diff_tree_to_tree(
|
|
git_diff_list **diff,
|
|
git_repository *repo,
|
|
git_tree *old_tree,
|
|
git_tree *new_tree,
|
|
const git_diff_options *opts)
|
|
{
|
|
int error = 0;
|
|
|
|
assert(diff && repo);
|
|
|
|
DIFF_FROM_ITERATORS(
|
|
git_iterator_for_tree_range(&a, repo, old_tree, pfx, pfx),
|
|
git_iterator_for_tree_range(&b, repo, new_tree, pfx, pfx)
|
|
);
|
|
|
|
return error;
|
|
}
|
|
|
|
int git_diff_index_to_tree(
|
|
git_diff_list **diff,
|
|
git_repository *repo,
|
|
git_tree *old_tree,
|
|
git_index *index,
|
|
const git_diff_options *opts)
|
|
{
|
|
int error = 0;
|
|
|
|
assert(diff && repo);
|
|
|
|
if (!index && (error = git_repository_index__weakptr(&index, repo)) < 0)
|
|
return error;
|
|
|
|
DIFF_FROM_ITERATORS(
|
|
git_iterator_for_tree_range(&a, repo, old_tree, pfx, pfx),
|
|
git_iterator_for_index_range(&b, index, pfx, pfx)
|
|
);
|
|
|
|
return error;
|
|
}
|
|
|
|
int git_diff_workdir_to_index(
|
|
git_diff_list **diff,
|
|
git_repository *repo,
|
|
git_index *index,
|
|
const git_diff_options *opts)
|
|
{
|
|
int error = 0;
|
|
|
|
assert(diff && repo);
|
|
|
|
if (!index && (error = git_repository_index__weakptr(&index, repo)) < 0)
|
|
return error;
|
|
|
|
DIFF_FROM_ITERATORS(
|
|
git_iterator_for_index_range(&a, index, pfx, pfx),
|
|
git_iterator_for_workdir_range(&b, repo, pfx, pfx)
|
|
);
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
int git_diff_workdir_to_tree(
|
|
git_diff_list **diff,
|
|
git_repository *repo,
|
|
git_tree *old_tree,
|
|
const git_diff_options *opts)
|
|
{
|
|
int error = 0;
|
|
|
|
assert(diff && repo);
|
|
|
|
DIFF_FROM_ITERATORS(
|
|
git_iterator_for_tree_range(&a, repo, old_tree, pfx, pfx),
|
|
git_iterator_for_workdir_range(&b, repo, pfx, pfx)
|
|
);
|
|
|
|
return error;
|
|
}
|