Merge branch 'jc/maint-split-diff-metainfo' into maint

* jc/maint-split-diff-metainfo:
  diff.c: output correct index lines for a split diff
This commit is contained in:
Junio C Hamano 2009-02-05 17:54:17 -08:00
commit cc91e1bd05
3 changed files with 99 additions and 67 deletions

146
diff.c
View File

@ -1999,16 +1999,86 @@ static void run_external_diff(const char *pgm,
} }
} }
static int similarity_index(struct diff_filepair *p)
{
return p->score * 100 / MAX_SCORE;
}
static void fill_metainfo(struct strbuf *msg,
const char *name,
const char *other,
struct diff_filespec *one,
struct diff_filespec *two,
struct diff_options *o,
struct diff_filepair *p)
{
strbuf_init(msg, PATH_MAX * 2 + 300);
switch (p->status) {
case DIFF_STATUS_COPIED:
strbuf_addf(msg, "similarity index %d%%", similarity_index(p));
strbuf_addstr(msg, "\ncopy from ");
quote_c_style(name, msg, NULL, 0);
strbuf_addstr(msg, "\ncopy to ");
quote_c_style(other, msg, NULL, 0);
strbuf_addch(msg, '\n');
break;
case DIFF_STATUS_RENAMED:
strbuf_addf(msg, "similarity index %d%%", similarity_index(p));
strbuf_addstr(msg, "\nrename from ");
quote_c_style(name, msg, NULL, 0);
strbuf_addstr(msg, "\nrename to ");
quote_c_style(other, msg, NULL, 0);
strbuf_addch(msg, '\n');
break;
case DIFF_STATUS_MODIFIED:
if (p->score) {
strbuf_addf(msg, "dissimilarity index %d%%\n",
similarity_index(p));
break;
}
/* fallthru */
default:
/* nothing */
;
}
if (one && two && hashcmp(one->sha1, two->sha1)) {
int abbrev = DIFF_OPT_TST(o, FULL_INDEX) ? 40 : DEFAULT_ABBREV;
if (DIFF_OPT_TST(o, BINARY)) {
mmfile_t mf;
if ((!fill_mmfile(&mf, one) && diff_filespec_is_binary(one)) ||
(!fill_mmfile(&mf, two) && diff_filespec_is_binary(two)))
abbrev = 40;
}
strbuf_addf(msg, "index %.*s..%.*s",
abbrev, sha1_to_hex(one->sha1),
abbrev, sha1_to_hex(two->sha1));
if (one->mode == two->mode)
strbuf_addf(msg, " %06o", one->mode);
strbuf_addch(msg, '\n');
}
if (msg->len)
strbuf_setlen(msg, msg->len - 1);
}
static void run_diff_cmd(const char *pgm, static void run_diff_cmd(const char *pgm,
const char *name, const char *name,
const char *other, const char *other,
const char *attr_path, const char *attr_path,
struct diff_filespec *one, struct diff_filespec *one,
struct diff_filespec *two, struct diff_filespec *two,
const char *xfrm_msg, struct strbuf *msg,
struct diff_options *o, struct diff_options *o,
int complete_rewrite) struct diff_filepair *p)
{ {
const char *xfrm_msg = NULL;
int complete_rewrite = (p->status == DIFF_STATUS_MODIFIED) && p->score;
if (msg) {
fill_metainfo(msg, name, other, one, two, o, p);
xfrm_msg = msg->len ? msg->buf : NULL;
}
if (!DIFF_OPT_TST(o, ALLOW_EXTERNAL)) if (!DIFF_OPT_TST(o, ALLOW_EXTERNAL))
pgm = NULL; pgm = NULL;
else { else {
@ -2048,11 +2118,6 @@ static void diff_fill_sha1_info(struct diff_filespec *one)
hashclr(one->sha1); hashclr(one->sha1);
} }
static int similarity_index(struct diff_filepair *p)
{
return p->score * 100 / MAX_SCORE;
}
static void strip_prefix(int prefix_length, const char **namep, const char **otherp) static void strip_prefix(int prefix_length, const char **namep, const char **otherp)
{ {
/* Strip the prefix but do not molest /dev/null and absolute paths */ /* Strip the prefix but do not molest /dev/null and absolute paths */
@ -2066,13 +2131,11 @@ static void run_diff(struct diff_filepair *p, struct diff_options *o)
{ {
const char *pgm = external_diff(); const char *pgm = external_diff();
struct strbuf msg; struct strbuf msg;
char *xfrm_msg;
struct diff_filespec *one = p->one; struct diff_filespec *one = p->one;
struct diff_filespec *two = p->two; struct diff_filespec *two = p->two;
const char *name; const char *name;
const char *other; const char *other;
const char *attr_path; const char *attr_path;
int complete_rewrite = 0;
name = p->one->path; name = p->one->path;
other = (strcmp(name, p->two->path) ? p->two->path : NULL); other = (strcmp(name, p->two->path) ? p->two->path : NULL);
@ -2082,83 +2145,34 @@ static void run_diff(struct diff_filepair *p, struct diff_options *o)
if (DIFF_PAIR_UNMERGED(p)) { if (DIFF_PAIR_UNMERGED(p)) {
run_diff_cmd(pgm, name, NULL, attr_path, run_diff_cmd(pgm, name, NULL, attr_path,
NULL, NULL, NULL, o, 0); NULL, NULL, NULL, o, p);
return; return;
} }
diff_fill_sha1_info(one); diff_fill_sha1_info(one);
diff_fill_sha1_info(two); diff_fill_sha1_info(two);
strbuf_init(&msg, PATH_MAX * 2 + 300);
switch (p->status) {
case DIFF_STATUS_COPIED:
strbuf_addf(&msg, "similarity index %d%%", similarity_index(p));
strbuf_addstr(&msg, "\ncopy from ");
quote_c_style(name, &msg, NULL, 0);
strbuf_addstr(&msg, "\ncopy to ");
quote_c_style(other, &msg, NULL, 0);
strbuf_addch(&msg, '\n');
break;
case DIFF_STATUS_RENAMED:
strbuf_addf(&msg, "similarity index %d%%", similarity_index(p));
strbuf_addstr(&msg, "\nrename from ");
quote_c_style(name, &msg, NULL, 0);
strbuf_addstr(&msg, "\nrename to ");
quote_c_style(other, &msg, NULL, 0);
strbuf_addch(&msg, '\n');
break;
case DIFF_STATUS_MODIFIED:
if (p->score) {
strbuf_addf(&msg, "dissimilarity index %d%%\n",
similarity_index(p));
complete_rewrite = 1;
break;
}
/* fallthru */
default:
/* nothing */
;
}
if (hashcmp(one->sha1, two->sha1)) {
int abbrev = DIFF_OPT_TST(o, FULL_INDEX) ? 40 : DEFAULT_ABBREV;
if (DIFF_OPT_TST(o, BINARY)) {
mmfile_t mf;
if ((!fill_mmfile(&mf, one) && diff_filespec_is_binary(one)) ||
(!fill_mmfile(&mf, two) && diff_filespec_is_binary(two)))
abbrev = 40;
}
strbuf_addf(&msg, "index %.*s..%.*s",
abbrev, sha1_to_hex(one->sha1),
abbrev, sha1_to_hex(two->sha1));
if (one->mode == two->mode)
strbuf_addf(&msg, " %06o", one->mode);
strbuf_addch(&msg, '\n');
}
if (msg.len)
strbuf_setlen(&msg, msg.len - 1);
xfrm_msg = msg.len ? msg.buf : NULL;
if (!pgm && if (!pgm &&
DIFF_FILE_VALID(one) && DIFF_FILE_VALID(two) && DIFF_FILE_VALID(one) && DIFF_FILE_VALID(two) &&
(S_IFMT & one->mode) != (S_IFMT & two->mode)) { (S_IFMT & one->mode) != (S_IFMT & two->mode)) {
/* a filepair that changes between file and symlink /*
* a filepair that changes between file and symlink
* needs to be split into deletion and creation. * needs to be split into deletion and creation.
*/ */
struct diff_filespec *null = alloc_filespec(two->path); struct diff_filespec *null = alloc_filespec(two->path);
run_diff_cmd(NULL, name, other, attr_path, run_diff_cmd(NULL, name, other, attr_path,
one, null, xfrm_msg, o, 0); one, null, &msg, o, p);
free(null); free(null);
strbuf_release(&msg);
null = alloc_filespec(one->path); null = alloc_filespec(one->path);
run_diff_cmd(NULL, name, other, attr_path, run_diff_cmd(NULL, name, other, attr_path,
null, two, xfrm_msg, o, 0); null, two, &msg, o, p);
free(null); free(null);
} }
else else
run_diff_cmd(pgm, name, other, attr_path, run_diff_cmd(pgm, name, other, attr_path,
one, two, xfrm_msg, o, complete_rewrite); one, two, &msg, o, p);
strbuf_release(&msg); strbuf_release(&msg);
} }

View File

@ -104,7 +104,7 @@ cat >expect.typechange <<'EOF'
-1 -1
diff --git a/file b/file diff --git a/file b/file
new file mode 120000 new file mode 120000
index ad8b3d2..67be421 index 0000000..67be421
--- /dev/null --- /dev/null
+++ b/file +++ b/file
@@ -0,0 +1 @@ @@ -0,0 +1 @@

View File

@ -25,6 +25,10 @@ test_expect_success 'setup repository and commits' '
git update-index foo && git update-index foo &&
git commit -m "foo back to file" && git commit -m "foo back to file" &&
git branch foo-back-to-file && git branch foo-back-to-file &&
printf "\0" > foo &&
git update-index foo &&
git commit -m "foo becomes binary" &&
git branch foo-becomes-binary &&
rm -f foo && rm -f foo &&
git update-index --remove foo && git update-index --remove foo &&
mkdir foo && mkdir foo &&
@ -85,6 +89,20 @@ test_expect_success 'symlink becomes file' '
' '
test_debug 'cat patch' test_debug 'cat patch'
test_expect_success 'binary file becomes symlink' '
git checkout -f foo-becomes-binary &&
git diff-tree -p --binary HEAD foo-symlinked-to-bar > patch &&
git apply --index < patch
'
test_debug 'cat patch'
test_expect_success 'symlink becomes binary file' '
git checkout -f foo-symlinked-to-bar &&
git diff-tree -p --binary HEAD foo-becomes-binary > patch &&
git apply --index < patch
'
test_debug 'cat patch'
test_expect_success 'symlink becomes directory' ' test_expect_success 'symlink becomes directory' '
git checkout -f foo-symlinked-to-bar && git checkout -f foo-symlinked-to-bar &&