1 #include "clar_libgit2.h"
2 #include "git2/repository.h"
3 #include "git2/merge.h"
7 #include "../merge_helpers.h"
8 #include "../conflict_data.h"
10 static git_repository
*repo
;
12 #define TEST_REPO_PATH "merge-resolve"
13 #define TEST_INDEX_PATH TEST_REPO_PATH "/.git/index"
15 #define THEIRS_AUTOMERGE_BRANCH "branch"
17 #define THEIRS_UNRELATED_BRANCH "unrelated"
18 #define THEIRS_UNRELATED_OID "55b4e4687e7a0d9ca367016ed930f385d4022e6f"
19 #define THEIRS_UNRELATED_PARENT "d6cf6c7741b3316826af1314042550c97ded1d50"
21 #define OURS_DIRECTORY_FILE "df_side1"
22 #define THEIRS_DIRECTORY_FILE "df_side2"
24 /* Non-conflicting files, index entries are common to every merge operation */
25 #define ADDED_IN_MASTER_INDEX_ENTRY \
26 { 0100644, "233c0919c998ed110a4b6ff36f353aec8b713487", 0, "added-in-master.txt" }
27 #define AUTOMERGEABLE_INDEX_ENTRY \
28 { 0100644, "f2e1550a0c9e53d5811175864a29536642ae3821", 0, "automergeable.txt" }
29 #define CHANGED_IN_BRANCH_INDEX_ENTRY \
30 { 0100644, "4eb04c9e79e88f6640d01ff5b25ca2a60764f216", 0, "changed-in-branch.txt" }
31 #define CHANGED_IN_MASTER_INDEX_ENTRY \
32 { 0100644, "11deab00b2d3a6f5a3073988ac050c2d7b6655e2", 0, "changed-in-master.txt" }
33 #define UNCHANGED_INDEX_ENTRY \
34 { 0100644, "c8f06f2e3bb2964174677e91f0abead0e43c9e5d", 0, "unchanged.txt" }
36 /* Expected REUC entries */
37 #define AUTOMERGEABLE_REUC_ENTRY \
38 { "automergeable.txt", 0100644, 0100644, 0100644, \
39 "6212c31dab5e482247d7977e4f0dd3601decf13b", \
40 "ee3fa1b8c00aff7fe02065fdb50864bb0d932ccf", \
41 "058541fc37114bfc1dddf6bd6bffc7fae5c2e6fe" }
42 #define CONFLICTING_REUC_ENTRY \
43 { "conflicting.txt", 0100644, 0100644, 0100644, \
44 "d427e0b2e138501a3d15cc376077a3631e15bd46", \
45 "4e886e602529caa9ab11d71f86634bd1b6e0de10", \
46 "2bd0a343aeef7a2cf0d158478966a6e587ff3863" }
47 #define REMOVED_IN_BRANCH_REUC_ENTRY \
48 { "removed-in-branch.txt", 0100644, 0100644, 0, \
49 "dfe3f22baa1f6fce5447901c3086bae368de6bdd", \
50 "dfe3f22baa1f6fce5447901c3086bae368de6bdd", \
52 #define REMOVED_IN_MASTER_REUC_ENTRY \
53 { "removed-in-master.txt", 0100644, 0, 0100644, \
54 "5c3b68a71fc4fa5d362fd3875e53137c6a5ab7a5", \
56 "5c3b68a71fc4fa5d362fd3875e53137c6a5ab7a5" }
58 /* Fixture setup and teardown */
59 void test_merge_trees_automerge__initialize(void)
61 repo
= cl_git_sandbox_init(TEST_REPO_PATH
);
64 void test_merge_trees_automerge__cleanup(void)
66 cl_git_sandbox_cleanup();
69 void test_merge_trees_automerge__automerge(void)
72 const git_index_entry
*entry
;
73 git_merge_options opts
= GIT_MERGE_OPTIONS_INIT
;
76 struct merge_index_entry merge_index_entries
[] = {
77 ADDED_IN_MASTER_INDEX_ENTRY
,
78 AUTOMERGEABLE_INDEX_ENTRY
,
79 CHANGED_IN_BRANCH_INDEX_ENTRY
,
80 CHANGED_IN_MASTER_INDEX_ENTRY
,
82 { 0100644, "d427e0b2e138501a3d15cc376077a3631e15bd46", 1, "conflicting.txt" },
83 { 0100644, "4e886e602529caa9ab11d71f86634bd1b6e0de10", 2, "conflicting.txt" },
84 { 0100644, "2bd0a343aeef7a2cf0d158478966a6e587ff3863", 3, "conflicting.txt" },
86 UNCHANGED_INDEX_ENTRY
,
89 struct merge_reuc_entry merge_reuc_entries
[] = {
90 AUTOMERGEABLE_REUC_ENTRY
,
91 REMOVED_IN_BRANCH_REUC_ENTRY
,
92 REMOVED_IN_MASTER_REUC_ENTRY
95 cl_git_pass(merge_trees_from_branches(&index
, repo
, "master", THEIRS_AUTOMERGE_BRANCH
, &opts
));
97 cl_assert(merge_test_index(index
, merge_index_entries
, 8));
98 cl_assert(merge_test_reuc(index
, merge_reuc_entries
, 3));
100 cl_assert((entry
= git_index_get_bypath(index
, "automergeable.txt", 0)) != NULL
);
101 cl_assert(entry
->file_size
== strlen(AUTOMERGEABLE_MERGED_FILE
));
103 cl_git_pass(git_object_lookup((git_object
**)&blob
, repo
, &entry
->id
, GIT_OBJECT_BLOB
));
104 cl_assert(memcmp(git_blob_rawcontent(blob
), AUTOMERGEABLE_MERGED_FILE
, (size_t)entry
->file_size
) == 0);
106 git_index_free(index
);
110 void test_merge_trees_automerge__favor_ours(void)
113 git_merge_options opts
= GIT_MERGE_OPTIONS_INIT
;
115 struct merge_index_entry merge_index_entries
[] = {
116 ADDED_IN_MASTER_INDEX_ENTRY
,
117 AUTOMERGEABLE_INDEX_ENTRY
,
118 CHANGED_IN_BRANCH_INDEX_ENTRY
,
119 CHANGED_IN_MASTER_INDEX_ENTRY
,
120 { 0100644, "4e886e602529caa9ab11d71f86634bd1b6e0de10", 0, "conflicting.txt" },
121 UNCHANGED_INDEX_ENTRY
,
124 struct merge_reuc_entry merge_reuc_entries
[] = {
125 AUTOMERGEABLE_REUC_ENTRY
,
126 CONFLICTING_REUC_ENTRY
,
127 REMOVED_IN_BRANCH_REUC_ENTRY
,
128 REMOVED_IN_MASTER_REUC_ENTRY
,
131 opts
.file_favor
= GIT_MERGE_FILE_FAVOR_OURS
;
133 cl_git_pass(merge_trees_from_branches(&index
, repo
, "master", THEIRS_AUTOMERGE_BRANCH
, &opts
));
135 cl_assert(merge_test_index(index
, merge_index_entries
, 6));
136 cl_assert(merge_test_reuc(index
, merge_reuc_entries
, 4));
138 git_index_free(index
);
141 void test_merge_trees_automerge__favor_theirs(void)
144 git_merge_options opts
= GIT_MERGE_OPTIONS_INIT
;
146 struct merge_index_entry merge_index_entries
[] = {
147 ADDED_IN_MASTER_INDEX_ENTRY
,
148 AUTOMERGEABLE_INDEX_ENTRY
,
149 CHANGED_IN_BRANCH_INDEX_ENTRY
,
150 CHANGED_IN_MASTER_INDEX_ENTRY
,
151 { 0100644, "2bd0a343aeef7a2cf0d158478966a6e587ff3863", 0, "conflicting.txt" },
152 UNCHANGED_INDEX_ENTRY
,
155 struct merge_reuc_entry merge_reuc_entries
[] = {
156 AUTOMERGEABLE_REUC_ENTRY
,
157 CONFLICTING_REUC_ENTRY
,
158 REMOVED_IN_BRANCH_REUC_ENTRY
,
159 REMOVED_IN_MASTER_REUC_ENTRY
,
162 opts
.file_favor
= GIT_MERGE_FILE_FAVOR_THEIRS
;
164 cl_git_pass(merge_trees_from_branches(&index
, repo
, "master", THEIRS_AUTOMERGE_BRANCH
, &opts
));
166 cl_assert(merge_test_index(index
, merge_index_entries
, 6));
167 cl_assert(merge_test_reuc(index
, merge_reuc_entries
, 4));
169 git_index_free(index
);
172 void test_merge_trees_automerge__unrelated(void)
175 git_merge_options opts
= GIT_MERGE_OPTIONS_INIT
;
177 struct merge_index_entry merge_index_entries
[] = {
178 { 0100644, "233c0919c998ed110a4b6ff36f353aec8b713487", 0, "added-in-master.txt" },
179 { 0100644, "ee3fa1b8c00aff7fe02065fdb50864bb0d932ccf", 2, "automergeable.txt" },
180 { 0100644, "d07ec190c306ec690bac349e87d01c4358e49bb2", 3, "automergeable.txt" },
181 { 0100644, "ab6c44a2e84492ad4b41bb6bac87353e9d02ac8b", 0, "changed-in-branch.txt" },
182 { 0100644, "11deab00b2d3a6f5a3073988ac050c2d7b6655e2", 0, "changed-in-master.txt" },
183 { 0100644, "4e886e602529caa9ab11d71f86634bd1b6e0de10", 2, "conflicting.txt" },
184 { 0100644, "4b253da36a0ae8bfce63aeabd8c5b58429925594", 3, "conflicting.txt" },
185 { 0100644, "ef58fdd8086c243bdc81f99e379acacfd21d32d6", 0, "new-in-unrelated1.txt" },
186 { 0100644, "948ba6e701c1edab0c2d394fb7c5538334129793", 0, "new-in-unrelated2.txt" },
187 { 0100644, "dfe3f22baa1f6fce5447901c3086bae368de6bdd", 0, "removed-in-branch.txt" },
188 { 0100644, "c8f06f2e3bb2964174677e91f0abead0e43c9e5d", 0, "unchanged.txt" },
191 cl_git_pass(merge_trees_from_branches(&index
, repo
, "master", THEIRS_UNRELATED_BRANCH
, &opts
));
193 cl_assert(merge_test_index(index
, merge_index_entries
, 11));
195 git_index_free(index
);