blob: 8319e73d1acd82da5796353838799f44e9ad1d1a [file] [log] [blame] [edit]
// 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"
"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 := os.CreateTemp("", "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 := os.CreateTemp("", "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 := os.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
}