// Copyright 2011 The LevelDB-Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Taken from: https://code.google.com/p/leveldb-go/source/browse/leveldb/record/record_test.go?r=df1fa28f7f3be6c3935548169002309c12967135 // License, authors and contributors informations can be found at bellow URLs respectively: // https://code.google.com/p/leveldb-go/source/browse/LICENSE // https://code.google.com/p/leveldb-go/source/browse/AUTHORS // https://code.google.com/p/leveldb-go/source/browse/CONTRIBUTORS package journal import ( "bytes" "encoding/binary" "fmt" "io" "io/ioutil" "math/rand" "strings" "testing" ) type dropper struct { t *testing.T } func (d dropper) Drop(err error) { d.t.Log(err) } func short(s string) string { if len(s) < 64 { return s } return fmt.Sprintf("%s...(skipping %d bytes)...%s", s[:20], len(s)-40, s[len(s)-20:]) } // big returns a string of length n, composed of repetitions of partial. func big(partial string, n int) string { return strings.Repeat(partial, n/len(partial)+1)[:n] } func TestEmpty(t *testing.T) { buf := new(bytes.Buffer) r := NewReader(buf, dropper{t}, true, true) if _, err := r.Next(); err != io.EOF { t.Fatalf("got %v, want %v", err, io.EOF) } } func testGenerator(t *testing.T, reset func(), gen func() (string, bool)) { buf := new(bytes.Buffer) reset() w := NewWriter(buf) for { s, ok := gen() if !ok { break } ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write([]byte(s)); err != nil { t.Fatal(err) } } if err := w.Close(); err != nil { t.Fatal(err) } reset() r := NewReader(buf, dropper{t}, true, true) for { s, ok := gen() if !ok { break } rr, err := r.Next() if err != nil { t.Fatal(err) } x, err := ioutil.ReadAll(rr) if err != nil { t.Fatal(err) } if string(x) != s { t.Fatalf("got %q, want %q", short(string(x)), short(s)) } } if _, err := r.Next(); err != io.EOF { t.Fatalf("got %v, want %v", err, io.EOF) } } func testLiterals(t *testing.T, s []string) { var i int reset := func() { i = 0 } gen := func() (string, bool) { if i == len(s) { return "", false } i++ return s[i-1], true } testGenerator(t, reset, gen) } func TestMany(t *testing.T) { const n = 1e5 var i int reset := func() { i = 0 } gen := func() (string, bool) { if i == n { return "", false } i++ return fmt.Sprintf("%d.", i-1), true } testGenerator(t, reset, gen) } func TestRandom(t *testing.T) { const n = 1e2 var ( i int r *rand.Rand ) reset := func() { i, r = 0, rand.New(rand.NewSource(0)) } gen := func() (string, bool) { if i == n { return "", false } i++ return strings.Repeat(string(uint8(i)), r.Intn(2*blockSize+16)), true } testGenerator(t, reset, gen) } func TestBasic(t *testing.T) { testLiterals(t, []string{ strings.Repeat("a", 1000), strings.Repeat("b", 97270), strings.Repeat("c", 8000), }) } func TestBoundary(t *testing.T) { for i := blockSize - 16; i < blockSize+16; i++ { s0 := big("abcd", i) for j := blockSize - 16; j < blockSize+16; j++ { s1 := big("ABCDE", j) testLiterals(t, []string{s0, s1}) testLiterals(t, []string{s0, "", s1}) testLiterals(t, []string{s0, "x", s1}) } } } func TestFlush(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // Write a couple of records. Everything should still be held // in the record.Writer buffer, so that buf.Len should be 0. w0, _ := w.Next() w0.Write([]byte("0")) w1, _ := w.Next() w1.Write([]byte("11")) if got, want := buf.Len(), 0; got != want { t.Fatalf("buffer length #0: got %d want %d", got, want) } // Flush the record.Writer buffer, which should yield 17 bytes. // 17 = 2*7 + 1 + 2, which is two headers and 1 + 2 payload bytes. if err := w.Flush(); err != nil { t.Fatal(err) } if got, want := buf.Len(), 17; got != want { t.Fatalf("buffer length #1: got %d want %d", got, want) } // Do another write, one that isn't large enough to complete the block. // The write should not have flowed through to buf. w2, _ := w.Next() w2.Write(bytes.Repeat([]byte("2"), 10000)) if got, want := buf.Len(), 17; got != want { t.Fatalf("buffer length #2: got %d want %d", got, want) } // Flushing should get us up to 10024 bytes written. // 10024 = 17 + 7 + 10000. if err := w.Flush(); err != nil { t.Fatal(err) } if got, want := buf.Len(), 10024; got != want { t.Fatalf("buffer length #3: got %d want %d", got, want) } // Do a bigger write, one that completes the current block. // We should now have 32768 bytes (a complete block), without // an explicit flush. w3, _ := w.Next() w3.Write(bytes.Repeat([]byte("3"), 40000)) if got, want := buf.Len(), 32768; got != want { t.Fatalf("buffer length #4: got %d want %d", got, want) } // Flushing should get us up to 50038 bytes written. // 50038 = 10024 + 2*7 + 40000. There are two headers because // the one record was split into two chunks. if err := w.Flush(); err != nil { t.Fatal(err) } if got, want := buf.Len(), 50038; got != want { t.Fatalf("buffer length #5: got %d want %d", got, want) } // Check that reading those records give the right lengths. r := NewReader(buf, dropper{t}, true, true) wants := []int64{1, 2, 10000, 40000} for i, want := range wants { rr, _ := r.Next() n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #%d: %v", i, err) } if n != want { t.Fatalf("read #%d: got %d bytes want %d", i, n, want) } } } func TestNonExhaustiveRead(t *testing.T) { const n = 100 buf := new(bytes.Buffer) p := make([]byte, 10) rnd := rand.New(rand.NewSource(1)) w := NewWriter(buf) for i := 0; i < n; i++ { length := len(p) + rnd.Intn(3*blockSize) s := string(uint8(i)) + "123456789abcdefgh" ww, _ := w.Next() ww.Write([]byte(big(s, length))) } if err := w.Close(); err != nil { t.Fatal(err) } r := NewReader(buf, dropper{t}, true, true) for i := 0; i < n; i++ { rr, _ := r.Next() _, err := io.ReadFull(rr, p) if err != nil { t.Fatal(err) } want := string(uint8(i)) + "123456789" if got := string(p); got != want { t.Fatalf("read #%d: got %q want %q", i, got, want) } } } func TestStaleReader(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) w0, err := w.Next() if err != nil { t.Fatal(err) } w0.Write([]byte("0")) w1, err := w.Next() if err != nil { t.Fatal(err) } w1.Write([]byte("11")) if err := w.Close(); err != nil { t.Fatal(err) } r := NewReader(buf, dropper{t}, true, true) r0, err := r.Next() if err != nil { t.Fatal(err) } r1, err := r.Next() if err != nil { t.Fatal(err) } p := make([]byte, 1) if _, err := r0.Read(p); err == nil || !strings.Contains(err.Error(), "stale") { t.Fatalf("stale read #0: unexpected error: %v", err) } if _, err := r1.Read(p); err != nil { t.Fatalf("fresh read #1: got %v want nil error", err) } if p[0] != '1' { t.Fatalf("fresh read #1: byte contents: got '%c' want '1'", p[0]) } } func TestStaleWriter(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) w0, err := w.Next() if err != nil { t.Fatal(err) } w1, err := w.Next() if err != nil { t.Fatal(err) } if _, err := w0.Write([]byte("0")); err == nil || !strings.Contains(err.Error(), "stale") { t.Fatalf("stale write #0: unexpected error: %v", err) } if _, err := w1.Write([]byte("11")); err != nil { t.Fatalf("fresh write #1: got %v want nil error", err) } if err := w.Flush(); err != nil { t.Fatalf("flush: %v", err) } if _, err := w1.Write([]byte("0")); err == nil || !strings.Contains(err.Error(), "stale") { t.Fatalf("stale write #1: unexpected error: %v", err) } } func TestCorrupt_MissingLastBlock(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-1024)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } // Cut the last block. b := buf.Bytes()[:blockSize] r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read. rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if n != blockSize-1024 { t.Fatalf("read #0: got %d bytes want %d", n, blockSize-1024) } // Second read. rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != io.ErrUnexpectedEOF { t.Fatalf("read #1: unexpected error: %v", err) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } } func TestCorrupt_CorruptedFirstBlock(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize/2)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } // Third record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+1)); err != nil { t.Fatalf("write #2: unexpected error: %v", err) } // Fourth record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+2)); err != nil { t.Fatalf("write #3: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } b := buf.Bytes() // Corrupting block #0. for i := 0; i < 1024; i++ { b[i] = '1' } r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read (third record). rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if want := int64(blockSize-headerSize) + 1; n != want { t.Fatalf("read #0: got %d bytes want %d", n, want) } // Second read (fourth record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #1: %v", err) } if want := int64(blockSize-headerSize) + 2; n != want { t.Fatalf("read #1: got %d bytes want %d", n, want) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } } func TestCorrupt_CorruptedMiddleBlock(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize/2)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } // Third record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+1)); err != nil { t.Fatalf("write #2: unexpected error: %v", err) } // Fourth record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+2)); err != nil { t.Fatalf("write #3: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } b := buf.Bytes() // Corrupting block #1. for i := 0; i < 1024; i++ { b[blockSize+i] = '1' } r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read (first record). rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if want := int64(blockSize / 2); n != want { t.Fatalf("read #0: got %d bytes want %d", n, want) } // Second read (second record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != io.ErrUnexpectedEOF { t.Fatalf("read #1: unexpected error: %v", err) } // Third read (fourth record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #2: %v", err) } if want := int64(blockSize-headerSize) + 2; n != want { t.Fatalf("read #2: got %d bytes want %d", n, want) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } } func TestCorrupt_CorruptedLastBlock(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize/2)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } // Third record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+1)); err != nil { t.Fatalf("write #2: unexpected error: %v", err) } // Fourth record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+2)); err != nil { t.Fatalf("write #3: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } b := buf.Bytes() // Corrupting block #3. for i := len(b) - 1; i > len(b)-1024; i-- { b[i] = '1' } r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read (first record). rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if want := int64(blockSize / 2); n != want { t.Fatalf("read #0: got %d bytes want %d", n, want) } // Second read (second record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #1: %v", err) } if want := int64(blockSize - headerSize); n != want { t.Fatalf("read #1: got %d bytes want %d", n, want) } // Third read (third record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #2: %v", err) } if want := int64(blockSize-headerSize) + 1; n != want { t.Fatalf("read #2: got %d bytes want %d", n, want) } // Fourth read (fourth record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != io.ErrUnexpectedEOF { t.Fatalf("read #3: unexpected error: %v", err) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } } func TestCorrupt_FirstChuckLengthOverflow(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize/2)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } // Third record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+1)); err != nil { t.Fatalf("write #2: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } b := buf.Bytes() // Corrupting record #1. x := blockSize binary.LittleEndian.PutUint16(b[x+4:], 0xffff) r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read (first record). rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if want := int64(blockSize / 2); n != want { t.Fatalf("read #0: got %d bytes want %d", n, want) } // Second read (second record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != io.ErrUnexpectedEOF { t.Fatalf("read #1: unexpected error: %v", err) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } } func TestCorrupt_MiddleChuckLengthOverflow(t *testing.T) { buf := new(bytes.Buffer) w := NewWriter(buf) // First record. ww, err := w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize/2)); err != nil { t.Fatalf("write #0: unexpected error: %v", err) } // Second record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), blockSize-headerSize)); err != nil { t.Fatalf("write #1: unexpected error: %v", err) } // Third record. ww, err = w.Next() if err != nil { t.Fatal(err) } if _, err := ww.Write(bytes.Repeat([]byte("0"), (blockSize-headerSize)+1)); err != nil { t.Fatalf("write #2: unexpected error: %v", err) } if err := w.Close(); err != nil { t.Fatal(err) } b := buf.Bytes() // Corrupting record #1. x := blockSize/2 + headerSize binary.LittleEndian.PutUint16(b[x+4:], 0xffff) r := NewReader(bytes.NewReader(b), dropper{t}, false, true) // First read (first record). rr, err := r.Next() if err != nil { t.Fatal(err) } n, err := io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #0: %v", err) } if want := int64(blockSize / 2); n != want { t.Fatalf("read #0: got %d bytes want %d", n, want) } // Second read (third record). rr, err = r.Next() if err != nil { t.Fatal(err) } n, err = io.Copy(ioutil.Discard, rr) if err != nil { t.Fatalf("read #1: %v", err) } if want := int64(blockSize-headerSize) + 1; n != want { t.Fatalf("read #1: got %d bytes want %d", n, want) } if _, err := r.Next(); err != io.EOF { t.Fatalf("last next: unexpected error: %v", err) } }