blob: e7983453196e285891c4bce087f93dd770a00e56 [file] [log] [blame] [edit]
#!/bin/sh
#
# Copyright 2008 The open-vcdiff Authors. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http:#www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# This script tests the correctness of the vcdiff command-line executable.
# If you add a new test here, please add the same test to the Windows script
# src/vcdiff_test.bat.
#
# The caller should pass path to PROJECT_SOURCE_DIR to this script. CTest
# automatically pass it from ```make test```
srcdir=$1
# Find input files
VCDIFF=./vcdiff
# These options are only needed for the encoder;
# the decoder will recognize the interleaved and checksum formats
# without needing to specify any options.
VCD_OPTIONS="-interleaved -checksum"
DICTIONARY_FILE=$srcdir/testdata/configure.ac.v0.1
TARGET_FILE=$srcdir/testdata/configure.ac.v0.2
TEST_TMPDIR=${TMPDIR-/tmp}
DELTA_FILE=$TEST_TMPDIR/configure.ac.vcdiff
OUTPUT_TARGET_FILE=$TEST_TMPDIR/configure.ac.output
MALICIOUS_ENCODING=$srcdir/testdata/allocates_4gb.vcdiff
OVERFLOW_DELTA_FILE=$srcdir/testdata/size-overflow-delta
OVERFLOW_DICTIONARY_FILE=$srcdir/testdata/size-overflow-dictionary
OVERFLOW_ERROR_32=$srcdir/testdata/size-overflow-error-32
OVERFLOW_ERROR_64=$srcdir/testdata/size-overflow-error-64
# vcdiff with no arguments shows usage information & error result
$VCDIFF \
&& { echo "vcdiff with no arguments should fail, but succeeded"; \
exit 1; }
echo "Test 1 ok";
# vcdiff with three arguments but without "encode" or "decode"
# shows usage information & error result
$VCDIFF $VCD_OPTIONS \
-dictionary $DICTIONARY_FILE -target $TARGET_FILE -delta $DELTA_FILE \
&& { echo "vcdiff without operation argument should fail, but succeeded"; \
exit 1; }
echo "Test 2 ok";
# vcdiff with all three arguments. Verify that output file matches target file
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
|| { echo "Encode with three arguments failed"; \
exit 1; }
$VCDIFF decode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
-target $OUTPUT_TARGET_FILE \
|| { echo "Decode with three arguments failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 3 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# vcdiff using stdin/stdout. Verify that output file matches target file
{ $VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
< $TARGET_FILE \
> $DELTA_FILE; } \
|| { echo "Encode using stdin/stdout failed"; \
exit 1; }
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
< $DELTA_FILE \
> $OUTPUT_TARGET_FILE; } \
|| { echo "Decode using stdin/stdout failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 4 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# vcdiff with mixed stdin/stdout.
{ $VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
> $DELTA_FILE; } \
|| { echo "Encode with mixed arguments failed"; \
exit 1; }
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
> $OUTPUT_TARGET_FILE; } \
|| { echo "Decode with mixed arguments failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 5 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
{ $VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
< $TARGET_FILE \
-delta $DELTA_FILE; } \
|| { echo "Encode with mixed arguments failed"; \
exit 1; }
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
< $DELTA_FILE \
-target $OUTPUT_TARGET_FILE; } \
|| { echo "Decode with mixed arguments failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 6 ok";
rm $OUTPUT_TARGET_FILE
# Don't remove $DELTA_FILE; use it for the next test
# If using the wrong dictionary, and dictionary is smaller than the original
# dictionary, vcdiff will spot the mistake and return an error. (It can't
# detect the case where the wrong dictionary is larger than the right one.)
$VCDIFF decode -dictionary $TARGET_FILE \
-delta $DELTA_FILE \
-target $OUTPUT_TARGET_FILE \
&& { echo "Decode using larger dictionary should fail, but succeeded"; \
exit 1; }
echo "Test 7 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# "vcdiff test" with all three arguments.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
|| { echo "vcdiff test with three arguments failed"; \
exit 1; }
echo "Test 8 ok";
rm $DELTA_FILE
# Dictionary file not found.
$VCDIFF $VCD_OPTIONS \
encode -dictionary $TEST_TMPDIR/nonexistent_file \
-target $TARGET_FILE \
-delta $DELTA_FILE \
&& { echo "vcdiff with missing dictionary file should fail, but succeeded"; \
exit 1; }
echo "Test 9 ok";
# Target file not found.
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TEST_TMPDIR/nonexistent_file \
-delta $DELTA_FILE \
&& { echo "vcdiff with missing target file should fail, but succeeded"; \
exit 1; }
echo "Test 10 ok";
# Delta file not found.
$VCDIFF decode -dictionary $DICTIONARY_FILE \
-delta $TEST_TMPDIR/nonexistent_file \
-target $OUTPUT_TARGET_FILE \
&& { echo "vcdiff with missing delta file should fail, but succeeded"; \
exit 1; }
echo "Test 11 ok";
# Try traversing an infinite loop of symbolic links.
ln -s $TEST_TMPDIR/infinite_loop1 $TEST_TMPDIR/infinite_loop2
ln -s $TEST_TMPDIR/infinite_loop2 $TEST_TMPDIR/infinite_loop1
$VCDIFF $VCD_OPTIONS \
encode -dictionary $TEST_TMPDIR/infinite_loop1 \
-target $TEST_TMPDIR/infinite_loop2 \
-delta $DELTA_FILE \
&& { echo "vcdiff with symbolic link loop should fail, but succeeded"; \
exit 1; }
echo "Test 12 ok";
rm $TEST_TMPDIR/infinite_loop1 $TEST_TMPDIR/infinite_loop2
# Test using -stats flag
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-stats \
|| { echo "Encode with -stats failed"; \
exit 1; }
$VCDIFF -stats \
decode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
-target $OUTPUT_TARGET_FILE \
|| { echo "Decode with -stats failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 13 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# Using /dev/null as dictionary should work, but (because dictionary is empty)
# it will not produce a small delta file.
$VCDIFF $VCD_OPTIONS \
test -dictionary /dev/null \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-stats \
|| { echo "vcdiff test with /dev/null as dictionary failed"; \
exit 1; }
echo "Test 14 ok";
rm $DELTA_FILE
# Using /dev/kmem as dictionary or target should produce an error
# (permission denied, or too large, or special file type)
$VCDIFF $VCD_OPTIONS \
encode -dictionary /dev/kmem \
-target $TARGET_FILE \
-delta $DELTA_FILE \
&& { echo "vcdiff with /dev/kmem as dictionary should fail, but succeeded"; \
exit 1; }
echo "Test 15 ok";
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target /dev/kmem \
-delta $DELTA_FILE \
&& { echo "vcdiff with /dev/kmem as target should fail, but succeeded"; \
exit 1; }
echo "Test 16 ok";
# Decode using something that isn't a delta file
$VCDIFF decode -dictionary $DICTIONARY_FILE \
-delta /etc/fstab \
-target $OUTPUT_TARGET_FILE \
&& { echo "vcdiff with invalid delta file should fail, but succeeded"; \
exit 1; }
echo "Test 17 ok";
$VCDIFF $VCD_OPTIONS \
encode -target $TARGET_FILE \
-delta $DELTA_FILE \
-dictionary \
&& { echo "-dictionary option with no file name should fail, but succeeded"; \
exit 1; }
echo "Test 18 ok";
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
-target \
&& { echo "-target option with no file name should fail, but succeeded"; \
exit 1; }
echo "Test 19 ok";
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta \
&& { echo "-delta option with no file name should fail, but succeeded"; \
exit 1; }
echo "Test 20 ok";
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-buffersize \
&& { echo "-buffersize option with no argument should fail, but succeeded"; \
exit 1; }
echo "Test 21 ok";
# Using -buffersize=1 should still work.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-buffersize 1 \
-stats \
|| { echo "vcdiff test with -buffersize=1 failed"; \
exit 1; }
echo "Test 22 ok";
rm $DELTA_FILE
# Using -buffersize=1 with stdin/stdout means that vcdiff
# will create a separate target window for each byte read.
{ $VCDIFF encode -dictionary $DICTIONARY_FILE \
-buffersize 1 \
-stats \
< $TARGET_FILE \
> $DELTA_FILE; } \
|| { echo "Encode using stdin/stdout with -buffersize=1 failed"; \
exit 1; }
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
-buffersize 1 \
-stats \
< $DELTA_FILE \
> $OUTPUT_TARGET_FILE; } \
|| { echo "Decode using stdin/stdout with -buffersize=1 failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original with -buffersize=1"; \
exit 1; }
echo "Test 23 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# Using -buffersize=0 should fail.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-buffersize 0 \
&& { echo "vcdiff test with -buffersize=0 should fail, but succeeded"; \
exit 1; }
echo "Test 24 ok";
rm $DELTA_FILE
# Using -buffersize=128M (larger than default maximum) should still work.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-buffersize 134217728 \
-stats \
|| { echo "vcdiff test with -buffersize=128M failed"; \
exit 1; }
echo "Test 25 ok";
rm $DELTA_FILE
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-froobish \
&& { echo "vdiff test with unrecognized option should fail, but succeeded"; \
exit 1; }
echo "Test 26 ok";
$VCDIFF $VCD_OPTIONS \
encode -target $TARGET_FILE \
-delta $DELTA_FILE \
&& { echo "encode with no dictionary option should fail, but succeeded"; \
exit 1; }
echo "Test 27 ok";
$VCDIFF decode -target $TARGET_FILE \
-delta $DELTA_FILE \
&& { echo "decode with no dictionary option should fail, but succeeded"; \
exit 1; }
echo "Test 28 ok";
# Remove -interleaved and -checksum options
{ $VCDIFF encode -dictionary $DICTIONARY_FILE \
< $TARGET_FILE \
> $DELTA_FILE; } \
|| { echo "Encode without -interleaved and -checksum options failed"; \
exit 1; }
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
< $DELTA_FILE \
> $OUTPUT_TARGET_FILE; } \
|| { echo "Decode non-interleaved output failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original with -interleaved"; \
exit 1; }
echo "Test 29 ok";
# -target_matches option
{ $VCDIFF encode -dictionary $DICTIONARY_FILE \
-target_matches \
-stats \
< $TARGET_FILE \
> $DELTA_FILE; } \
|| { echo "Encode with -target_matches option failed"; \
exit 1; }
# The decode operation ignores the -target_matches option.
{ $VCDIFF decode -dictionary $DICTIONARY_FILE \
< $DELTA_FILE \
> $OUTPUT_TARGET_FILE; } \
|| { echo "Decode output failed with -target_matches"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original with -target_matches"; \
exit 1; }
echo "Test 30 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
$VCDIFF $VCD_OPTIONS \
dencode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
&& { echo "vdiff with unrecognized action should fail, but succeeded"; \
exit 1; }
echo "Test 31 ok";
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
&& { echo "vdiff test without delta option should fail, but succeeded"; \
exit 1; }
echo "Test 32 ok";
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
&& { echo "vdiff test without target option should fail, but succeeded"; \
exit 1; }
echo "Test 33 ok";
# open-vcdiff bug 8 (https://github.com/google/open-vcdiff/issues/8)
# A malicious encoding that tries to produce a 4GB target file made up of 64
# windows, each window having a size of 64MB.
# Limit memory usage to 256MB per process, so the test doesn't take forever
# to run out of memory.
OLD_ULIMIT=$(ulimit -v)
echo "Old ulimit: $OLD_ULIMIT"
ulimit -S -v 262144
echo "New ulimit: $(ulimit -v)"
$VCDIFF $VCD_OPTIONS \
decode -dictionary $DICTIONARY_FILE \
-delta $MALICIOUS_ENCODING \
-target /dev/null \
-max_target_file_size=65536 \
&& { echo "Decoding malicious file should fail, but succeeded"; \
exit 1; }
echo "Test 34 ok";
$VCDIFF $VCD_OPTIONS \
decode -dictionary $DICTIONARY_FILE \
-delta $MALICIOUS_ENCODING \
-target /dev/null \
-max_target_window_size=65536 \
&& { echo "Decoding malicious file should fail, but succeeded"; \
exit 1; }
echo "Test 35 ok";
ulimit -S -v $OLD_ULIMIT
# Decoding a small target with the -max_target_file_size option should succeed.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-max_target_file_size=65536 \
|| { echo "vcdiff test with -max_target_file_size failed"; \
exit 1; }
echo "Test 36 ok";
# Decoding a small target with -max_target_window_size option should succeed.
$VCDIFF $VCD_OPTIONS \
test -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-max_target_window_size=65536 \
|| { echo "vcdiff test with -max_target_window_size failed"; \
exit 1; }
echo "Test 37 ok";
rm $DELTA_FILE
# Test using -allow_vcd_target=false
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-allow_vcd_target=false \
|| { echo "Encode with -allow_vcd_target=false failed"; \
exit 1; }
$VCDIFF $VCD_OPTIONS \
decode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
-target $OUTPUT_TARGET_FILE \
-allow_vcd_target=false \
|| { echo "Decode with -allow_vcd_target=false failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 38 ok";
rm $DELTA_FILE
rm $OUTPUT_TARGET_FILE
# Test using -allow_vcd_target=true
$VCDIFF $VCD_OPTIONS \
encode -dictionary $DICTIONARY_FILE \
-target $TARGET_FILE \
-delta $DELTA_FILE \
-allow_vcd_target=true \
|| { echo "Encode with -allow_vcd_target=true failed"; \
exit 1; }
$VCDIFF $VCD_OPTIONS \
decode -dictionary $DICTIONARY_FILE \
-delta $DELTA_FILE \
-target $OUTPUT_TARGET_FILE \
-allow_vcd_target=true \
|| { echo "Decode with -allow_vcd_target=true failed"; \
exit 1; }
cmp $TARGET_FILE $OUTPUT_TARGET_FILE \
|| { echo "Decoded target does not match original"; \
exit 1; }
echo "Test 39 ok";
# Test for overflow in size parsing. Check for the specific overflow error
# message and make sure that it's emitted.
$VCDIFF $VCD_OPTIONS \
decode -dictionary $OVERFLOW_DICTIONARY_FILE \
-delta $OVERFLOW_DELTA_FILE \
-target $OUTPUT_TARGET_FILE 2>$TEST_TMPDIR/overflow-err\
&& { echo "Size overflow didn't crash or error vcdiff"; \
exit 1; }
cmp $TEST_TMPDIR/overflow-err $OVERFLOW_ERROR_32 \
|| cmp $TEST_TMPDIR/overflow-err $OVERFLOW_ERROR_64 \
|| { echo "Overflow error message does not match"; \
exit 1; }
echo "Test 40 ok"
echo "PASS"