return git_buf_printf(out, template, oldpath, newpath);
}
-int diff_delta_format_rename_header(
+int diff_delta_format_similarity_header(
git_buf *out,
const git_diff_delta *delta)
{
git_buf old_path = GIT_BUF_INIT, new_path = GIT_BUF_INIT;
+ const char *type;
int error = 0;
if (delta->similarity > 100) {
goto done;
}
+ if (delta->status == GIT_DELTA_RENAMED)
+ type = "rename";
+ else if (delta->status == GIT_DELTA_COPIED)
+ type = "copy";
+ else
+ abort();
+
if ((error = git_buf_puts(&old_path, delta->old_file.path)) < 0 ||
(error = git_buf_puts(&new_path, delta->new_file.path)) < 0 ||
(error = git_buf_quote(&old_path)) < 0 ||
git_buf_printf(out,
"similarity index %d%%\n"
- "rename from %s\n"
- "rename to %s\n",
+ "%s from %s\n"
+ "%s to %s\n",
delta->similarity,
- old_path.ptr,
- new_path.ptr);
+ type, old_path.ptr,
+ type, new_path.ptr);
if (git_buf_oom(out))
error = -1;
return error;
}
+static bool delta_is_unchanged(const git_diff_delta *delta)
+{
+ if (git_oid_iszero(&delta->old_file.id) &&
+ git_oid_iszero(&delta->new_file.id))
+ return true;
+
+ if (delta->old_file.mode == GIT_FILEMODE_COMMIT ||
+ delta->new_file.mode == GIT_FILEMODE_COMMIT)
+ return false;
+
+ if (git_oid_equal(&delta->old_file.id, &delta->new_file.id))
+ return true;
+
+ return false;
+}
+
int git_diff_delta__format_file_header(
git_buf *out,
const git_diff_delta *delta,
int id_strlen)
{
git_buf old_path = GIT_BUF_INIT, new_path = GIT_BUF_INIT;
- bool unchanged;
+ bool unchanged = delta_is_unchanged(delta);
int error = 0;
if (!oldpfx)
git_buf_printf(out, "diff --git %s %s\n",
old_path.ptr, new_path.ptr);
- if (delta->status == GIT_DELTA_RENAMED) {
- if ((error = diff_delta_format_rename_header(out, delta)) < 0)
+ if (delta->status == GIT_DELTA_RENAMED ||
+ (delta->status == GIT_DELTA_COPIED && unchanged)) {
+ if ((error = diff_delta_format_similarity_header(out, delta)) < 0)
goto done;
}
- unchanged = (git_oid_iszero(&delta->old_file.id) &&
- git_oid_iszero(&delta->new_file.id));
-
if (!unchanged) {
if ((error = diff_print_oid_range(out, delta, id_strlen)) < 0)
goto done;
return parse_header_rename(&patch->rename_new_path, ctx);
}
+static int parse_header_copyfrom(
+ git_patch_parsed *patch, git_patch_parse_ctx *ctx)
+{
+ patch->base.delta->status = GIT_DELTA_COPIED;
+ return parse_header_rename(&patch->rename_old_path, ctx);
+}
+
+static int parse_header_copyto(
+ git_patch_parsed *patch, git_patch_parse_ctx *ctx)
+{
+ patch->base.delta->status = GIT_DELTA_COPIED;
+ return parse_header_rename(&patch->rename_new_path, ctx);
+}
+
static int parse_header_percent(uint16_t *out, git_patch_parse_ctx *ctx)
{
int32_t val;
{ "rename to ", parse_header_renameto },
{ "rename old ", parse_header_renamefrom },
{ "rename new ", parse_header_renameto },
+ { "copy from ", parse_header_copyfrom },
+ { "copy to ", parse_header_copyto },
{ "similarity index ", parse_header_similarity },
{ "dissimilarity index ", parse_header_dissimilarity },
};
git_diff_print_callback__to_file_handle, fp ? fp : stderr));
}
+static size_t num_modified_deltas(git_diff *diff)
+{
+ const git_diff_delta *delta;
+ size_t i, cnt = 0;
+
+ for (i = 0; i < git_diff_num_deltas(diff); i++) {
+ delta = git_diff_get_delta(diff, i);
+
+ if (delta->status != GIT_DELTA_UNMODIFIED)
+ cnt++;
+ }
+
+ return cnt;
+}
+
void diff_assert_equal(git_diff *a, git_diff *b)
{
const git_diff_delta *ad, *bd;
- size_t i;
+ size_t i, j;
assert(a && b);
- cl_assert_equal_i(git_diff_num_deltas(a), git_diff_num_deltas(b));
+ cl_assert_equal_i(num_modified_deltas(a), num_modified_deltas(b));
+
+ for (i = 0, j = 0;
+ i < git_diff_num_deltas(a) && j < git_diff_num_deltas(b); ) {
- for (i = 0; i < git_diff_num_deltas(a); i++) {
ad = git_diff_get_delta(a, i);
- bd = git_diff_get_delta(b, i);
+ bd = git_diff_get_delta(b, j);
+
+ if (ad->status == GIT_DELTA_UNMODIFIED) {
+ i++;
+ continue;
+ }
+ if (bd->status == GIT_DELTA_UNMODIFIED) {
+ j++;
+ continue;
+ }
cl_assert_equal_i(ad->status, bd->status);
cl_assert_equal_i(ad->flags, bd->flags);
* computed deltas will have flags of `VALID_ID` and
* `EXISTS` (parsed deltas will not query the ODB.)
*/
- cl_assert_equal_oid(&ad->old_file.id, &bd->old_file.id);
- cl_assert_equal_i(ad->old_file.id_abbrev, bd->old_file.id_abbrev);
+
+ /* an empty id indicates that it wasn't presented, because
+ * the diff was identical. (eg, pure rename, mode change only, etc)
+ */
+ if (ad->old_file.id_abbrev && bd->old_file.id_abbrev) {
+ cl_assert_equal_i(ad->old_file.id_abbrev, bd->old_file.id_abbrev);
+ cl_assert_equal_oid(&ad->old_file.id, &bd->old_file.id);
+ cl_assert_equal_i(ad->old_file.mode, bd->old_file.mode);
+ }
cl_assert_equal_s(ad->old_file.path, bd->old_file.path);
- cl_assert_equal_i(ad->old_file.mode, bd->old_file.mode);
- cl_assert_equal_oid(&ad->new_file.id, &bd->new_file.id);
- cl_assert_equal_i(ad->new_file.id_abbrev, bd->new_file.id_abbrev);
+ if (ad->new_file.id_abbrev && bd->new_file.id_abbrev) {
+ cl_assert_equal_oid(&ad->new_file.id, &bd->new_file.id);
+ cl_assert_equal_i(ad->new_file.id_abbrev, bd->new_file.id_abbrev);
+ cl_assert_equal_i(ad->new_file.mode, bd->new_file.mode);
+ }
cl_assert_equal_s(ad->new_file.path, bd->new_file.path);
- cl_assert_equal_i(ad->new_file.mode, bd->new_file.mode);
+
+ i++;
+ j++;
}
}
"diff --git a/file1.txt.renamed b/file1.txt.renamed\n" \
"old mode 100644\n" \
"new mode 100755\n" \
- "index a97157a..a97157a\n" \
- "--- a/file1.txt.renamed\n" \
- "+++ b/file1.txt.renamed\n" \
"--\n" \
"libgit2 " LIBGIT2_VERSION "\n" \
"\n";
"31e47d8c1fa36d7f8d537b96158e3f024de0a9f2",
"2bc7f351d20b53f1c72c16c4b036e491c478c49a",
0, GIT_DIFF_FIND_RENAMES);
+ test_tree_to_tree_computed_to_parsed("renames",
+ "31e47d8c1fa36d7f8d537b96158e3f024de0a9f2",
+ "2bc7f351d20b53f1c72c16c4b036e491c478c49a",
+ GIT_DIFF_INCLUDE_UNMODIFIED,
+ 0);
test_tree_to_tree_computed_to_parsed("renames",
"31e47d8c1fa36d7f8d537b96158e3f024de0a9f2",
"2bc7f351d20b53f1c72c16c4b036e491c478c49a",
expect_files_not_renamed("", "\n\n\n\n", GIT_DIFF_FIND_DONT_IGNORE_WHITESPACE);
expect_files_not_renamed("\n\n\n\n", "\r\n\r\n\r\n", GIT_DIFF_FIND_DONT_IGNORE_WHITESPACE);
}
+
+/* test that 100% renames and copies emit the correct patch file
+ * git diff --find-copies-harder -M100 -B100 \
+ * 31e47d8c1fa36d7f8d537b96158e3f024de0a9f2 \
+ * 2bc7f351d20b53f1c72c16c4b036e491c478c49a
+ */
+void test_diff_rename__identical(void)
+{
+ const char *old_sha = "31e47d8c1fa36d7f8d537b96158e3f024de0a9f2";
+ const char *new_sha = "2bc7f351d20b53f1c72c16c4b036e491c478c49a";
+ git_tree *old_tree, *new_tree;
+ git_diff *diff;
+ git_diff_options diff_opts = GIT_DIFF_OPTIONS_INIT;
+ git_diff_find_options find_opts = GIT_DIFF_FIND_OPTIONS_INIT;
+ git_buf diff_buf = GIT_BUF_INIT;
+ const char *expected =
+ "diff --git a/serving.txt b/sixserving.txt\n"
+ "similarity index 100%\n"
+ "rename from serving.txt\n"
+ "rename to sixserving.txt\n"
+ "diff --git a/sevencities.txt b/songofseven.txt\n"
+ "similarity index 100%\n"
+ "copy from sevencities.txt\n"
+ "copy to songofseven.txt\n";
+
+ old_tree = resolve_commit_oid_to_tree(g_repo, old_sha);
+ new_tree = resolve_commit_oid_to_tree(g_repo, new_sha);
+
+ diff_opts.flags |= GIT_DIFF_INCLUDE_UNMODIFIED;
+ find_opts.flags = GIT_DIFF_FIND_COPIES_FROM_UNMODIFIED |
+ GIT_DIFF_FIND_EXACT_MATCH_ONLY;
+
+ cl_git_pass(git_diff_tree_to_tree(&diff,
+ g_repo, old_tree, new_tree, &diff_opts));
+ cl_git_pass(git_diff_find_similar(diff, &find_opts));
+
+ cl_git_pass(git_diff_to_buf(&diff_buf, diff, GIT_DIFF_FORMAT_PATCH));
+
+ cl_assert_equal_s(expected, diff_buf.ptr);
+
+ git_buf_free(&diff_buf);
+ git_diff_free(diff);
+ git_tree_free(old_tree);
+ git_tree_free(new_tree);
+}
+