41f222e87a
The git philosophy when it comes to disk accesses is "Laugh in the face of danger". Notably, since we never modify an existing object, we don't really care that deeply about flushing things to disk, since even if the machine crashes in the middle of a git operation, you can never really have lost any old work. At most, you'd need to figure out the proper heads (which git-fsck-objects can do for you) and re-do the operation. However, there's two exceptions to this: pruning and repacking. Those operations will actually _delete_ old objects that they know about in other ways (ie that they just repacked, or that they have found in other places). However, since they actually modify old state, we should thus be a bit more careful about them. If the machine crashes and the duplicate new objects haven't been flushed to disk, you can actually be in trouble. This is trivially stupid about it by calling "sync" before removing the objects. Not very smart, but we're talking about special operations than are usually done once a week if that. Signed-off-by: Linus Torvalds <torvalds@osdl.org> Signed-off-by: Junio C Hamano <junkio@cox.net>
82 lines
1.6 KiB
Bash
Executable File
82 lines
1.6 KiB
Bash
Executable File
#!/bin/sh
|
|
#
|
|
# Copyright (c) 2005 Linus Torvalds
|
|
#
|
|
|
|
. git-sh-setup || die "Not a git archive"
|
|
|
|
no_update_info= all_into_one= remove_redundant= local=
|
|
while case "$#" in 0) break ;; esac
|
|
do
|
|
case "$1" in
|
|
-n) no_update_info=t ;;
|
|
-a) all_into_one=t ;;
|
|
-d) remove_redandant=t ;;
|
|
-l) local=t ;;
|
|
*) break ;;
|
|
esac
|
|
shift
|
|
done
|
|
|
|
rm -f .tmp-pack-*
|
|
PACKDIR="$GIT_OBJECT_DIRECTORY/pack"
|
|
|
|
# There will be more repacking strategies to come...
|
|
case ",$all_into_one," in
|
|
,,)
|
|
rev_list='--unpacked'
|
|
rev_parse='--all'
|
|
pack_objects='--incremental'
|
|
;;
|
|
,t,)
|
|
rev_list=
|
|
rev_parse='--all'
|
|
pack_objects=
|
|
# This part is a stop-gap until we have proper pack redundancy
|
|
# checker.
|
|
existing=`cd "$PACKDIR" && \
|
|
find . -type f \( -name '*.pack' -o -name '*.idx' \) -print`
|
|
;;
|
|
esac
|
|
if [ "$local" ]; then
|
|
pack_objects="$pack_objects --local"
|
|
fi
|
|
name=$(git-rev-list --objects $rev_list $(git-rev-parse $rev_parse) |
|
|
git-pack-objects --non-empty $pack_objects .tmp-pack) ||
|
|
exit 1
|
|
if [ -z "$name" ]; then
|
|
echo Nothing new to pack.
|
|
exit 0
|
|
fi
|
|
echo "Pack pack-$name created."
|
|
|
|
mkdir -p "$PACKDIR" || exit
|
|
|
|
mv .tmp-pack-$name.pack "$PACKDIR/pack-$name.pack" &&
|
|
mv .tmp-pack-$name.idx "$PACKDIR/pack-$name.idx" ||
|
|
exit
|
|
|
|
if test "$remove_redandant" = t
|
|
then
|
|
# We know $existing are all redandant only when
|
|
# all-into-one is used.
|
|
if test "$all_into_one" != '' && test "$existing" != ''
|
|
then
|
|
sync
|
|
( cd "$PACKDIR" &&
|
|
for e in $existing
|
|
do
|
|
case "$e" in
|
|
./pack-$name.pack | ./pack-$name.idx) ;;
|
|
*) rm -f $e ;;
|
|
esac
|
|
done
|
|
)
|
|
fi
|
|
fi
|
|
|
|
case "$no_update_info" in
|
|
t) : ;;
|
|
*) git-update-server-info ;;
|
|
esac
|