3 # This program resolves merge conflicts in git
5 # Copyright (c) 2006 Theodore Y. Ts'o
7 # This file is licensed under the GPL v2, or a later version
8 # at the discretion of Junio C Hamano.
11 USAGE
='[--tool=tool] [--tool-help] [-y|--no-prompt|--prompt] [file to merge] ...'
19 # Returns true if the mode reflects a symlink
33 test -n "$remote_mode"
40 cleanup_temp_files
() {
41 if test "$1" = --save-backup
43 rm -rf -- "$MERGED.orig"
44 test -e "$BACKUP" && mv -- "$BACKUP" "$MERGED.orig"
45 rm -f -- "$LOCAL" "$REMOTE" "$BASE"
47 rm -f -- "$LOCAL" "$REMOTE" "$BASE" "$BACKUP"
56 printf " {%s}: " "$branch"
60 elif is_symlink
"$mode"
62 echo "a symbolic link -> '$(cat "$file")'"
63 elif is_submodule
"$mode"
65 echo "submodule commit $file"
74 resolve_symlink_merge
() {
77 printf "Use (l)ocal or (r)emote, or (a)bort? "
81 git checkout-index
-f --stage=2 -- "$MERGED"
83 cleanup_temp_files
--save-backup
87 git checkout-index
-f --stage=3 -- "$MERGED"
89 cleanup_temp_files
--save-backup
99 resolve_deleted_merge
() {
104 printf "Use (m)odified or (d)eleted file, or (a)bort? "
106 printf "Use (c)reated or (d)eleted file, or (a)bort? "
112 cleanup_temp_files
--save-backup
116 git
rm -- "$MERGED" > /dev
/null
127 resolve_submodule_merge
() {
130 printf "Use (l)ocal or (r)emote, or (a)bort? "
136 if test -n "$(git ls-tree HEAD -- "$MERGED")"
138 # Local isn't present, but it's a subdirectory
139 git ls-tree
--full-name -r HEAD
-- "$MERGED" |
140 git update-index
--index-info ||
exit $?
142 test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
143 git update-index
--force-remove "$MERGED"
144 cleanup_temp_files
--save-backup
146 elif is_submodule
"$local_mode"
148 stage_submodule
"$MERGED" "$local_sha1"
150 git checkout-index
-f --stage=2 -- "$MERGED"
158 if test -n "$(git ls-tree MERGE_HEAD -- "$MERGED")"
160 # Remote isn't present, but it's a subdirectory
161 git ls-tree
--full-name -r MERGE_HEAD
-- "$MERGED" |
162 git update-index
--index-info ||
exit $?
164 test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
165 git update-index
--force-remove "$MERGED"
167 elif is_submodule
"$remote_mode"
169 ! is_submodule
"$local_mode" &&
171 mv -- "$MERGED" "$BACKUP"
172 stage_submodule
"$MERGED" "$remote_sha1"
174 test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
175 git checkout-index
-f --stage=3 -- "$MERGED"
178 cleanup_temp_files
--save-backup
192 die
"fatal: unable to create directory for module at $path"
193 # Find $path relative to work tree
194 work_tree_root
=$
(cd_to_toplevel
&& pwd)
195 work_rel_path
=$
(cd "$path" &&
196 GIT_WORK_TREE
="${work_tree_root}" git rev-parse
--show-prefix
198 test -n "$work_rel_path" ||
199 die
"fatal: unable to get path of module $path relative to work tree"
200 git update-index
--add --replace --cacheinfo 160000 "$submodule_sha1" "${work_rel_path%/}" || die
203 checkout_staged_file
() {
205 "$(git checkout-index --temp --stage="$1" "$2" 2>/dev/null)" \
208 if test $?
-eq 0 -a -n "$tmpfile"
210 mv -- "$(git rev-parse --show-cdup)$tmpfile" "$3"
219 f
=$
(git ls-files
-u -- "$MERGED")
222 if test ! -f "$MERGED"
224 echo "$MERGED: file not found"
226 echo "$MERGED: file does not need merging"
231 if BASE
=$
(expr "$MERGED" : '\(.*\)\.[^/]*$')
233 ext
=$
(expr "$MERGED" : '.*\(\.[^/]*\)$')
238 BACKUP
="./${BASE}_BACKUP_$$$ext"
239 LOCAL
="./${BASE}_LOCAL_$$$ext"
240 REMOTE
="./${BASE}_REMOTE_$$$ext"
241 BASE
="./${BASE}_BASE_$$$ext"
243 base_mode
=$
(git ls-files
-u -- "$MERGED" |
awk '{if ($3==1) print $1;}')
244 local_mode
=$
(git ls-files
-u -- "$MERGED" |
awk '{if ($3==2) print $1;}')
245 remote_mode
=$
(git ls-files
-u -- "$MERGED" |
awk '{if ($3==3) print $1;}')
247 if is_submodule
"$local_mode" || is_submodule
"$remote_mode"
249 echo "Submodule merge conflict for '$MERGED':"
250 local_sha1
=$
(git ls-files
-u -- "$MERGED" |
awk '{if ($3==2) print $2;}')
251 remote_sha1
=$
(git ls-files
-u -- "$MERGED" |
awk '{if ($3==3) print $2;}')
252 describe_file
"$local_mode" "local" "$local_sha1"
253 describe_file
"$remote_mode" "remote" "$remote_sha1"
254 resolve_submodule_merge
258 mv -- "$MERGED" "$BACKUP"
259 cp -- "$BACKUP" "$MERGED"
261 checkout_staged_file
1 "$MERGED" "$BASE"
262 checkout_staged_file
2 "$MERGED" "$LOCAL"
263 checkout_staged_file
3 "$MERGED" "$REMOTE"
265 if test -z "$local_mode" -o -z "$remote_mode"
267 echo "Deleted merge conflict for '$MERGED':"
268 describe_file
"$local_mode" "local" "$LOCAL"
269 describe_file
"$remote_mode" "remote" "$REMOTE"
270 resolve_deleted_merge
274 if is_symlink
"$local_mode" || is_symlink
"$remote_mode"
276 echo "Symbolic link merge conflict for '$MERGED':"
277 describe_file
"$local_mode" "local" "$LOCAL"
278 describe_file
"$remote_mode" "remote" "$REMOTE"
279 resolve_symlink_merge
283 echo "Normal merge conflict for '$MERGED':"
284 describe_file
"$local_mode" "local" "$LOCAL"
285 describe_file
"$remote_mode" "remote" "$REMOTE"
288 printf "Hit return to start merge resolution tool (%s): " "$merge_tool"
299 if ! run_merge_tool
"$merge_tool" "$present"
301 echo "merge of $MERGED failed" 1>&2
302 mv -- "$BACKUP" "$MERGED"
304 if test "$merge_keep_temporaries" = "false"
312 if test "$merge_keep_backup" = "true"
314 mv -- "$BACKUP" "$MERGED.orig"
324 prompt
=$
(git config
--bool mergetool.prompt ||
echo true
)
330 TOOL_MODE
=${1#--tool-help=}
339 merge_tool
=$
(expr "z$1" : 'z-[^=]*=\(.*\)')
368 prompt_after_failed_merge
() {
371 printf "Continue merging other unresolved paths (y/n) ? "
387 if test -z "$merge_tool"
389 merge_tool
=$
(get_merge_tool
"$merge_tool") ||
exit
391 merge_keep_backup
="$(git config --bool mergetool.keepBackup || echo true)"
392 merge_keep_temporaries
="$(git config --bool mergetool.keepTemporaries || echo false)"
402 if test -e "$GIT_DIR/MERGE_RR"
404 files
=$
(git rerere remaining
)
406 files
=$
(git ls-files
-u |
sed -e 's/^[^ ]* //' |
sort -u)
409 files
=$
(git ls-files
-u -- "$@" |
sed -e 's/^[^ ]* //' |
sort -u)
414 echo "No files need merging"
419 printf "%s\n" "$files"
425 if test $last_status -ne 0
427 prompt_after_failed_merge ||
exit 1
432 if test $last_status -ne 0