You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
207 lines
4.6 KiB
207 lines
4.6 KiB
#!/bin/sh |
|
# |
|
# Copyright (c) 2005 Linus Torvalds |
|
# Copyright (c) 2005 Junio C Hamano |
|
# |
|
|
|
case "$0" in |
|
*-revert* ) |
|
test -t 0 && edit=-e |
|
replay= |
|
me=revert |
|
USAGE='[--edit | --no-edit] [-n] <commit-ish>' ;; |
|
*-cherry-pick* ) |
|
replay=t |
|
edit= |
|
me=cherry-pick |
|
USAGE='[--edit] [-n] [-r] [-x] <commit-ish>' ;; |
|
* ) |
|
echo >&2 "What are you talking about?" |
|
exit 1 ;; |
|
esac |
|
|
|
SUBDIRECTORY_OK=Yes ;# we will cd up |
|
. git-sh-setup |
|
require_work_tree |
|
cd_to_toplevel |
|
|
|
no_commit= |
|
xopt= |
|
while case "$#" in 0) break ;; esac |
|
do |
|
case "$1" in |
|
-n|--n|--no|--no-|--no-c|--no-co|--no-com|--no-comm|\ |
|
--no-commi|--no-commit) |
|
no_commit=t |
|
;; |
|
-e|--e|--ed|--edi|--edit) |
|
edit=-e |
|
;; |
|
--n|--no|--no-|--no-e|--no-ed|--no-edi|--no-edit) |
|
edit= |
|
;; |
|
-r) |
|
: no-op ;; |
|
-x|--i-really-want-to-expose-my-private-commit-object-name) |
|
replay= |
|
;; |
|
-X?*) |
|
xopt="$xopt$(git rev-parse --sq-quote "--${1#-X}")" |
|
;; |
|
--strategy-option=*) |
|
xopt="$xopt$(git rev-parse --sq-quote "--${1#--strategy-option=}")" |
|
;; |
|
-X|--strategy-option) |
|
shift |
|
xopt="$xopt$(git rev-parse --sq-quote "--$1")" |
|
;; |
|
-*) |
|
usage |
|
;; |
|
*) |
|
break |
|
;; |
|
esac |
|
shift |
|
done |
|
|
|
set_reflog_action "$me" |
|
|
|
test "$me,$replay" = "revert,t" && usage |
|
|
|
case "$no_commit" in |
|
t) |
|
# We do not intend to commit immediately. We just want to |
|
# merge the differences in. |
|
head=$(git-write-tree) || |
|
die "Your index file is unmerged." |
|
;; |
|
*) |
|
head=$(git-rev-parse --verify HEAD) || |
|
die "You do not have a valid HEAD" |
|
files=$(git-diff-index --cached --name-only $head) || exit |
|
if [ "$files" ]; then |
|
die "Dirty index: cannot $me (dirty: $files)" |
|
fi |
|
;; |
|
esac |
|
|
|
rev=$(git-rev-parse --verify "$@") && |
|
commit=$(git-rev-parse --verify "$rev^0") || |
|
die "Not a single commit $@" |
|
prev=$(git-rev-parse --verify "$commit^1" 2>/dev/null) || |
|
die "Cannot run $me a root commit" |
|
git-rev-parse --verify "$commit^2" >/dev/null 2>&1 && |
|
die "Cannot run $me a multi-parent commit." |
|
|
|
encoding=$(git config i18n.commitencoding || echo UTF-8) |
|
|
|
# "commit" is an existing commit. We would want to apply |
|
# the difference it introduces since its first parent "prev" |
|
# on top of the current HEAD if we are cherry-pick. Or the |
|
# reverse of it if we are revert. |
|
|
|
case "$me" in |
|
revert) |
|
git show -s --pretty=oneline --encoding="$encoding" $commit | |
|
sed -e ' |
|
s/^[^ ]* /Revert "/ |
|
s/$/"/ |
|
' |
|
echo |
|
echo "This reverts commit $commit." |
|
test "$rev" = "$commit" || |
|
echo "(original 'git revert' arguments: $@)" |
|
base=$commit next=$prev |
|
;; |
|
|
|
cherry-pick) |
|
pick_author_script=' |
|
/^author /{ |
|
s/'\''/'\''\\'\'\''/g |
|
h |
|
s/^author \([^<]*\) <[^>]*> .*$/\1/ |
|
s/'\''/'\''\'\'\''/g |
|
s/.*/GIT_AUTHOR_NAME='\''&'\''/p |
|
|
|
g |
|
s/^author [^<]* <\([^>]*\)> .*$/\1/ |
|
s/'\''/'\''\'\'\''/g |
|
s/.*/GIT_AUTHOR_EMAIL='\''&'\''/p |
|
|
|
g |
|
s/^author [^<]* <[^>]*> \(.*\)$/\1/ |
|
s/'\''/'\''\'\'\''/g |
|
s/.*/GIT_AUTHOR_DATE='\''&'\''/p |
|
|
|
q |
|
}' |
|
|
|
logmsg=$(git show -s --pretty=raw --encoding="$encoding" "$commit") |
|
set_author_env=$(echo "$logmsg" | |
|
LANG=C LC_ALL=C sed -ne "$pick_author_script") |
|
eval "$set_author_env" |
|
export GIT_AUTHOR_NAME |
|
export GIT_AUTHOR_EMAIL |
|
export GIT_AUTHOR_DATE |
|
|
|
echo "$logmsg" | |
|
sed -e '1,/^$/d' -e 's/^ //' |
|
case "$replay" in |
|
'') |
|
echo "(cherry picked from commit $commit)" |
|
test "$rev" = "$commit" || |
|
echo "(original 'git cherry-pick' arguments: $@)" |
|
;; |
|
esac |
|
base=$prev next=$commit |
|
;; |
|
|
|
esac >.msg |
|
|
|
eval GITHEAD_$head=HEAD |
|
eval GITHEAD_$next='$(git show -s \ |
|
--pretty=oneline --encoding="$encoding" "$commit" | |
|
sed -e "s/^[^ ]* //")' |
|
export GITHEAD_$head GITHEAD_$next |
|
|
|
# This three way merge is an interesting one. We are at |
|
# $head, and would want to apply the change between $commit |
|
# and $prev on top of us (when reverting), or the change between |
|
# $prev and $commit on top of us (when cherry-picking or replaying). |
|
|
|
eval "git merge-recursive $xopt $base -- $head $next" && |
|
result=$(git-write-tree 2>/dev/null) || { |
|
mv -f .msg "$GIT_DIR/MERGE_MSG" |
|
{ |
|
echo ' |
|
Conflicts: |
|
' |
|
git ls-files --unmerged | |
|
sed -e 's/^[^ ]* / /' | |
|
uniq |
|
} >>"$GIT_DIR/MERGE_MSG" |
|
echo >&2 "Automatic $me failed. After resolving the conflicts," |
|
echo >&2 "mark the corrected paths with 'git-add <paths>'" |
|
echo >&2 "and commit the result." |
|
case "$me" in |
|
cherry-pick) |
|
echo >&2 "You may choose to use the following when making" |
|
echo >&2 "the commit:" |
|
echo >&2 "$set_author_env" |
|
esac |
|
exit 1 |
|
} |
|
|
|
# If we are cherry-pick, and if the merge did not result in |
|
# hand-editing, we will hit this commit and inherit the original |
|
# author date and name. |
|
# If we are revert, or if our cherry-pick results in a hand merge, |
|
# we had better say that the current user is responsible for that. |
|
|
|
case "$no_commit" in |
|
'') |
|
git-commit -n -F .msg $edit |
|
rm -f .msg |
|
;; |
|
esac
|
|
|