403 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			403 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable file
		
	
	
	
	
#!/bin/sh
 | 
						|
#
 | 
						|
# Copyright (c) 2008 Nicolas Pitre
 | 
						|
#
 | 
						|
 | 
						|
test_description='resilience to pack corruptions with redundant objects'
 | 
						|
. ./test-lib.sh
 | 
						|
 | 
						|
# Note: the test objects are created with knowledge of their pack encoding
 | 
						|
# to ensure good code path coverage, and to facilitate direct alteration
 | 
						|
# later on.  The assumed characteristics are:
 | 
						|
#
 | 
						|
# 1) blob_2 is a delta with blob_1 for base and blob_3 is a delta with blob2
 | 
						|
#    for base, such that blob_3 delta depth is 2;
 | 
						|
#
 | 
						|
# 2) the bulk of object data is uncompressible so the text part remains
 | 
						|
#    visible;
 | 
						|
#
 | 
						|
# 3) object header is always 2 bytes.
 | 
						|
 | 
						|
create_test_files() {
 | 
						|
    test-tool genrandom "foo" 2000 > file_1 &&
 | 
						|
    test-tool genrandom "foo" 1800 > file_2 &&
 | 
						|
    test-tool genrandom "foo" 1800 > file_3 &&
 | 
						|
    echo " base " >> file_1 &&
 | 
						|
    echo " delta1 " >> file_2 &&
 | 
						|
    echo " delta delta2 " >> file_3 &&
 | 
						|
    test-tool genrandom "bar" 150 >> file_2 &&
 | 
						|
    test-tool genrandom "baz" 100 >> file_3
 | 
						|
}
 | 
						|
 | 
						|
create_new_pack() {
 | 
						|
    rm -rf .git &&
 | 
						|
    git init &&
 | 
						|
    blob_1=$(git hash-object -t blob -w file_1) &&
 | 
						|
    blob_2=$(git hash-object -t blob -w file_2) &&
 | 
						|
    blob_3=$(git hash-object -t blob -w file_3) &&
 | 
						|
    pack=$(printf "$blob_1\n$blob_2\n$blob_3\n" |
 | 
						|
          git pack-objects $@ .git/objects/pack/pack) &&
 | 
						|
    pack=".git/objects/pack/pack-${pack}" &&
 | 
						|
    git verify-pack -v ${pack}.pack
 | 
						|
}
 | 
						|
 | 
						|
do_repack() {
 | 
						|
    pack=$(printf "$blob_1\n$blob_2\n$blob_3\n" |
 | 
						|
          git pack-objects $@ .git/objects/pack/pack) &&
 | 
						|
    pack=".git/objects/pack/pack-${pack}"
 | 
						|
}
 | 
						|
 | 
						|
do_corrupt_object() {
 | 
						|
    ofs=$(git show-index < ${pack}.idx | grep $1 | cut -f1 -d" ") &&
 | 
						|
    ofs=$(($ofs + $2)) &&
 | 
						|
    chmod +w ${pack}.pack &&
 | 
						|
    dd of=${pack}.pack bs=1 conv=notrunc seek=$ofs &&
 | 
						|
    test_must_fail git verify-pack ${pack}.pack
 | 
						|
}
 | 
						|
 | 
						|
printf '\0' > zero
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'initial setup validation' \
 | 
						|
    'create_test_files &&
 | 
						|
     create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'create corruption in header of first object' \
 | 
						|
    'do_corrupt_object $blob_1 0 < zero &&
 | 
						|
     test_must_fail git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_1 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and loose copy of first delta allows for partial recovery' \
 | 
						|
    'git prune-packed &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     test_must_fail git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'create corruption in data of first object' \
 | 
						|
    'create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     chmod +w ${pack}.pack &&
 | 
						|
     perl -i.bak -pe "s/ base /abcdef/" ${pack}.pack &&
 | 
						|
     test_must_fail git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_1 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and loose copy of second object allows for partial recovery' \
 | 
						|
    'git prune-packed &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     test_must_fail git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'create corruption in header of first delta' \
 | 
						|
    'create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     do_corrupt_object $blob_2 0 < zero &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'create corruption in data of first delta' \
 | 
						|
    'create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     chmod +w ${pack}.pack &&
 | 
						|
     perl -i.bak -pe "s/ delta1 /abcdefgh/" ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'corruption in delta base reference of first delta (OBJ_REF_DELTA)' \
 | 
						|
    'create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     do_corrupt_object $blob_2 2 < zero &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'corruption #0 in delta base reference of first delta (OBJ_OFS_DELTA)' \
 | 
						|
    'create_new_pack --delta-base-offset &&
 | 
						|
     git prune-packed &&
 | 
						|
     do_corrupt_object $blob_2 2 < zero &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack --delta-base-offset &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'corruption #1 in delta base reference of first delta (OBJ_OFS_DELTA)' \
 | 
						|
    'create_new_pack --delta-base-offset &&
 | 
						|
     git prune-packed &&
 | 
						|
     printf "\001" | do_corrupt_object $blob_2 2 &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack --delta-base-offset &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and a redundant pack allows for full recovery too' \
 | 
						|
    'do_corrupt_object $blob_2 2 < zero &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null &&
 | 
						|
     mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_1 &&
 | 
						|
     git hash-object -t blob -w file_2 &&
 | 
						|
     printf "$blob_1\n$blob_2\n" | git pack-objects .git/objects/pack/pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'corruption of delta base reference pointing to wrong object' \
 | 
						|
    'create_new_pack --delta-base-offset &&
 | 
						|
     git prune-packed &&
 | 
						|
     printf "\220\033" | do_corrupt_object $blob_3 2 &&
 | 
						|
     git cat-file blob $blob_1 >/dev/null &&
 | 
						|
     git cat-file blob $blob_2 >/dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 >/dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... but having a loose copy allows for full recovery' \
 | 
						|
    'mv ${pack}.idx tmp &&
 | 
						|
     git hash-object -t blob -w file_3 &&
 | 
						|
     mv tmp ${pack}.idx &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    '... and then a repack "clears" the corruption' \
 | 
						|
    'do_repack --delta-base-offset --no-reuse-delta &&
 | 
						|
     git prune-packed &&
 | 
						|
     git verify-pack ${pack}.pack &&
 | 
						|
     git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
test_expect_success \
 | 
						|
    'corrupting header to have too small output buffer fails unpack' \
 | 
						|
    'create_new_pack &&
 | 
						|
     git prune-packed &&
 | 
						|
     printf "\262\001" | do_corrupt_object $blob_1 0 &&
 | 
						|
     test_must_fail git cat-file blob $blob_1 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_2 > /dev/null &&
 | 
						|
     test_must_fail git cat-file blob $blob_3 > /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \1 - one byte in result
 | 
						|
# \1 - one literal byte (X)
 | 
						|
test_expect_success \
 | 
						|
    'apply good minimal delta' \
 | 
						|
    'printf "\0\1\1X" > minimal_delta &&
 | 
						|
     test-tool delta -p /dev/null minimal_delta /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \1 - 1 byte in result
 | 
						|
# \2 - two literal bytes (one too many)
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with too many literal bytes' \
 | 
						|
    'printf "\0\1\2XX" > too_big_literal &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null too_big_literal /dev/null'
 | 
						|
 | 
						|
# \4 - four bytes in base
 | 
						|
# \1 - one byte in result
 | 
						|
# \221 - copy, one byte offset, one byte size
 | 
						|
#   \0 - copy from offset 0
 | 
						|
#   \2 - copy two bytes (one too many)
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with too many copied bytes' \
 | 
						|
    'printf "\4\1\221\0\2" > too_big_copy &&
 | 
						|
     printf base >base &&
 | 
						|
     test_must_fail test-tool delta -p base too_big_copy /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \2 - two bytes in result
 | 
						|
# \2 - two literal bytes (we are short one)
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with too few literal bytes' \
 | 
						|
    'printf "\0\2\2X" > truncated_delta &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null truncated_delta /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \1 - one byte in result
 | 
						|
# \221 - copy, one byte offset, one byte size
 | 
						|
#   \0 - copy from offset 0
 | 
						|
#   \1 - copy one byte (we are short one)
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with too few bytes in base' \
 | 
						|
    'printf "\0\1\221\0\1" > truncated_base &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null truncated_base /dev/null'
 | 
						|
 | 
						|
# \4 - four bytes in base
 | 
						|
# \2 - two bytes in result
 | 
						|
# \1 - one literal byte (X)
 | 
						|
# \221 - copy, one byte offset, one byte size
 | 
						|
#        (offset/size missing)
 | 
						|
#
 | 
						|
# Note that the literal byte is necessary to get past the uninteresting minimum
 | 
						|
# delta size check.
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with truncated copy parameters' \
 | 
						|
    'printf "\4\2\1X\221" > truncated_copy_delta &&
 | 
						|
     printf base >base &&
 | 
						|
     test_must_fail test-tool delta -p base truncated_copy_delta /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \1 - one byte in result
 | 
						|
# \1 - one literal byte (X)
 | 
						|
# \1 - trailing garbage command
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with trailing garbage literal' \
 | 
						|
    'printf "\0\1\1X\1" > tail_garbage_literal &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null tail_garbage_literal /dev/null'
 | 
						|
 | 
						|
# \4 - four bytes in base
 | 
						|
# \1 - one byte in result
 | 
						|
# \1 - one literal byte (X)
 | 
						|
# \221 - copy, one byte offset, one byte size
 | 
						|
#   \0 - copy from offset 0
 | 
						|
#   \1 - copy 1 byte
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with trailing garbage copy' \
 | 
						|
    'printf "\4\1\1X\221\0\1" > tail_garbage_copy &&
 | 
						|
     printf base >base &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null tail_garbage_copy /dev/null'
 | 
						|
 | 
						|
# \0 - empty base
 | 
						|
# \1 - one byte in result
 | 
						|
# \1 - one literal byte (X)
 | 
						|
# \0 - bogus opcode
 | 
						|
test_expect_success \
 | 
						|
    'apply delta with trailing garbage opcode' \
 | 
						|
    'printf "\0\1\1X\0" > tail_garbage_opcode &&
 | 
						|
     test_must_fail test-tool delta -p /dev/null tail_garbage_opcode /dev/null'
 | 
						|
 | 
						|
test_done
 |