hgmerge
143 lines
| 3.7 KiB
| text/plain
|
TextLexer
Thomas Arendsen Hein
|
r544 | #!/bin/sh | ||
mpm@selenic.com
|
r240 | # | ||
# hgmerge - default merge helper for Mercurial | ||||
# | ||||
# This tries to find a way to do three-way merge on the current system. | ||||
# The result ought to end up in $1. | ||||
set -e # bail out quickly on failure | ||||
Thomas Arendsen Hein
|
r795 | LOCAL="$1" | ||
BASE="$2" | ||||
OTHER="$3" | ||||
mpm@selenic.com
|
r240 | |||
Thomas Arendsen Hein
|
r795 | if [ -z "$EDITOR" ]; then | ||
EDITOR="vi" | ||||
fi | ||||
Thomas Arendsen Hein
|
r304 | |||
levon@movementarian.org
|
r1434 | # find decent versions of our utilities, insisting on the GNU versions where we | ||
# need to | ||||
DIFF3=gdiff3 | ||||
DIFF=gdiff | ||||
PATCH=gpatch | ||||
type $DIFF3 >/dev/null 2>&1 || DIFF3=diff3 | ||||
type $DIFF >/dev/null 2>&1 || DIFF=diff | ||||
type $PATCH >/dev/null 2>&1 || PATCH=patch | ||||
$DIFF3 --version >/dev/null 2>&1 || DIFF3= | ||||
mpm@selenic.com
|
r240 | # Back up our file | ||
Thomas Arendsen Hein
|
r795 | cp "$LOCAL" "$LOCAL.orig" | ||
mpm@selenic.com
|
r240 | |||
# Attempt to do a non-interactive merge | ||||
Thomas Arendsen Hein
|
r828 | if type merge > /dev/null 2>&1; then | ||
Thomas Arendsen Hein
|
r829 | merge "$LOCAL" "$BASE" "$OTHER" 2> /dev/null && exit 0 | ||
Thomas Arendsen Hein
|
r795 | cp "$LOCAL.orig" "$LOCAL" | ||
levon@movementarian.org
|
r1434 | elif [ -n "$DIFF3" ]; then | ||
echo $DIFF3 -m "$LOCAL.orig" "$BASE" "$OTHER" | ||||
$DIFF3 -m "$LOCAL.orig" "$BASE" "$OTHER" > "$LOCAL" && exit 0 | ||||
if [ $? -eq 2 ]; then | ||||
echo "$DIFF3 failed! Exiting." 1>&2 | ||||
cp "$LOCAL.orig" "$LOCAL" | ||||
exit 1 | ||||
fi | ||||
Thomas Arendsen Hein
|
r795 | cp "$LOCAL.orig" "$LOCAL" | ||
mpm@selenic.com
|
r240 | fi | ||
Brendan Cully
|
r1664 | # on MacOS X try FileMerge.app, shipped with Apple's developer tools | ||
# TODO: make proper temp files. foo.orig and foo.link are dangerous | ||||
FILEMERGE='/Developer/Applications/Utilities/FileMerge.app/Contents/MacOS/FileMerge' | ||||
if type "$FILEMERGE" > /dev/null 2>&1; then | ||||
cp "$LOCAL.orig" "$LOCAL" | ||||
ln "$LOCAL" "$LOCAL.link" | ||||
# filemerge prefers the right by default | ||||
if ! "$FILEMERGE" -left "$OTHER" -right "$LOCAL" -ancestor "$BASE" -merge "$LOCAL" | ||||
then | ||||
echo "FileMerge failed to launch" | ||||
exit 1 | ||||
fi | ||||
if ! test "$LOCAL" -ef "$LOCAL.link" | ||||
then | ||||
rm "$LOCAL.orig" "$LOCAL.link" | ||||
exit 0 | ||||
else | ||||
rm "$LOCAL.link" | ||||
echo "$LOCAL is unchanged. Was the merge successful?" | ||||
select answer in yes no | ||||
do | ||||
if test "$answer" == "yes" | ||||
then | ||||
rm "$LOCAL.orig" | ||||
exit 0 | ||||
else | ||||
exit 1 | ||||
fi | ||||
done | ||||
exit 1 | ||||
fi | ||||
Christian Ebert
|
r1647 | fi | ||
Thomas Arendsen Hein
|
r303 | if [ -n "$DISPLAY" ]; then | ||
# try using kdiff3, which is fairly nice | ||||
Thomas Arendsen Hein
|
r828 | if type kdiff3 > /dev/null 2>&1; then | ||
Thomas Arendsen Hein
|
r829 | kdiff3 --auto "$BASE" "$LOCAL" "$OTHER" -o "$LOCAL" || exit 1 | ||
exit 0 | ||||
mpm@selenic.com
|
r240 | fi | ||
Thomas Arendsen Hein
|
r303 | # try using tkdiff, which is a bit less sophisticated | ||
Thomas Arendsen Hein
|
r828 | if type tkdiff > /dev/null 2>&1; then | ||
Thomas Arendsen Hein
|
r829 | tkdiff "$LOCAL" "$OTHER" -a "$BASE" -o "$LOCAL" || exit 1 | ||
exit 0 | ||||
mpm@selenic.com
|
r240 | fi | ||
fi | ||||
# Attempt to do a merge with $EDITOR | ||||
Thomas Arendsen Hein
|
r828 | if type merge > /dev/null 2>&1; then | ||
mpm@selenic.com
|
r240 | echo "conflicts detected in $LOCAL" | ||
Thomas Arendsen Hein
|
r795 | merge "$LOCAL" "$BASE" "$OTHER" 2>/dev/null || $EDITOR "$LOCAL" | ||
mpm@selenic.com
|
r242 | exit 0 | ||
fi | ||||
levon@movementarian.org
|
r1434 | if [ -n "$DIFF3" ]; then | ||
mpm@selenic.com
|
r242 | echo "conflicts detected in $LOCAL" | ||
levon@movementarian.org
|
r1434 | $DIFF3 -m "$LOCAL.orig" "$BASE" "$OTHER" > "$LOCAL" || { | ||
case $? in | ||||
1) | ||||
$EDITOR "$LOCAL" ;; | ||||
2) echo "$DIFF3 failed! Exiting." 1>&2 | ||||
cp "$LOCAL.orig" "$LOCAL" | ||||
exit 1 ;; | ||||
esac | ||||
exit 0 | ||||
} | ||||
mpm@selenic.com
|
r240 | fi | ||
Thomas Arendsen Hein
|
r795 | HGTMP="" | ||
cleanup_exit() { | ||||
rm -rf "$HGTMP" | ||||
} | ||||
mpm@selenic.com
|
r240 | # attempt to manually merge with diff and patch | ||
levon@movementarian.org
|
r1434 | if [ -n "$DIFF" -a -n "$PATCH" ]; then | ||
Thomas Arendsen Hein
|
r829 | # Remove temporary files even if we get interrupted | ||
Thomas Arendsen Hein
|
r831 | trap "cleanup_exit" 0 # normal exit | ||
trap "exit 1" 1 2 3 6 15 # HUP INT QUIT ABRT TERM | ||||
Thomas Arendsen Hein
|
r795 | |||
Thomas Arendsen Hein
|
r829 | HGTMP="${TMPDIR-/tmp}/hgmerge.$RANDOM.$RANDOM.$RANDOM.$$" | ||
(umask 077 && mkdir "$HGTMP") || { | ||||
echo "Could not create temporary directory! Exiting." 1>&2 | ||||
exit 1 | ||||
} | ||||
Thomas Arendsen Hein
|
r795 | |||
levon@movementarian.org
|
r1434 | $DIFF -u "$BASE" "$OTHER" > "$HGTMP/diff" || : | ||
if $PATCH "$LOCAL" < "$HGTMP/diff"; then | ||||
Thomas Arendsen Hein
|
r831 | exit 0 | ||
Thomas Arendsen Hein
|
r829 | else | ||
Thomas Arendsen Hein
|
r830 | # If rejects are empty after using the editor, merge was ok | ||
Thomas Arendsen Hein
|
r831 | $EDITOR "$LOCAL" "$LOCAL.rej" && test -s "$LOCAL.rej" || exit 0 | ||
mpm@selenic.com
|
r240 | fi | ||
Thomas Arendsen Hein
|
r831 | exit 1 | ||
mpm@selenic.com
|
r240 | fi | ||
echo "hgmerge: unable to find merge, tkdiff, kdiff3, or diff+patch!" | ||||
exit 1 | ||||