package main import ( "fmt" "io" "path" "os" "sort" "time" "xdelta" ) const ( xdataset = "/volume/home/jmacd/src/testdata" xcompare = "/volume/home/jmacd/src/xdelta-devel/xdelta3/build/x86_64-pc-linux-gnu-m64/xoff64/xdelta3" xdelta3 = "/volume/home/jmacd/src/xdelta-64bithash/xdelta3/build/x86_64-pc-linux-gnu-m64/usize64/xoff64/xdelta3" seed = 1422253499919909358 ) type Config struct { srcbuf_size int64 window_size int64 blocksize int } func NewC() Config { // TODO make these (and above) flags return Config{1<<26, 1<<22, 1<<16} } func (c Config) smokeTest(t *xdelta.TestGroup, p xdelta.Program) { target := "Hello world!" source := "Hello world, nice to meet you!" enc, err := t.Exec("encode", p, true, []string{"-e"}) if err != nil { t.Panic(err) } dec, err := t.Exec("decode", p, true, []string{"-d"}) if err != nil { t.Panic(err) } encodeout := t.Drain(enc.Stdout, "encode.stdout") decodeout := t.Drain(dec.Stdout, "decode.stdout") t.Empty(enc.Stderr, "encode") t.Empty(dec.Stderr, "decode") t.TestWrite("encode.stdin", enc.Stdin, []byte(target)) t.TestWrite("encode.srcin", enc.Srcin, []byte(source)) t.TestWrite("decode.stdin", dec.Stdin, <-encodeout) t.TestWrite("decode.srcin", dec.Srcin, []byte(source)) if do := string(<-decodeout); do != target { t.Panic(fmt.Errorf("It's not working! %s\n!=\n%s\n", do, target)) } t.Wait(enc, dec) } type PairTest struct { // Input Config program xdelta.Program source, target string // Output TestOutput } type TestOutput struct { encoded int64 encDuration time.Duration decDuration time.Duration encSysDuration time.Duration decSysDuration time.Duration } func (to *TestOutput) Add(a TestOutput) { to.encoded += a.encoded to.encDuration += a.encDuration to.decDuration += a.decDuration to.encSysDuration += a.encSysDuration to.decSysDuration += a.decSysDuration } func (to *TestOutput) String() string { return fmt.Sprintf("SIZE: %v\tT: %v\tTSYS: %v\tDT: %v\tDTSYS: %v", to.encoded, to.encDuration, to.encSysDuration, to.decDuration, to.encSysDuration) } // P is the test program, Q is the reference version. func (cfg Config) datasetTest(t *xdelta.TestGroup, p, q xdelta.Program) { dir, err := os.Open(xdataset) if err != nil { t.Panic(err) } dents, err := dir.Readdir(-1) if err != nil { t.Panic(err) } paths := make([]string, len(dents)) var total int64 for i, d := range dents { if !d.Mode().IsRegular() { continue } paths[i] = fmt.Sprint(xdataset, "/", d.Name()) total += d.Size() } meansize := total / int64(len(dents)) largest := uint(20) for ; largest <= 31 && 1< (1.05 * float64(expect)) { t.Fail("encoded size should be ~=", expect, ", actual ", encoded_size) } } func main() { r, err := xdelta.NewRunner() if err != nil { panic(err) } defer r.Cleanup() cfg := NewC() prog := xdelta.Program{xdelta3} r.RunTest("smoketest", func(t *xdelta.TestGroup) { cfg.smokeTest(t, prog) }) for i := uint(29); i <= 33; i += 1 { // The arguments to offsetTest are offset, source // window size, and file size. The source window size // is (2 << i) and (in the 3.0x release branch) is // limited to 2^31, so the the greatest value of i is // 30. cfg.srcbuf_size = 2 << i r.RunTest(fmt.Sprint("offset", i), func(t *xdelta.TestGroup) { cfg.offsetTest(t, prog, 1 << i, 3 << i) }) } comp := xdelta.Program{xcompare} r.RunTest("dataset", func(t *xdelta.TestGroup) { cfg.datasetTest(t, prog, comp) }) }