//nolint:unparam,maligned package bbolt import ( "context" "io/ioutil" "os" "reflect" "sync" "testing" "time" "github.com/safing/portbase/database/query" "github.com/safing/portbase/database/record" "github.com/safing/portbase/database/storage" ) var ( // Compile time interface checks. _ storage.Interface = &BBolt{} _ storage.Batcher = &BBolt{} _ storage.Purger = &BBolt{} ) type TestRecord struct { record.Base sync.Mutex S string I int I8 int8 I16 int16 I32 int32 I64 int64 UI uint UI8 uint8 UI16 uint16 UI32 uint32 UI64 uint64 F32 float32 F64 float64 B bool } func TestBBolt(t *testing.T) { testDir, err := ioutil.TempDir("", "testing-") if err != nil { t.Fatal(err) } defer os.RemoveAll(testDir) // clean up // start db, err := NewBBolt("test", testDir) if err != nil { t.Fatal(err) } a := &TestRecord{ S: "banana", I: 42, I8: 42, I16: 42, I32: 42, I64: 42, UI: 42, UI8: 42, UI16: 42, UI32: 42, UI64: 42, F32: 42.42, F64: 42.42, B: true, } a.SetMeta(&record.Meta{}) a.Meta().Update() a.SetKey("test:A") // put record _, err = db.Put(a) if err != nil { t.Fatal(err) } // get and compare r1, err := db.Get("A") if err != nil { t.Fatal(err) } a1 := &TestRecord{} err = record.Unwrap(r1, a1) if err != nil { t.Fatal(err) } if !reflect.DeepEqual(a, a1) { t.Fatalf("mismatch, got %v", a1) } // setup query test records qA := &TestRecord{} qA.SetKey("test:path/to/A") qA.CreateMeta() qB := &TestRecord{} qB.SetKey("test:path/to/B") qB.CreateMeta() qC := &TestRecord{} qC.SetKey("test:path/to/C") qC.CreateMeta() qZ := &TestRecord{} qZ.SetKey("test:z") qZ.CreateMeta() // put _, err = db.Put(qA) if err == nil { _, err = db.Put(qB) } if err == nil { _, err = db.Put(qC) } if err == nil { _, err = db.Put(qZ) } if err != nil { t.Fatal(err) } // test query q := query.New("test:path/to/").MustBeValid() it, err := db.Query(q, true, true) if err != nil { t.Fatal(err) } cnt := 0 for range it.Next { cnt++ } if it.Err() != nil { t.Fatal(it.Err()) } if cnt != 3 { t.Fatalf("unexpected query result count: %d", cnt) } // delete err = db.Delete("A") if err != nil { t.Fatal(err) } // check if its gone _, err = db.Get("A") if err == nil { t.Fatal("should fail") } // maintenance err = db.MaintainRecordStates(context.TODO(), time.Now(), true) if err != nil { t.Fatal(err) } // maintenance err = db.MaintainRecordStates(context.TODO(), time.Now(), false) if err != nil { t.Fatal(err) } // purging purger, ok := db.(storage.Purger) if ok { n, err := purger.Purge(context.TODO(), query.New("test:path/to/").MustBeValid(), true, true, false) if err != nil { t.Fatal(err) } if n != 3 { t.Fatalf("unexpected purge delete count: %d", n) } } else { t.Fatal("should implement Purger") } // test query q = query.New("test").MustBeValid() it, err = db.Query(q, true, true) if err != nil { t.Fatal(err) } cnt = 0 for range it.Next { cnt++ } if it.Err() != nil { t.Fatal(it.Err()) } if cnt != 1 { t.Fatalf("unexpected query result count: %d", cnt) } // shutdown err = db.Shutdown() if err != nil { t.Fatal(err) } }