2007-03-06 13:05:16 +08:00
|
|
|
#!/bin/sh
|
|
|
|
#
|
|
|
|
# This program resolves merge conflicts in git
|
|
|
|
#
|
|
|
|
# Copyright (c) 2006 Theodore Y. Ts'o
|
2016-10-08 07:58:03 +08:00
|
|
|
# Copyright (c) 2009-2016 David Aguilar
|
2007-03-06 13:05:16 +08:00
|
|
|
#
|
|
|
|
# This file is licensed under the GPL v2, or a later version
|
2007-06-06 12:24:19 +08:00
|
|
|
# at the discretion of Junio C Hamano.
|
2007-03-06 13:05:16 +08:00
|
|
|
#
|
|
|
|
|
2018-10-25 00:25:31 +08:00
|
|
|
USAGE='[--tool=tool] [--tool-help] [-y|--no-prompt|--prompt] [-g|--gui|--no-gui] [-O<orderfile>] [file to merge] ...'
|
2007-03-06 13:05:16 +08:00
|
|
|
SUBDIRECTORY_OK=Yes
|
2014-10-11 16:39:37 +08:00
|
|
|
NONGIT_OK=Yes
|
2007-11-06 17:50:02 +08:00
|
|
|
OPTIONS_SPEC=
|
2009-04-08 15:17:20 +08:00
|
|
|
TOOL_MODE=merge
|
2007-03-06 13:05:16 +08:00
|
|
|
. git-sh-setup
|
2009-04-08 15:17:20 +08:00
|
|
|
. git-mergetool--lib
|
2007-03-06 13:05:16 +08:00
|
|
|
|
|
|
|
# Returns true if the mode reflects a symlink
|
2007-03-29 18:55:11 +08:00
|
|
|
is_symlink () {
|
2012-08-23 13:33:15 +08:00
|
|
|
test "$1" = 120000
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
is_submodule () {
|
2012-08-23 13:33:15 +08:00
|
|
|
test "$1" = 160000
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
}
|
|
|
|
|
2007-03-29 18:55:11 +08:00
|
|
|
local_present () {
|
2012-08-23 13:33:15 +08:00
|
|
|
test -n "$local_mode"
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
2007-03-29 18:55:11 +08:00
|
|
|
remote_present () {
|
2012-08-23 13:33:15 +08:00
|
|
|
test -n "$remote_mode"
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
2007-03-29 18:55:11 +08:00
|
|
|
base_present () {
|
2012-08-23 13:33:15 +08:00
|
|
|
test -n "$base_mode"
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
2014-10-12 01:04:45 +08:00
|
|
|
mergetool_tmpdir_init () {
|
|
|
|
if test "$(git config --bool mergetool.writeToTemp)" != true
|
|
|
|
then
|
|
|
|
MERGETOOL_TMPDIR=.
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
if MERGETOOL_TMPDIR=$(mktemp -d -t "git-mergetool-XXXXXX" 2>/dev/null)
|
|
|
|
then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
die "error: mktemp is needed when 'mergetool.writeToTemp' is true"
|
|
|
|
}
|
|
|
|
|
2007-03-06 13:05:16 +08:00
|
|
|
cleanup_temp_files () {
|
2012-08-23 13:33:15 +08:00
|
|
|
if test "$1" = --save-backup
|
|
|
|
then
|
|
|
|
rm -rf -- "$MERGED.orig"
|
|
|
|
test -e "$BACKUP" && mv -- "$BACKUP" "$MERGED.orig"
|
|
|
|
rm -f -- "$LOCAL" "$REMOTE" "$BASE"
|
|
|
|
else
|
|
|
|
rm -f -- "$LOCAL" "$REMOTE" "$BASE" "$BACKUP"
|
|
|
|
fi
|
2014-10-12 01:04:45 +08:00
|
|
|
if test "$MERGETOOL_TMPDIR" != "."
|
|
|
|
then
|
|
|
|
rmdir "$MERGETOOL_TMPDIR"
|
|
|
|
fi
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
2007-03-29 18:55:11 +08:00
|
|
|
describe_file () {
|
2012-08-23 13:33:15 +08:00
|
|
|
mode="$1"
|
|
|
|
branch="$2"
|
|
|
|
file="$3"
|
|
|
|
|
|
|
|
printf " {%s}: " "$branch"
|
|
|
|
if test -z "$mode"
|
|
|
|
then
|
|
|
|
echo "deleted"
|
|
|
|
elif is_symlink "$mode"
|
|
|
|
then
|
|
|
|
echo "a symbolic link -> '$(cat "$file")'"
|
|
|
|
elif is_submodule "$mode"
|
|
|
|
then
|
|
|
|
echo "submodule commit $file"
|
|
|
|
elif base_present
|
|
|
|
then
|
|
|
|
echo "modified file"
|
2007-03-06 13:05:16 +08:00
|
|
|
else
|
2012-08-23 13:33:15 +08:00
|
|
|
echo "created file"
|
2007-03-06 13:05:16 +08:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve_symlink_merge () {
|
2012-08-23 13:33:15 +08:00
|
|
|
while true
|
|
|
|
do
|
|
|
|
printf "Use (l)ocal or (r)emote, or (a)bort? "
|
|
|
|
read ans || return 1
|
|
|
|
case "$ans" in
|
|
|
|
[lL]*)
|
|
|
|
git checkout-index -f --stage=2 -- "$MERGED"
|
|
|
|
git add -- "$MERGED"
|
|
|
|
cleanup_temp_files --save-backup
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[rR]*)
|
|
|
|
git checkout-index -f --stage=3 -- "$MERGED"
|
|
|
|
git add -- "$MERGED"
|
|
|
|
cleanup_temp_files --save-backup
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[aA]*)
|
|
|
|
return 1
|
|
|
|
;;
|
|
|
|
esac
|
2007-03-06 13:05:16 +08:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve_deleted_merge () {
|
2012-08-23 13:33:15 +08:00
|
|
|
while true
|
|
|
|
do
|
|
|
|
if base_present
|
|
|
|
then
|
|
|
|
printf "Use (m)odified or (d)eleted file, or (a)bort? "
|
|
|
|
else
|
|
|
|
printf "Use (c)reated or (d)eleted file, or (a)bort? "
|
|
|
|
fi
|
|
|
|
read ans || return 1
|
|
|
|
case "$ans" in
|
|
|
|
[mMcC]*)
|
|
|
|
git add -- "$MERGED"
|
2016-03-10 15:13:59 +08:00
|
|
|
if test "$merge_keep_backup" = "true"
|
|
|
|
then
|
|
|
|
cleanup_temp_files --save-backup
|
|
|
|
else
|
|
|
|
cleanup_temp_files
|
|
|
|
fi
|
2012-08-23 13:33:15 +08:00
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[dD]*)
|
|
|
|
git rm -- "$MERGED" > /dev/null
|
|
|
|
cleanup_temp_files
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[aA]*)
|
2016-03-10 15:13:59 +08:00
|
|
|
if test "$merge_keep_temporaries" = "false"
|
|
|
|
then
|
|
|
|
cleanup_temp_files
|
|
|
|
fi
|
2012-08-23 13:33:15 +08:00
|
|
|
return 1
|
|
|
|
;;
|
|
|
|
esac
|
2007-03-06 13:05:16 +08:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
resolve_submodule_merge () {
|
2012-08-23 13:33:15 +08:00
|
|
|
while true
|
|
|
|
do
|
|
|
|
printf "Use (l)ocal or (r)emote, or (a)bort? "
|
|
|
|
read ans || return 1
|
|
|
|
case "$ans" in
|
|
|
|
[lL]*)
|
|
|
|
if ! local_present
|
|
|
|
then
|
|
|
|
if test -n "$(git ls-tree HEAD -- "$MERGED")"
|
|
|
|
then
|
|
|
|
# Local isn't present, but it's a subdirectory
|
|
|
|
git ls-tree --full-name -r HEAD -- "$MERGED" |
|
|
|
|
git update-index --index-info || exit $?
|
|
|
|
else
|
|
|
|
test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
|
|
|
|
git update-index --force-remove "$MERGED"
|
|
|
|
cleanup_temp_files --save-backup
|
|
|
|
fi
|
|
|
|
elif is_submodule "$local_mode"
|
|
|
|
then
|
|
|
|
stage_submodule "$MERGED" "$local_sha1"
|
|
|
|
else
|
|
|
|
git checkout-index -f --stage=2 -- "$MERGED"
|
|
|
|
git add -- "$MERGED"
|
|
|
|
fi
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[rR]*)
|
|
|
|
if ! remote_present
|
|
|
|
then
|
|
|
|
if test -n "$(git ls-tree MERGE_HEAD -- "$MERGED")"
|
|
|
|
then
|
|
|
|
# Remote isn't present, but it's a subdirectory
|
|
|
|
git ls-tree --full-name -r MERGE_HEAD -- "$MERGED" |
|
|
|
|
git update-index --index-info || exit $?
|
|
|
|
else
|
|
|
|
test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
|
|
|
|
git update-index --force-remove "$MERGED"
|
|
|
|
fi
|
|
|
|
elif is_submodule "$remote_mode"
|
|
|
|
then
|
|
|
|
! is_submodule "$local_mode" &&
|
|
|
|
test -e "$MERGED" &&
|
|
|
|
mv -- "$MERGED" "$BACKUP"
|
|
|
|
stage_submodule "$MERGED" "$remote_sha1"
|
|
|
|
else
|
|
|
|
test -e "$MERGED" && mv -- "$MERGED" "$BACKUP"
|
|
|
|
git checkout-index -f --stage=3 -- "$MERGED"
|
|
|
|
git add -- "$MERGED"
|
|
|
|
fi
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
cleanup_temp_files --save-backup
|
2012-08-23 13:33:15 +08:00
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[aA]*)
|
|
|
|
return 1
|
|
|
|
;;
|
|
|
|
esac
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
stage_submodule () {
|
2012-08-23 13:33:15 +08:00
|
|
|
path="$1"
|
|
|
|
submodule_sha1="$2"
|
|
|
|
mkdir -p "$path" ||
|
|
|
|
die "fatal: unable to create directory for module at $path"
|
|
|
|
# Find $path relative to work tree
|
|
|
|
work_tree_root=$(cd_to_toplevel && pwd)
|
|
|
|
work_rel_path=$(cd "$path" &&
|
|
|
|
GIT_WORK_TREE="${work_tree_root}" git rev-parse --show-prefix
|
|
|
|
)
|
|
|
|
test -n "$work_rel_path" ||
|
|
|
|
die "fatal: unable to get path of module $path relative to work tree"
|
|
|
|
git update-index --add --replace --cacheinfo 160000 "$submodule_sha1" "${work_rel_path%/}" || die
|
mergetool: Teach about submodules
When the index has conflicted submodules, mergetool used to mildly
clobber the module, renaming it to mymodule.BACKUP.nnnn, then failing to
copy it non-recursively.
Recognize submodules and offer a resolution instead:
Submodule merge conflict for 'Shared':
{local}: submodule commit ad9f12e3e6205381bf2163a793d1e596a9e211d0
{remote}: submodule commit f5893fb70ec5646efcd9aa643c5136753ac89253
Use (l)ocal or (r)emote, or (a)bort?
Selecting a commit will stage it, but not update the submodule (as git
does had there been no conflict). Type changes are also supported,
should the path be a submodule on one side, and a file, symlink,
directory, or deleted on the other.
Signed-off-by: Jonathon Mah <me@JonathonMah.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2011-04-13 18:00:48 +08:00
|
|
|
}
|
|
|
|
|
2009-01-22 06:57:48 +08:00
|
|
|
checkout_staged_file () {
|
2019-06-13 00:33:47 +08:00
|
|
|
tmpfile="$(git checkout-index --temp --stage="$1" "$2" 2>/dev/null)" &&
|
|
|
|
tmpfile=${tmpfile%%' '*}
|
2012-08-23 13:33:15 +08:00
|
|
|
|
2014-06-06 22:55:51 +08:00
|
|
|
if test $? -eq 0 && test -n "$tmpfile"
|
2012-08-23 13:33:15 +08:00
|
|
|
then
|
|
|
|
mv -- "$(git rev-parse --show-cdup)$tmpfile" "$3"
|
|
|
|
else
|
|
|
|
>"$3"
|
|
|
|
fi
|
2009-01-22 06:57:48 +08:00
|
|
|
}
|
|
|
|
|
2021-02-10 04:07:10 +08:00
|
|
|
hide_resolved () {
|
|
|
|
git merge-file --ours -q -p "$LOCAL" "$BASE" "$REMOTE" >"$LCONFL"
|
|
|
|
git merge-file --theirs -q -p "$LOCAL" "$BASE" "$REMOTE" >"$RCONFL"
|
|
|
|
mv -- "$LCONFL" "$LOCAL"
|
|
|
|
mv -- "$RCONFL" "$REMOTE"
|
|
|
|
}
|
|
|
|
|
2007-03-06 13:05:16 +08:00
|
|
|
merge_file () {
|
2012-08-23 13:33:15 +08:00
|
|
|
MERGED="$1"
|
2007-03-06 13:05:16 +08:00
|
|
|
|
2012-08-23 13:33:15 +08:00
|
|
|
f=$(git ls-files -u -- "$MERGED")
|
|
|
|
if test -z "$f"
|
|
|
|
then
|
|
|
|
if test ! -f "$MERGED"
|
|
|
|
then
|
|
|
|
echo "$MERGED: file not found"
|
|
|
|
else
|
|
|
|
echo "$MERGED: file does not need merging"
|
|
|
|
fi
|
|
|
|
return 1
|
2007-03-28 06:00:03 +08:00
|
|
|
fi
|
2007-03-06 13:05:16 +08:00
|
|
|
|
2019-06-13 00:33:47 +08:00
|
|
|
# extract file extension from the last path component
|
|
|
|
case "${MERGED##*/}" in
|
|
|
|
*.*)
|
|
|
|
ext=.${MERGED##*.}
|
|
|
|
BASE=${MERGED%"$ext"}
|
|
|
|
;;
|
|
|
|
*)
|
2014-10-10 16:19:47 +08:00
|
|
|
BASE=$MERGED
|
|
|
|
ext=
|
2019-06-13 00:33:47 +08:00
|
|
|
esac
|
2014-10-12 01:04:45 +08:00
|
|
|
|
2021-02-10 04:07:11 +08:00
|
|
|
initialize_merge_tool "$merge_tool" || return
|
|
|
|
|
2014-10-12 01:04:45 +08:00
|
|
|
mergetool_tmpdir_init
|
|
|
|
|
|
|
|
if test "$MERGETOOL_TMPDIR" != "."
|
|
|
|
then
|
|
|
|
# If we're using a temporary directory then write to the
|
|
|
|
# top-level of that directory.
|
|
|
|
BASE=${BASE##*/}
|
|
|
|
fi
|
|
|
|
|
|
|
|
BACKUP="$MERGETOOL_TMPDIR/${BASE}_BACKUP_$$$ext"
|
|
|
|
LOCAL="$MERGETOOL_TMPDIR/${BASE}_LOCAL_$$$ext"
|
2021-02-10 04:07:10 +08:00
|
|
|
LCONFL="$MERGETOOL_TMPDIR/${BASE}_LOCAL_LCONFL_$$$ext"
|
2014-10-12 01:04:45 +08:00
|
|
|
REMOTE="$MERGETOOL_TMPDIR/${BASE}_REMOTE_$$$ext"
|
2021-02-10 04:07:10 +08:00
|
|
|
RCONFL="$MERGETOOL_TMPDIR/${BASE}_REMOTE_RCONFL_$$$ext"
|
2014-10-12 01:04:45 +08:00
|
|
|
BASE="$MERGETOOL_TMPDIR/${BASE}_BASE_$$$ext"
|
2012-08-23 13:33:15 +08:00
|
|
|
|
2019-06-13 00:33:48 +08:00
|
|
|
base_mode= local_mode= remote_mode=
|
|
|
|
|
|
|
|
# here, $IFS is just a LF
|
|
|
|
for line in $f
|
|
|
|
do
|
|
|
|
mode=${line%% *} # 1st word
|
|
|
|
sha1=${line#"$mode "}
|
|
|
|
sha1=${sha1%% *} # 2nd word
|
|
|
|
case "${line#$mode $sha1 }" in # remainder
|
|
|
|
'1 '*)
|
|
|
|
base_mode=$mode
|
|
|
|
;;
|
|
|
|
'2 '*)
|
|
|
|
local_mode=$mode local_sha1=$sha1
|
|
|
|
;;
|
|
|
|
'3 '*)
|
|
|
|
remote_mode=$mode remote_sha1=$sha1
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
2012-08-23 13:33:15 +08:00
|
|
|
|
|
|
|
if is_submodule "$local_mode" || is_submodule "$remote_mode"
|
|
|
|
then
|
|
|
|
echo "Submodule merge conflict for '$MERGED':"
|
|
|
|
describe_file "$local_mode" "local" "$local_sha1"
|
|
|
|
describe_file "$remote_mode" "remote" "$remote_sha1"
|
|
|
|
resolve_submodule_merge
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
2016-03-10 15:13:58 +08:00
|
|
|
if test -f "$MERGED"
|
|
|
|
then
|
|
|
|
mv -- "$MERGED" "$BACKUP"
|
|
|
|
cp -- "$BACKUP" "$MERGED"
|
|
|
|
fi
|
|
|
|
# Create a parent directory to handle delete/delete conflicts
|
|
|
|
# where the base's directory no longer exists.
|
|
|
|
mkdir -p "$(dirname "$MERGED")"
|
2012-08-23 13:33:15 +08:00
|
|
|
|
|
|
|
checkout_staged_file 1 "$MERGED" "$BASE"
|
|
|
|
checkout_staged_file 2 "$MERGED" "$LOCAL"
|
|
|
|
checkout_staged_file 3 "$MERGED" "$REMOTE"
|
|
|
|
|
2021-02-10 04:07:12 +08:00
|
|
|
# hideResolved preferences hierarchy.
|
|
|
|
global_config="mergetool.hideResolved"
|
|
|
|
tool_config="mergetool.${merge_tool}.hideResolved"
|
|
|
|
|
|
|
|
if enabled=$(git config --type=bool "$tool_config")
|
|
|
|
then
|
|
|
|
# The user has a specific preference for a specific tool and no
|
|
|
|
# other preferences should override that.
|
|
|
|
: ;
|
|
|
|
elif enabled=$(git config --type=bool "$global_config")
|
|
|
|
then
|
|
|
|
# The user has a general preference for all tools.
|
|
|
|
#
|
|
|
|
# 'true' means the user likes the feature so we should use it
|
|
|
|
# where possible but tool authors can still override.
|
|
|
|
#
|
|
|
|
# 'false' means the user doesn't like the feature so we should
|
|
|
|
# not use it anywhere.
|
|
|
|
if test "$enabled" = true && hide_resolved_enabled
|
|
|
|
then
|
|
|
|
enabled=true
|
|
|
|
else
|
|
|
|
enabled=false
|
|
|
|
fi
|
|
|
|
else
|
mergetool: do not enable hideResolved by default
When 98ea309b3f (mergetool: add hideResolved configuration,
2021-02-09) introduced the mergetool.hideResolved setting to reduce
the clutter in viewing non-conflicted sections of files in a
mergetool, it enabled it by default, explaining:
No adverse effects were noted in a small survey of popular mergetools[1]
so this behavior defaults to `true`.
In practice, alas, adverse effects do appear. A few issues:
1. No indication is shown in the UI that the base, local, and remote
versions shown have been modified by additional resolution. This
is inherent in the design: the idea of mergetool.hideResolved is to
convince a mergetool that expects pristine local, base, and remote
files to show partially resolved verisons of those files instead;
there is no additional source of information accessible to the
mergetool to see where the resolution has happened.
(By contrast, a mergetool generating the partial resolution from
conflict markers for itself would be able to hilight the resolved
sections with a different color.)
A user accustomed to seeing the files without partial resolution
gets no indication that this behavior has changed when they upgrade
Git.
2. If the computed merge did not line up the files correctly (for
example due to repeated sections in the file), the partially
resolved files can be misleading and do not have enough information
to reconstruct what happened and compute the correct merge result.
3. Resolving a conflict can involve information beyond the textual
conflict. For example, if the local and remote versions added
overlapping functionality in different ways, seeing the full
unresolved versions of each alongside the base gives information
about each side's intent that makes it possible to come up with a
resolution that combines those two intents. By contrast, when
starting with partially resolved versions of those files, one can
produce a subtly wrong resolution that includes redundant extra
code added by one side that is not needed in the approach taken
on the other.
All that said, a user wanting to focus on textual conflicts with
reduced clutter can still benefit from mergetool.hideResolved=true as
a way to deemphasize sections of the code that resolve cleanly without
requiring any changes to the invoked mergetool. The caveats described
above are reduced when the user has explicitly turned this on, because
then the user is aware of them.
Flip the default to 'false'.
Reported-by: Dana Dahlstrom <dahlstrom@google.com>
Helped-by: Seth House <seth@eseth.com>
Signed-off-by: Jonathan Nieder <jrnieder@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2021-03-13 16:38:48 +08:00
|
|
|
# The user does not have a preference. Default to disabled.
|
|
|
|
enabled=false
|
2021-02-10 04:07:12 +08:00
|
|
|
fi
|
|
|
|
|
|
|
|
if test "$enabled" = true
|
2021-02-10 04:07:10 +08:00
|
|
|
then
|
|
|
|
hide_resolved
|
|
|
|
fi
|
|
|
|
|
2014-06-06 22:55:51 +08:00
|
|
|
if test -z "$local_mode" || test -z "$remote_mode"
|
2012-08-23 13:33:15 +08:00
|
|
|
then
|
|
|
|
echo "Deleted merge conflict for '$MERGED':"
|
|
|
|
describe_file "$local_mode" "local" "$LOCAL"
|
|
|
|
describe_file "$remote_mode" "remote" "$REMOTE"
|
|
|
|
resolve_deleted_merge
|
2016-03-10 15:13:58 +08:00
|
|
|
status=$?
|
|
|
|
rmdir -p "$(dirname "$MERGED")" 2>/dev/null
|
|
|
|
return $status
|
2012-08-23 13:33:15 +08:00
|
|
|
fi
|
|
|
|
|
|
|
|
if is_symlink "$local_mode" || is_symlink "$remote_mode"
|
|
|
|
then
|
|
|
|
echo "Symbolic link merge conflict for '$MERGED':"
|
|
|
|
describe_file "$local_mode" "local" "$LOCAL"
|
|
|
|
describe_file "$remote_mode" "remote" "$REMOTE"
|
|
|
|
resolve_symlink_merge
|
|
|
|
return
|
|
|
|
fi
|
|
|
|
|
|
|
|
echo "Normal merge conflict for '$MERGED':"
|
2007-03-06 13:05:16 +08:00
|
|
|
describe_file "$local_mode" "local" "$LOCAL"
|
|
|
|
describe_file "$remote_mode" "remote" "$REMOTE"
|
2014-04-21 08:17:34 +08:00
|
|
|
if test "$guessed_merge_tool" = true || test "$prompt" = true
|
2012-08-23 13:33:15 +08:00
|
|
|
then
|
|
|
|
printf "Hit return to start merge resolution tool (%s): " "$merge_tool"
|
|
|
|
read ans || return 1
|
2008-12-13 05:48:41 +08:00
|
|
|
fi
|
|
|
|
|
2012-08-23 13:33:15 +08:00
|
|
|
if base_present
|
|
|
|
then
|
|
|
|
present=true
|
|
|
|
else
|
|
|
|
present=false
|
|
|
|
fi
|
|
|
|
|
|
|
|
if ! run_merge_tool "$merge_tool" "$present"
|
|
|
|
then
|
|
|
|
echo "merge of $MERGED failed" 1>&2
|
|
|
|
mv -- "$BACKUP" "$MERGED"
|
|
|
|
|
|
|
|
if test "$merge_keep_temporaries" = "false"
|
|
|
|
then
|
|
|
|
cleanup_temp_files
|
|
|
|
fi
|
2008-02-22 07:30:02 +08:00
|
|
|
|
2012-08-23 13:33:15 +08:00
|
|
|
return 1
|
|
|
|
fi
|
2008-02-22 07:30:02 +08:00
|
|
|
|
2012-08-23 13:33:15 +08:00
|
|
|
if test "$merge_keep_backup" = "true"
|
|
|
|
then
|
|
|
|
mv -- "$BACKUP" "$MERGED.orig"
|
|
|
|
else
|
|
|
|
rm -- "$BACKUP"
|
|
|
|
fi
|
|
|
|
|
|
|
|
git add -- "$MERGED"
|
|
|
|
cleanup_temp_files
|
|
|
|
return 0
|
2007-03-06 13:05:16 +08:00
|
|
|
}
|
|
|
|
|
2012-08-23 13:33:15 +08:00
|
|
|
prompt_after_failed_merge () {
|
|
|
|
while true
|
|
|
|
do
|
2016-04-12 22:44:20 +08:00
|
|
|
printf "Continue merging other unresolved paths [y/n]? "
|
2012-08-23 13:33:15 +08:00
|
|
|
read ans || return 1
|
|
|
|
case "$ans" in
|
|
|
|
[yY]*)
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
[nN]*)
|
|
|
|
return 1
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
done
|
2008-12-13 05:48:40 +08:00
|
|
|
}
|
2007-10-18 01:16:11 +08:00
|
|
|
|
2016-10-08 07:58:05 +08:00
|
|
|
print_noop_and_exit () {
|
|
|
|
echo "No files need merging"
|
|
|
|
exit 0
|
|
|
|
}
|
|
|
|
|
2016-10-08 07:58:04 +08:00
|
|
|
main () {
|
|
|
|
prompt=$(git config --bool mergetool.prompt)
|
mergetool: new config guiDefault supports auto-toggling gui by DISPLAY
When no merge.tool or diff.tool is configured or manually selected, the
selection of a default tool is sensitive to the DISPLAY variable; in a
GUI session a gui-specific tool will be proposed if found, and
otherwise a terminal-based one. This "GUI-optimizing" behavior is
important because a GUI can make a huge difference to a user's ability
to understand and correctly complete a non-trivial conflicting merge.
Some time ago the merge.guitool and diff.guitool config options were
introduced to enable users to configure both a GUI tool, and a non-GUI
tool (with fallback if no GUI tool configured), in the same environment.
Unfortunately, the --gui argument introduced to support the selection of
the guitool is still explicit. When using configured tools, there is no
equivalent of the no-tool-configured "propose a GUI tool if we are in a GUI
environment" behavior.
As proposed in <xmqqmtb8jsej.fsf@gitster.g>, introduce new configuration
options, difftool.guiDefault and mergetool.guiDefault, supporting a special
value "auto" which causes the corresponding tool or guitool to be selected
depending on the presence of a non-empty DISPLAY value. Also support "true"
to say "default to the guitool (unless --no-gui is passed on the
commandline)", and "false" as the previous default behavior when these new
configuration options are not specified.
Signed-off-by: Tao Klerks <tao@klerks.biz>
Acked-by: David Aguilar <davvid@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2023-03-18 23:27:43 +08:00
|
|
|
GIT_MERGETOOL_GUI=
|
2016-10-08 07:58:04 +08:00
|
|
|
guessed_merge_tool=false
|
2016-10-08 08:01:30 +08:00
|
|
|
orderfile=
|
2016-10-08 07:58:04 +08:00
|
|
|
|
|
|
|
while test $# != 0
|
|
|
|
do
|
|
|
|
case "$1" in
|
|
|
|
--tool-help=*)
|
|
|
|
TOOL_MODE=${1#--tool-help=}
|
|
|
|
show_tool_help
|
|
|
|
;;
|
|
|
|
--tool-help)
|
|
|
|
show_tool_help
|
|
|
|
;;
|
|
|
|
-t|--tool*)
|
|
|
|
case "$#,$1" in
|
|
|
|
*,*=*)
|
2019-06-13 00:33:47 +08:00
|
|
|
merge_tool=${1#*=}
|
2016-10-08 07:58:04 +08:00
|
|
|
;;
|
|
|
|
1,*)
|
|
|
|
usage ;;
|
|
|
|
*)
|
|
|
|
merge_tool="$2"
|
|
|
|
shift ;;
|
|
|
|
esac
|
|
|
|
;;
|
2018-10-25 00:25:31 +08:00
|
|
|
--no-gui)
|
2019-04-29 14:21:08 +08:00
|
|
|
GIT_MERGETOOL_GUI=false
|
2018-10-25 00:25:31 +08:00
|
|
|
;;
|
|
|
|
-g|--gui)
|
2019-04-29 14:21:08 +08:00
|
|
|
GIT_MERGETOOL_GUI=true
|
2018-10-25 00:25:31 +08:00
|
|
|
;;
|
2016-10-08 07:58:04 +08:00
|
|
|
-y|--no-prompt)
|
|
|
|
prompt=false
|
|
|
|
;;
|
|
|
|
--prompt)
|
|
|
|
prompt=true
|
|
|
|
;;
|
2016-10-08 08:01:30 +08:00
|
|
|
-O*)
|
2017-01-11 04:42:01 +08:00
|
|
|
orderfile="${1#-O}"
|
2016-10-08 08:01:30 +08:00
|
|
|
;;
|
2016-10-08 07:58:04 +08:00
|
|
|
--)
|
|
|
|
shift
|
|
|
|
break
|
|
|
|
;;
|
|
|
|
-*)
|
|
|
|
usage
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
break
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
shift
|
|
|
|
done
|
|
|
|
|
|
|
|
git_dir_init
|
|
|
|
require_work_tree
|
2014-10-11 16:39:37 +08:00
|
|
|
|
2014-04-21 08:17:34 +08:00
|
|
|
if test -z "$merge_tool"
|
|
|
|
then
|
mergetool: new config guiDefault supports auto-toggling gui by DISPLAY
When no merge.tool or diff.tool is configured or manually selected, the
selection of a default tool is sensitive to the DISPLAY variable; in a
GUI session a gui-specific tool will be proposed if found, and
otherwise a terminal-based one. This "GUI-optimizing" behavior is
important because a GUI can make a huge difference to a user's ability
to understand and correctly complete a non-trivial conflicting merge.
Some time ago the merge.guitool and diff.guitool config options were
introduced to enable users to configure both a GUI tool, and a non-GUI
tool (with fallback if no GUI tool configured), in the same environment.
Unfortunately, the --gui argument introduced to support the selection of
the guitool is still explicit. When using configured tools, there is no
equivalent of the no-tool-configured "propose a GUI tool if we are in a GUI
environment" behavior.
As proposed in <xmqqmtb8jsej.fsf@gitster.g>, introduce new configuration
options, difftool.guiDefault and mergetool.guiDefault, supporting a special
value "auto" which causes the corresponding tool or guitool to be selected
depending on the presence of a non-empty DISPLAY value. Also support "true"
to say "default to the guitool (unless --no-gui is passed on the
commandline)", and "false" as the previous default behavior when these new
configuration options are not specified.
Signed-off-by: Tao Klerks <tao@klerks.biz>
Acked-by: David Aguilar <davvid@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2023-03-18 23:27:43 +08:00
|
|
|
merge_tool=$(get_merge_tool)
|
|
|
|
subshell_exit_status=$?
|
|
|
|
if test $subshell_exit_status = 1
|
2016-10-08 07:58:04 +08:00
|
|
|
then
|
|
|
|
guessed_merge_tool=true
|
mergetool: new config guiDefault supports auto-toggling gui by DISPLAY
When no merge.tool or diff.tool is configured or manually selected, the
selection of a default tool is sensitive to the DISPLAY variable; in a
GUI session a gui-specific tool will be proposed if found, and
otherwise a terminal-based one. This "GUI-optimizing" behavior is
important because a GUI can make a huge difference to a user's ability
to understand and correctly complete a non-trivial conflicting merge.
Some time ago the merge.guitool and diff.guitool config options were
introduced to enable users to configure both a GUI tool, and a non-GUI
tool (with fallback if no GUI tool configured), in the same environment.
Unfortunately, the --gui argument introduced to support the selection of
the guitool is still explicit. When using configured tools, there is no
equivalent of the no-tool-configured "propose a GUI tool if we are in a GUI
environment" behavior.
As proposed in <xmqqmtb8jsej.fsf@gitster.g>, introduce new configuration
options, difftool.guiDefault and mergetool.guiDefault, supporting a special
value "auto" which causes the corresponding tool or guitool to be selected
depending on the presence of a non-empty DISPLAY value. Also support "true"
to say "default to the guitool (unless --no-gui is passed on the
commandline)", and "false" as the previous default behavior when these new
configuration options are not specified.
Signed-off-by: Tao Klerks <tao@klerks.biz>
Acked-by: David Aguilar <davvid@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
2023-03-18 23:27:43 +08:00
|
|
|
elif test $subshell_exit_status -gt 1
|
|
|
|
then
|
|
|
|
exit $subshell_exit_status
|
2016-10-08 07:58:04 +08:00
|
|
|
fi
|
2014-04-21 08:17:34 +08:00
|
|
|
fi
|
2016-10-08 07:58:04 +08:00
|
|
|
merge_keep_backup="$(git config --bool mergetool.keepBackup || echo true)"
|
|
|
|
merge_keep_temporaries="$(git config --bool mergetool.keepTemporaries || echo false)"
|
2007-03-06 13:05:16 +08:00
|
|
|
|
2017-01-11 04:42:02 +08:00
|
|
|
prefix=$(git rev-parse --show-prefix) || exit 1
|
|
|
|
cd_to_toplevel
|
|
|
|
|
|
|
|
if test -n "$orderfile"
|
|
|
|
then
|
|
|
|
orderfile=$(
|
|
|
|
git rev-parse --prefix "$prefix" -- "$orderfile" |
|
|
|
|
sed -e 1d
|
|
|
|
)
|
|
|
|
fi
|
|
|
|
|
2016-10-08 07:58:05 +08:00
|
|
|
if test $# -eq 0 && test -e "$GIT_DIR/MERGE_RR"
|
2012-08-23 13:33:15 +08:00
|
|
|
then
|
2016-10-08 07:58:05 +08:00
|
|
|
set -- $(git rerere remaining)
|
|
|
|
if test $# -eq 0
|
2016-10-08 07:58:04 +08:00
|
|
|
then
|
2016-10-08 07:58:05 +08:00
|
|
|
print_noop_and_exit
|
2016-10-08 07:58:04 +08:00
|
|
|
fi
|
2017-01-11 04:42:02 +08:00
|
|
|
elif test $# -ge 0
|
|
|
|
then
|
|
|
|
# rev-parse provides the -- needed for 'set'
|
|
|
|
eval "set $(git rev-parse --sq --prefix "$prefix" -- "$@")"
|
2012-08-23 13:33:15 +08:00
|
|
|
fi
|
2016-10-08 07:58:04 +08:00
|
|
|
|
2016-10-08 07:58:05 +08:00
|
|
|
files=$(git -c core.quotePath=false \
|
2016-10-08 08:01:30 +08:00
|
|
|
diff --name-only --diff-filter=U \
|
2017-01-11 04:42:01 +08:00
|
|
|
${orderfile:+"-O$orderfile"} -- "$@")
|
2016-10-08 07:58:05 +08:00
|
|
|
|
2016-10-08 07:58:04 +08:00
|
|
|
if test -z "$files"
|
2012-08-23 13:33:15 +08:00
|
|
|
then
|
2016-10-08 07:58:05 +08:00
|
|
|
print_noop_and_exit
|
2012-08-23 13:33:15 +08:00
|
|
|
fi
|
2008-12-13 05:48:40 +08:00
|
|
|
|
2016-10-08 07:58:04 +08:00
|
|
|
printf "Merging:\n"
|
|
|
|
printf "%s\n" "$files"
|
|
|
|
|
|
|
|
rc=0
|
2018-08-13 13:09:29 +08:00
|
|
|
set -- $files
|
|
|
|
while test $# -ne 0
|
2016-10-08 07:58:04 +08:00
|
|
|
do
|
|
|
|
printf "\n"
|
2018-08-13 13:09:29 +08:00
|
|
|
if ! merge_file "$1"
|
2016-10-08 07:58:04 +08:00
|
|
|
then
|
|
|
|
rc=1
|
2018-08-13 13:09:29 +08:00
|
|
|
test $# -ne 1 && prompt_after_failed_merge || exit 1
|
2016-10-08 07:58:04 +08:00
|
|
|
fi
|
2018-08-13 13:09:29 +08:00
|
|
|
shift
|
2016-10-08 07:58:04 +08:00
|
|
|
done
|
|
|
|
|
|
|
|
exit $rc
|
|
|
|
}
|
|
|
|
|
|
|
|
main "$@"
|