154 lines
4.7 KiB
Bash
Executable file
154 lines
4.7 KiB
Bash
Executable file
#!/bin/sh
|
|
|
|
TOPDIR=..
|
|
|
|
set -x
|
|
|
|
# remember the refs that were uploaded already
|
|
git for-each-ref refs/namespaces/mine/ > .git/old-refs
|
|
|
|
# Unfortunately, git bundle omits prerequisites that are omitted once,
|
|
# even if they are used by a later ref.
|
|
# For example, where x is a ref that points at A, and y is a ref
|
|
# that points at B (which has A as its parent), git bundle x A..y
|
|
# will omit inclding the x ref in the bundle at all.
|
|
check_prereq () {
|
|
# So, if a sha is one of the other refs that will be included in the
|
|
# bundle, it cannot be treated as a prerequisite.
|
|
if git for-each-ref refs/namespaces/mine/ | grep -Pv "\t$2$" | awk '{print $1}' | grep -q "$1"; then
|
|
echo "$2"
|
|
else
|
|
# And, if one of the other refs that will be included in the bundle
|
|
# is an ancestor of the sha, it cannot be treated as a prerequisite.
|
|
if [ -n "$(for x in $(git for-each-ref refs/namespaces/mine/ | grep -Pv "\t$2$" | awk '{print $1}'); do git log --oneline -n1 $x..$1; done)" ]; then
|
|
echo "$2"
|
|
else
|
|
echo "$1..$2"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
while read foo; do
|
|
case "$foo" in
|
|
capabilities)
|
|
echo fetch
|
|
echo push
|
|
echo
|
|
;;
|
|
list*)
|
|
if [ -e "$TOPDIR/MANIFEST" ]; then
|
|
# Only list the refs in the last bundle
|
|
# listed in the manifest. Each push
|
|
# includes all refs in its bundle.
|
|
f=$(tail -n 1 $TOPDIR/MANIFEST)
|
|
if [ -n "$f" ]; then
|
|
# refs in the bundle may end up prefixed with refs/namespaces/mine/
|
|
# when the intent is for the bundle to include a
|
|
# ref with the name that comes after that.
|
|
git bundle list-heads $TOPDIR/$f | sed 's/refs\/namespaces\/mine\///'
|
|
fi
|
|
fi
|
|
echo
|
|
;;
|
|
fetch*)
|
|
dofetch=1
|
|
;;
|
|
push*)
|
|
set -- $foo
|
|
x="$2"
|
|
# src ref if prefixed with a + in a forced push
|
|
srcref="$(echo "$x" | cut -d : -f 1 | sed 's/^\+//')"
|
|
dstref="$(echo "$x" | cut -d : -f 2)"
|
|
if [ -z "$srcref" ]; then
|
|
git update-ref -d refs/namespaces/mine/"$dstref"
|
|
else
|
|
# Need to create a bundle containing $dstref, but
|
|
# don't want to overwrite that ref in the local
|
|
# repo. Unfortunately, git bundle does not support
|
|
# GIT_NAMESPACE, so it's not possible to do that
|
|
# without making a clone of the whole git repo.
|
|
# Instead, just create a ref under the namespace
|
|
# refs/namespaces/mine/ that will be put in the
|
|
# bundle.
|
|
git update-ref refs/namespaces/mine/"$dstref" "$srcref"
|
|
fi
|
|
dopush=1
|
|
;;
|
|
# docs say a blank line ends communication, but that's not
|
|
# accurate, actually a blank line comes after a series of
|
|
# fetch or push commands, and also according to the docs,
|
|
# another series of commands could follow
|
|
"")
|
|
if [ "$dofetch" ]; then
|
|
if [ -e "$TOPDIR/MANIFEST" ]; then
|
|
for f in $(cat $TOPDIR/MANIFEST); do
|
|
git bundle unbundle "$TOPDIR/$f" >/dev/null 2>&1
|
|
done
|
|
fi
|
|
echo
|
|
dofetch=""
|
|
fi
|
|
if [ "$dopush" ]; then
|
|
if [ -z "$(git for-each-ref refs/namespaces/mine/)" ]; then
|
|
# deleted all refs
|
|
if [ -e "$TOPDIR/MANIFEST" ]; then
|
|
for f in $(cat $TOPDIR/MANIFEST); do
|
|
rm "$TOPDIR/$f"
|
|
done
|
|
rm $TOPDIR/MANIFEST
|
|
touch $TOPDIR/MANIFEST
|
|
fi
|
|
else
|
|
# set REPUSH=1 to do a full push
|
|
# rather than incremental
|
|
if [ "$REPUSH" ]; then
|
|
rm $TOPDIR/MANIFEST
|
|
rm $TOPDIR/*.bundle
|
|
git for-each-ref refs/namespaces/mine/ | awk '{print $3}' | \
|
|
git bundle create --quiet $TOPDIR/new.bundle --stdin
|
|
else
|
|
# incremental bundle
|
|
IFS="
|
|
"
|
|
(for l in $(git for-each-ref refs/namespaces/mine/); do
|
|
r=$(echo "$l" | awk '{print $3}')
|
|
newsha=$(echo "$l" | awk '{print $1}')
|
|
oldsha=$(grep -P "\t$r$" .git/old-refs | awk '{print $1}')
|
|
if [ -n "$oldsha" ]; then
|
|
# include changes from $oldsha to $r when there are some
|
|
if [ -n "$(git log --oneline $oldsha..$r)" ]; then
|
|
check_prereq "$oldsha" "$r"
|
|
else
|
|
if [ "$oldsha" = "$newsha" ]; then
|
|
# $r is unchanged from last push, so include
|
|
# the minimum data to make the bundle contain $r
|
|
rparentsha=$(git log -n 2 "$r" --format='%H' | tail -n+2)
|
|
if [ -n "$rparentsha" ]; then
|
|
check_prereq "$rparentsha" "$r"
|
|
else
|
|
# $r has no parent so include it as is
|
|
echo "$r"
|
|
fi
|
|
else
|
|
# $oldsha is not a parent of $r, so
|
|
# include $r and all its parents
|
|
echo "$r"
|
|
fi
|
|
fi
|
|
else
|
|
# no old version was pushed so include $r and all its parents
|
|
echo "$r"
|
|
fi
|
|
done) \
|
|
| git bundle create --quiet $TOPDIR/new.bundle --stdin
|
|
fi
|
|
sha1=$(sha1sum $TOPDIR/new.bundle | awk '{print $1}')
|
|
mv $TOPDIR/new.bundle "$TOPDIR/$sha1.bundle"
|
|
echo "$sha1.bundle" >> $TOPDIR/MANIFEST
|
|
fi
|
|
echo
|
|
dopush=""
|
|
fi
|
|
;;
|
|
esac
|
|
done
|