[PATCH] Multi-head fetch.
Traditionally, fetch takes these forms: $ git fetch <remote> $ git fetch <remote> <head> $ git fetch <remote> tag <tag> This patch updates it to take $ git fetch <remote> <refspec>... where: - A <refspec> of form "<src>:<dst>" is to fetch the objects needed for the remote ref that matches <src>, and if <dst> is not empty, store it as a local <dst>. - "tag" followed by <next> is just an old way of saying "refs/tags/<next>:refs/tags/<next>"; this mimics the current behaviour of the third form above and means "fetch that tag and store it under the same name". - A single token <refspec> without colon is a shorthand for "<refspec>:" That is, "fetch that ref but do not store anywhere". - when there is no <refspec> specified - if <remote> is the name of a file under $GIT_DIR/remotes/ (i.e. a new-style shorthand), then it is the same as giving the <refspec>s listed on Pull: line in that file. - if <remote> is the name of a file under $GIT_DIR/branches/ (i.e. an old-style shorthand, without trailing path), then it is the same as giving a single <refspec> "<remote-name>:refs/heads/<remote>" on the command line, where <remote-name> is the remote branch name (defaults to HEAD, but can be overridden by .git/branches/<remote> file having the URL fragment notation). That is, "fetch that branch head and store it in refs/heads/<remote>". - otherwise, it is the same as giving a single <refspec> that is "HEAD:". The SHA1 object names of fetched refs are stored in FETCH_HEAD, one name per line, with a comment to describe where it came from. This is later used by "git resolve" and "git octopus". Signed-off-by: Junio C Hamano <junkio@cox.net>
This commit is contained in:
parent
ac4b0cff00
commit
853a3697dc
191
git-fetch-script
191
git-fetch-script
@ -1,54 +1,159 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
#
|
#
|
||||||
. git-sh-setup-script || die "Not a git archive"
|
. git-sh-setup-script || die "Not a git archive"
|
||||||
. git-parse-remote "$@"
|
. git-parse-remote-script
|
||||||
merge_repo="$_remote_repo"
|
_x40='[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]'
|
||||||
merge_head="$_remote_head"
|
_x40="$_x40$_x40$_x40$_x40$_x40$_x40$_x40$_x40"
|
||||||
merge_store="$_remote_store"
|
|
||||||
|
|
||||||
TMP_HEAD="$GIT_DIR/TMP_HEAD"
|
append=
|
||||||
|
case "$#" in
|
||||||
|
0)
|
||||||
|
die "Where do you want to fetch from?" ;;
|
||||||
|
*)
|
||||||
|
case "$1" in
|
||||||
|
-a|--a|--ap|--app|--appe|--appen|--append)
|
||||||
|
append=t
|
||||||
|
shift ;;
|
||||||
|
esac
|
||||||
|
esac
|
||||||
|
remote_nick="$1"
|
||||||
|
remote=$(get_remote_url "$@")
|
||||||
|
refs=
|
||||||
|
rref=
|
||||||
|
rsync_slurped_objects=
|
||||||
|
|
||||||
case "$merge_repo" in
|
if test "" = "$append"
|
||||||
http://* | https://*)
|
then
|
||||||
if [ -n "$GIT_SSL_NO_VERIFY" ]; then
|
: >$GIT_DIR/FETCH_HEAD
|
||||||
curl_extra_args="-k"
|
fi
|
||||||
fi
|
|
||||||
_x40='[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]' &&
|
append_fetch_head () {
|
||||||
_x40="$_x40$_x40$_x40$_x40$_x40$_x40$_x40$_x40" &&
|
head_="$1"
|
||||||
head=$(curl -nsf $curl_extra_args "$merge_repo/$merge_head") &&
|
remote_="$2"
|
||||||
expr "$head" : "$_x40\$" >/dev/null || {
|
remote_name_="$3"
|
||||||
echo >&2 "Failed to fetch $merge_head from $merge_repo"
|
remote_nick_="$4"
|
||||||
exit 1
|
local_name_="$5"
|
||||||
}
|
|
||||||
echo Fetching "$merge_head" using http
|
# 2.6.11-tree tag would not be happy to be fed to resolve.
|
||||||
git-http-pull -v -a "$head" "$merge_repo/" || exit
|
if git-cat-file commit "$head_" >/dev/null 2>&1
|
||||||
|
then
|
||||||
|
head_=$(git-rev-parse --verify "$head_^0") || exit
|
||||||
|
note_="$head_ $remote_name_ from $remote_nick_"
|
||||||
|
echo "$note_" >>$GIT_DIR/FETCH_HEAD
|
||||||
|
echo >&2 "* committish: $note_"
|
||||||
|
else
|
||||||
|
echo >&2 "* non-commit: $note_"
|
||||||
|
fi
|
||||||
|
if test "$local_name_" != ""
|
||||||
|
then
|
||||||
|
# We are storing the head locally. Make sure that it is
|
||||||
|
# a fast forward (aka "reverse push").
|
||||||
|
fast_forward_local "$local_name_" "$head_" "$remote_" "$remote_name_"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
fast_forward_local () {
|
||||||
|
case "$1" in
|
||||||
|
refs/tags/*)
|
||||||
|
# Tags need not be pointing at commits so there
|
||||||
|
# is no way to guarantee "fast-forward" anyway.
|
||||||
|
echo "$2" >"$GIT_DIR/$1" ;;
|
||||||
|
refs/heads/*)
|
||||||
|
# NEEDSWORK: use the same cmpxchg protocol here.
|
||||||
|
echo "$2" >"$GIT_DIR/$1.lock"
|
||||||
|
if test -f "$GIT_DIR/$1"
|
||||||
|
then
|
||||||
|
local=$(git-rev-parse --verify "$1^0") &&
|
||||||
|
mb=$(git-merge-base "$local" "$2") &&
|
||||||
|
case "$2,$mb" in
|
||||||
|
$local,*)
|
||||||
|
echo >&2 "* $1: same as $4"
|
||||||
|
echo >&2 " from $3"
|
||||||
|
;;
|
||||||
|
*,$local)
|
||||||
|
echo >&2 "* $1: fast forward to $4"
|
||||||
|
echo >&2 " from $3"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
false
|
||||||
|
;;
|
||||||
|
esac || {
|
||||||
|
mv "$GIT_DIR/$1.lock" "$GIT_DIR/$1.remote"
|
||||||
|
echo >&2 "* $1: does not fast forward to $4"
|
||||||
|
echo >&2 " from $3; leaving it in '$1.remote'"
|
||||||
|
}
|
||||||
|
else
|
||||||
|
echo >&2 "* $1: storing $4"
|
||||||
|
echo >&2 " from $3."
|
||||||
|
fi
|
||||||
|
test -f "$GIT_DIR/$1.lock" &&
|
||||||
|
mv "$GIT_DIR/$1.lock" "$GIT_DIR/$1"
|
||||||
;;
|
;;
|
||||||
rsync://*)
|
esac
|
||||||
rsync -L "$merge_repo/$merge_head" "$TMP_HEAD" || exit 1
|
}
|
||||||
|
|
||||||
|
for ref in $(get_remote_refs_for_fetch "$@")
|
||||||
|
do
|
||||||
|
refs="$refs $ref"
|
||||||
|
|
||||||
|
# These are relative path from $GIT_DIR, typically starting at refs/
|
||||||
|
# but may be HEAD
|
||||||
|
remote_name=$(expr "$ref" : '\([^:]*\):')
|
||||||
|
local_name=$(expr "$ref" : '[^:]*:\(.*\)')
|
||||||
|
|
||||||
|
rref="$rref $remote_name"
|
||||||
|
|
||||||
|
# There are transports that can fetch only one head at a time...
|
||||||
|
case "$remote" in
|
||||||
|
http://* | https://*)
|
||||||
|
if [ -n "$GIT_SSL_NO_VERIFY" ]; then
|
||||||
|
curl_extra_args="-k"
|
||||||
|
fi
|
||||||
|
head=$(curl -nsf $curl_extra_args "$remote/$remote_name") &&
|
||||||
|
expr "$head" : "$_x40\$" >/dev/null ||
|
||||||
|
die "Failed to fetch $remote_name from $remote"
|
||||||
|
echo Fetching "$remote_name from $remote" using http
|
||||||
|
git-http-pull -v -a "$head" "$remote/" || exit
|
||||||
|
;;
|
||||||
|
rsync://*)
|
||||||
|
TMP_HEAD="$GIT_DIR/TMP_HEAD"
|
||||||
|
rsync -L "$remote/$remote_name" "$TMP_HEAD" || exit 1
|
||||||
head=$(git-rev-parse TMP_HEAD)
|
head=$(git-rev-parse TMP_HEAD)
|
||||||
rm -f "$TMP_HEAD"
|
rm -f "$TMP_HEAD"
|
||||||
rsync -avz --ignore-existing "$merge_repo/objects/" "$GIT_OBJECT_DIRECTORY/"
|
test "$rsync_slurped_objects" || {
|
||||||
|
rsync -avz --ignore-existing "$remote/objects/" \
|
||||||
|
"$GIT_OBJECT_DIRECTORY/" || exit
|
||||||
|
rsync_slurped_objects=t
|
||||||
|
}
|
||||||
;;
|
;;
|
||||||
|
*)
|
||||||
|
# We will do git native transport with just one call later.
|
||||||
|
continue ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
append_fetch_head "$head" "$remote" "$remote_name" "$remote_nick" "$local_name"
|
||||||
|
|
||||||
|
done
|
||||||
|
|
||||||
|
case "$remote" in
|
||||||
|
http://* | https://* | rsync://* )
|
||||||
|
;; # we are already done.
|
||||||
*)
|
*)
|
||||||
head=$(git-fetch-pack "$merge_repo" "$merge_head")
|
git-fetch-pack "$remote" $rref |
|
||||||
if h=`expr "$head" : '\([^ ][^ ]*\) '`
|
while read sha1 remote_name
|
||||||
then
|
do
|
||||||
head=$h
|
found=
|
||||||
fi
|
for ref in $refs
|
||||||
;;
|
do
|
||||||
esac || exit 1
|
case "$ref" in
|
||||||
|
$remote_name:*)
|
||||||
|
found="$ref"
|
||||||
|
break ;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
git-rev-parse --verify "$head" > /dev/null || exit 1
|
local_name=$(expr "$found" : '[^:]*:\(.*\)')
|
||||||
|
append_fetch_head "$sha1" "$remote" "$remote_name" "$remote_nick" "$local_name"
|
||||||
case "$merge_store" in
|
done
|
||||||
'')
|
;;
|
||||||
;;
|
esac
|
||||||
*)
|
|
||||||
echo "$head" > "$GIT_DIR/$merge_store"
|
|
||||||
esac &&
|
|
||||||
|
|
||||||
# FETCH_HEAD is fed to git-resolve-script which will eventually be
|
|
||||||
# passed to git-commit-tree as one of the parents. Make sure we do
|
|
||||||
# not give a tag object ID.
|
|
||||||
|
|
||||||
git-rev-parse "$head^0" >"$GIT_DIR/FETCH_HEAD"
|
|
||||||
|
Loading…
Reference in New Issue
Block a user