2018-05-17 20:42:23 +02:00
|
|
|
#!/bin/sh -e
|
|
|
|
|
|
|
|
[ "$GIT_CACHE_VERBOSE" != "1" ] && Q=-q
|
2016-03-20 12:40:20 +01:00
|
|
|
|
|
|
|
git_cache() {
|
|
|
|
git -C "${GIT_CACHE_DIR}" $*
|
|
|
|
}
|
|
|
|
|
2017-01-10 21:43:33 +01:00
|
|
|
git_cache_initialized() {
|
|
|
|
local _git_dir="$(git_cache rev-parse --git-dir 2>/dev/null)"
|
|
|
|
test "$_git_dir" = "." -o "$_git_dir" = ".git"
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
init() {
|
2017-01-10 21:43:33 +01:00
|
|
|
git_cache_initialized || {
|
2016-03-20 12:40:20 +01:00
|
|
|
mkdir -p "${GIT_CACHE_DIR}"
|
|
|
|
|
|
|
|
git_cache init --bare
|
|
|
|
git_cache config core.compression 1
|
|
|
|
}
|
2018-05-17 20:42:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
startswith() {
|
|
|
|
case "$1" in
|
|
|
|
"${2}"*)
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
return 1
|
|
|
|
esac
|
2016-03-20 12:40:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
add() {
|
2017-01-10 21:43:33 +01:00
|
|
|
local repo="$1"
|
|
|
|
|
|
|
|
_locked "$GIT_CACHE_DIR/$name.addlock" _add "$repo"
|
|
|
|
}
|
|
|
|
|
|
|
|
_add() {
|
2016-12-27 16:29:22 +01:00
|
|
|
local repo="$1"
|
|
|
|
local name="$(_remote_name $repo)"
|
2016-12-15 10:59:47 +01:00
|
|
|
|
|
|
|
if ! is_cached "$repo"; then
|
|
|
|
git_cache remote add "$name" "$repo"
|
2017-01-10 21:43:33 +01:00
|
|
|
git_cache config --add remote.${name}.fetch "+refs/tags/*:refs/tags/${name}/*"
|
2016-12-15 10:59:47 +01:00
|
|
|
else
|
|
|
|
echo "git-cache: $url already in cache"
|
|
|
|
fi
|
2016-03-20 12:40:20 +01:00
|
|
|
}
|
|
|
|
|
2017-01-12 17:31:57 +01:00
|
|
|
if [ "$(uname)" = Darwin ]; then
|
|
|
|
_locked() {
|
|
|
|
local lockfile="$1"
|
|
|
|
shift
|
2017-01-10 21:43:33 +01:00
|
|
|
|
2017-01-12 17:31:57 +01:00
|
|
|
while ! shlock -p $$ -f $lockfile; do
|
|
|
|
sleep 0.2
|
|
|
|
done
|
|
|
|
|
|
|
|
$*
|
|
|
|
|
|
|
|
rm $lockfile
|
|
|
|
}
|
|
|
|
else
|
|
|
|
_locked() {
|
|
|
|
local lockfile="$1"
|
|
|
|
shift
|
|
|
|
|
|
|
|
(
|
2017-01-10 21:43:33 +01:00
|
|
|
flock -w 600 9 || exit 1
|
|
|
|
$*
|
2017-01-12 17:31:57 +01:00
|
|
|
) 9>"$lockfile"
|
|
|
|
}
|
|
|
|
fi
|
2017-01-10 21:43:33 +01:00
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
update() {
|
2017-01-10 21:43:33 +01:00
|
|
|
local REMOTE=${1}
|
|
|
|
if [ -n "$REMOTE" ]; then
|
|
|
|
local REMOTES=$(_remote_name $REMOTE)
|
|
|
|
else
|
|
|
|
local REMOTES="$(git_cache remote show)"
|
|
|
|
fi
|
|
|
|
|
|
|
|
for remote in $REMOTES; do
|
|
|
|
echo "git-cache: updating remote $remote"
|
2018-05-17 20:42:23 +02:00
|
|
|
_locked "$GIT_CACHE_DIR/$remote.lock" git_cache --namespace $remote fetch $Q -n $remote
|
2017-01-10 21:43:33 +01:00
|
|
|
done
|
2016-03-20 12:40:20 +01:00
|
|
|
}
|
|
|
|
|
2016-12-15 10:59:47 +01:00
|
|
|
is_cached() {
|
|
|
|
local url="$1"
|
|
|
|
local REMOTES="$(git_cache remote show)"
|
|
|
|
for remote in $REMOTES; do
|
2016-12-27 16:29:22 +01:00
|
|
|
[ "$(git_cache ls-remote --get-url $remote)" = "$url" ] && return 0
|
2016-12-15 10:59:47 +01:00
|
|
|
done
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
list() {
|
|
|
|
local REMOTES="$(git_cache remote show)"
|
|
|
|
for remote in $REMOTES; do
|
2016-12-27 16:29:22 +01:00
|
|
|
echo "$(git_cache ls-remote --get-url $remote)"
|
2016-03-20 12:40:20 +01:00
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
drop() {
|
|
|
|
local REMOTE=${1}
|
|
|
|
[ -z "$REMOTE" ] && {
|
2016-12-27 16:29:22 +01:00
|
|
|
echo "usage: git cache drop <url>"
|
2016-03-20 12:40:20 +01:00
|
|
|
exit 1
|
|
|
|
}
|
2016-12-27 16:29:22 +01:00
|
|
|
local REMOTES="$(git_cache remote show)"
|
|
|
|
for remote in $REMOTES; do
|
|
|
|
[ "$(git_cache ls-remote --get-url $remote)" = "$REMOTE" ] && {
|
|
|
|
git_cache remote remove $remote
|
|
|
|
break
|
|
|
|
}
|
|
|
|
done
|
2016-03-20 12:40:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
_check_commit() {
|
2017-01-10 21:43:33 +01:00
|
|
|
git_cache cat-file -e ${1}^{commit} 2>/dev/null
|
2016-03-20 12:40:20 +01:00
|
|
|
}
|
|
|
|
|
2016-12-15 10:59:47 +01:00
|
|
|
_remote_name() {
|
2017-01-12 17:31:57 +01:00
|
|
|
echo "$*" | git hash-object --stdin
|
2016-12-15 10:59:47 +01:00
|
|
|
}
|
|
|
|
|
2017-01-10 21:43:33 +01:00
|
|
|
_tag_to_sha1() {
|
2018-05-17 20:42:23 +02:00
|
|
|
local out="$(git_cache log -n 1 --pretty=oneline $1 -- 2>/dev/null || true)"
|
2017-01-10 21:43:33 +01:00
|
|
|
[ -n "$out" ] && echo $out | cut -f 1 -d" "
|
|
|
|
}
|
|
|
|
|
|
|
|
_check_tag_or_commit() {
|
|
|
|
local SHA1=$1
|
|
|
|
local REMOTE_NAME=$2
|
|
|
|
|
|
|
|
if _check_commit $SHA1 ; then
|
2019-07-15 12:01:05 +02:00
|
|
|
local tag=commit$SHA1-$$
|
|
|
|
git_cache tag $tag $SHA1 2> /dev/null || true # ignore possibly already existing tag
|
|
|
|
echo "$tag"
|
2017-01-10 21:43:33 +01:00
|
|
|
elif _tag_to_sha1 ${REMOTE_NAME}/$SHA1 > /dev/null; then
|
|
|
|
echo "${REMOTE_NAME}/$SHA1"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
clone() {
|
|
|
|
local REMOTE="${1}"
|
|
|
|
local SHA1="${2}"
|
2016-12-15 10:59:47 +01:00
|
|
|
local REMOTE_NAME="$(_remote_name $REMOTE)"
|
2017-01-10 21:43:33 +01:00
|
|
|
local TARGET_PATH="${3}"
|
|
|
|
|
|
|
|
[ -z "$TARGET_PATH" ] && TARGET_PATH="$(basename $REMOTE)"
|
|
|
|
|
|
|
|
# make sure git won't ask for credentials
|
|
|
|
export GIT_TERMINAL_PROMPT=0
|
2016-03-20 12:40:20 +01:00
|
|
|
|
2017-01-10 21:43:33 +01:00
|
|
|
if git_cache_initialized; then
|
2016-12-15 10:59:47 +01:00
|
|
|
if ! is_cached "$REMOTE"; then
|
2016-12-27 16:29:22 +01:00
|
|
|
echo "git cache: auto-adding $REMOTE"
|
2016-12-15 10:59:47 +01:00
|
|
|
add "$REMOTE"
|
|
|
|
fi
|
|
|
|
|
2018-05-17 20:42:23 +02:00
|
|
|
local pull=0
|
|
|
|
if startswith "$SHA1" "pull/"; then
|
|
|
|
pull=1
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ $pull -eq 0 ]; then
|
|
|
|
local tag="$(_check_tag_or_commit $SHA1 $REMOTE_NAME)"
|
|
|
|
if [ -z "$tag" ]; then
|
|
|
|
# commit / tag not in cache, try updating repo
|
|
|
|
update "$REMOTE"
|
|
|
|
tag="$(_check_tag_or_commit $SHA1 $REMOTE_NAME)"
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
local tag="remotes/$REMOTE_NAME/master"
|
|
|
|
if [ -z "$(_tag_to_sha1 $tag)" ]; then
|
|
|
|
update "$REMOTE"
|
|
|
|
fi
|
|
|
|
tag="$(_check_tag_or_commit $(_tag_to_sha1 $tag) $REMOTE_NAME)"
|
|
|
|
if [ -z "$tag" ]; then
|
|
|
|
echo "git-cache: cannot checkout master branch of $REMOTE"
|
|
|
|
false
|
|
|
|
fi
|
2017-01-10 21:43:33 +01:00
|
|
|
fi
|
|
|
|
|
|
|
|
if [ -n "$tag" ]; then
|
2018-05-17 20:42:23 +02:00
|
|
|
echo "git-cache: cloning from cache. tag=$tag"
|
|
|
|
git -c advice.detachedHead=false clone $Q --reference "${GIT_CACHE_DIR}" --shared "${GIT_CACHE_DIR}" "${TARGET_PATH}" --branch $tag
|
2019-07-15 12:01:05 +02:00
|
|
|
|
|
|
|
# rename tags from <remote-hash>/* to *
|
2018-05-17 20:42:23 +02:00
|
|
|
git -C "${TARGET_PATH}" fetch $Q origin "refs/tags/${REMOTE_NAME}/*:refs/tags/*"
|
2019-07-15 12:01:05 +02:00
|
|
|
|
|
|
|
# remove all commit* and <remote-hash>/* tags
|
|
|
|
git -C "${TARGET_PATH}" tag -l \
|
|
|
|
| grep -P '(^[a-f0-9]{40}/|^commit[a-f0-9]{40}(-\d+)?$)' \
|
|
|
|
| xargs git -C "${TARGET_PATH}" tag -d > /dev/null
|
|
|
|
|
|
|
|
# cleanup possibly created helper tag
|
|
|
|
case $tag in
|
|
|
|
commit*)
|
|
|
|
git_cache tag -d $tag 2>&1 > /dev/null || true
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
|
2018-05-17 20:42:23 +02:00
|
|
|
if [ $pull -eq 1 ]; then
|
|
|
|
git -C "${TARGET_PATH}" fetch $Q $REMOTE $SHA1:$SHA1
|
|
|
|
git -C "${TARGET_PATH}" checkout $Q $SHA1
|
|
|
|
fi
|
2017-01-10 21:43:33 +01:00
|
|
|
else
|
|
|
|
echo "git-cache: trying checkout from source"
|
2018-05-17 20:42:23 +02:00
|
|
|
git clone $Q --reference "${GIT_CACHE_DIR}" --shared "${REMOTE}" "${TARGET_PATH}"
|
|
|
|
git -c advice.detachedHead=false -C "${TARGET_PATH}" checkout $Q $SHA1
|
2017-01-10 21:43:33 +01:00
|
|
|
fi
|
2016-03-20 12:40:20 +01:00
|
|
|
else
|
2017-01-10 21:43:33 +01:00
|
|
|
git clone "${REMOTE}" "${TARGET_PATH}"
|
|
|
|
git -c advice.detachedHead=false -C "${TARGET_PATH}" checkout $SHA1
|
2016-03-20 12:40:20 +01:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2019-07-15 12:01:05 +02:00
|
|
|
cleanup() {
|
|
|
|
git_cache tag -l \
|
|
|
|
| grep -P '(^commit[a-f0-9]{40}(-\d+)?$)' \
|
|
|
|
| xargs git -C "${GIT_CACHE_DIR}" tag -d > /dev/null
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
usage() {
|
|
|
|
echo "git cache uses a bare git repository containing all objects from multiple"
|
|
|
|
echo "upstream git repositories."
|
|
|
|
echo ""
|
|
|
|
echo "usage:"
|
|
|
|
echo ""
|
|
|
|
echo " git cache init initialize git cache"
|
2016-12-27 16:29:22 +01:00
|
|
|
echo " git cache add <url> add repository <url>"
|
2016-03-20 12:40:20 +01:00
|
|
|
echo " git cache list list cached repositories"
|
2016-12-27 16:29:22 +01:00
|
|
|
echo " git cache drop <url> drop repo from cache"
|
|
|
|
echo " git cache update [<url>] fetch repo <url> (or all)"
|
2016-03-20 12:40:20 +01:00
|
|
|
echo " git cache clone <url> <SHA1> clone repository <url> from cache"
|
|
|
|
echo " git cache show-path print's the path that can be used as "
|
|
|
|
echo " '--reference' parameter"
|
2019-07-15 12:01:05 +02:00
|
|
|
echo " git cache cleanup cleanup dangling temporary tags"
|
|
|
|
echo " (appear if git-cache gets inter-"
|
|
|
|
echo " rupted, but are harmless)"
|
2016-03-20 12:40:20 +01:00
|
|
|
echo ""
|
|
|
|
echo "To retrieve objects from cache (will use remote repository if needed):"
|
|
|
|
echo ' git clone --reference $(git cache show-path) <repo>'
|
|
|
|
}
|
|
|
|
|
2016-12-27 16:29:22 +01:00
|
|
|
[ $# -eq 0 ] && {
|
|
|
|
usage
|
|
|
|
exit 1
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:40:20 +01:00
|
|
|
ACTION=$1
|
2016-12-27 16:29:22 +01:00
|
|
|
shift 1
|
2016-03-20 12:40:20 +01:00
|
|
|
|
|
|
|
export GIT_CACHE_DIR=${GIT_CACHE_DIR:-${HOME}/.gitcache}
|
|
|
|
|
|
|
|
case $ACTION in
|
|
|
|
init)
|
|
|
|
init $*
|
|
|
|
;;
|
|
|
|
add)
|
|
|
|
add $*
|
|
|
|
;;
|
|
|
|
update)
|
|
|
|
update $*
|
|
|
|
;;
|
|
|
|
list)
|
|
|
|
list $*
|
|
|
|
;;
|
|
|
|
drop)
|
|
|
|
drop $*
|
|
|
|
;;
|
|
|
|
show-path)
|
|
|
|
echo ${GIT_CACHE_DIR}
|
|
|
|
;;
|
|
|
|
clone)
|
|
|
|
clone $*
|
|
|
|
;;
|
2019-07-15 12:01:05 +02:00
|
|
|
cleanup)
|
|
|
|
cleanup
|
|
|
|
;;
|
2016-03-20 12:40:20 +01:00
|
|
|
*)
|
|
|
|
usage
|
|
|
|
;;
|
|
|
|
esac
|