// Copyright 2017 syzkaller project authors. All rights reserved. // Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file. package db import ( "fmt" "io/ioutil" "math/rand" "os" "reflect" "testing" "github.com/google/syzkaller/pkg/osutil" ) func TestBasic(t *testing.T) { fn := tempFile(t) defer os.Remove(fn) db, err := Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } if len(db.Records) != 0 { t.Fatalf("empty db contains records") } db.Save("", nil, 0) db.Save("1", []byte("ab"), 1) db.Save("23", []byte("abcd"), 2) want := map[string]Record{ "": {Val: nil, Seq: 0}, "1": {Val: []byte("ab"), Seq: 1}, "23": {Val: []byte("abcd"), Seq: 2}, } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after save: %v, want: %v", db.Records, want) } if err := db.Flush(); err != nil { t.Fatalf("failed to flush db: %v", err) } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after flush: %v, want: %v", db.Records, want) } db, err = Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after reopen: %v, want: %v", db.Records, want) } } func TestModify(t *testing.T) { fn := tempFile(t) defer os.Remove(fn) db, err := Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } db.Save("1", []byte("ab"), 0) db.Save("23", nil, 1) db.Save("456", []byte("abcd"), 1) db.Save("7890", []byte("a"), 0) db.Delete("23") db.Save("1", nil, 5) db.Save("456", []byte("ef"), 6) db.Delete("7890") db.Save("456", []byte("efg"), 0) db.Save("7890", []byte("bc"), 0) want := map[string]Record{ "1": {Val: nil, Seq: 5}, "456": {Val: []byte("efg"), Seq: 0}, "7890": {Val: []byte("bc"), Seq: 0}, } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after modification: %v, want: %v", db.Records, want) } if err := db.Flush(); err != nil { t.Fatalf("failed to flush db: %v", err) } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after flush: %v, want: %v", db.Records, want) } db, err = Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } if !reflect.DeepEqual(db.Records, want) { t.Fatalf("bad db after reopen: %v, want: %v", db.Records, want) } } func TestLarge(t *testing.T) { fn := tempFile(t) defer os.Remove(fn) db, err := Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } const nrec = 1000 val := make([]byte, 1000) for i := range val { val[i] = byte(rand.Intn(256)) } for i := 0; i < nrec; i++ { db.Save(fmt.Sprintf("%v", i), val, 0) } if err := db.Flush(); err != nil { t.Fatalf("failed to flush db: %v", err) } db, err = Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } if len(db.Records) != nrec { t.Fatalf("wrong record count: %v, want %v", len(db.Records), nrec) } } func TestOpenInvalid(t *testing.T) { f, err := ioutil.TempFile("", "syz-db-test") if err != nil { t.Error(err) } defer f.Close() defer os.Remove(f.Name()) if _, err := f.Write([]byte(`some invalid data`)); err != nil { t.Error(err) } if db, err := Open(f.Name(), true); err == nil { t.Fatal("opened invalid db") } else if db == nil { t.Fatal("db is nil") } } func TestOpenInaccessible(t *testing.T) { if os.Getuid() == 0 { t.Skip("opening inaccessible file won't fail under root") } f, err := ioutil.TempFile("", "syz-db-test") if err != nil { t.Error(err) } f.Close() os.Chmod(f.Name(), 0) defer os.Chmod(f.Name(), 0777) defer os.Remove(f.Name()) if db, err := Open(f.Name(), false); err == nil { t.Fatal("opened inaccessible db") } else if db != nil { t.Fatal("db is not nil") } } func TestOpenCorrupted(t *testing.T) { fn := tempFile(t) defer os.Remove(fn) db, err := Open(fn, false) if err != nil { t.Fatalf("failed to open db: %v", err) } // Write 1000 records, then wipe half of the file and test that we // (1) get an error, (2) still get 450-550 records. for i := 0; i < 1000; i++ { db.Save(fmt.Sprintf("%v", i), []byte{byte(i)}, 0) } if err := db.Flush(); err != nil { t.Fatalf("failed to flush db: %v", err) } data, err := ioutil.ReadFile(fn) if err != nil { t.Fatalf("failed to read db: %v", err) } for i := len(data) / 2; i < len(data); i++ { data[i] = 0 } if err := osutil.WriteFile(fn, data); err != nil { t.Fatalf("failed to write db: %v", err) } db, err = Open(fn, true) if err == nil { t.Fatalf("no error for corrutped db") } t.Logf("records %v, error: %v", len(db.Records), err) if len(db.Records) < 450 || len(db.Records) > 550 { t.Fatalf("wrong record count: %v", len(db.Records)) } } func tempFile(t *testing.T) string { fn, err := osutil.TempFile("syzkaller.test.db") if err != nil { t.Fatal(err) } return fn }