mirror of https://github.com/etcd-io/bbolt.git
Tests: Migrate from test DB to btesting.DB
Signed-off-by: Piotr Tabor <ptab@google.com>pull/360/head
parent
f3e164d8dd
commit
26cc5e3e8b
144
bucket_test.go
144
bucket_test.go
|
@ -15,13 +15,14 @@ import (
|
|||
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
bolt "go.etcd.io/bbolt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
)
|
||||
|
||||
// Ensure that a bucket that gets a non-existent key returns nil.
|
||||
func TestBucket_Get_NonExistent(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -39,8 +40,7 @@ func TestBucket_Get_NonExistent(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can read a value that is not flushed yet.
|
||||
func TestBucket_Get_FromNode(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -61,8 +61,7 @@ func TestBucket_Get_FromNode(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket retrieved via Get() returns a nil.
|
||||
func TestBucket_Get_IncompatibleValue(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -87,8 +86,7 @@ func TestBucket_Get_IncompatibleValue(t *testing.T) {
|
|||
//
|
||||
// https://github.com/boltdb/bolt/issues/544
|
||||
func TestBucket_Get_Capacity(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Write key to a bucket.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -125,8 +123,7 @@ func TestBucket_Get_Capacity(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can write a key/value.
|
||||
func TestBucket_Put(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -148,8 +145,7 @@ func TestBucket_Put(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can rewrite a key in the same transaction.
|
||||
func TestBucket_Put_Repeat(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -174,8 +170,7 @@ func TestBucket_Put_Repeat(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can write a bunch of large values.
|
||||
func TestBucket_Put_Large(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
count, factor := 100, 200
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -216,8 +211,7 @@ func TestDB_Put_VeryLarge(t *testing.T) {
|
|||
n, batchN := 400000, 200000
|
||||
ksize, vsize := 8, 500
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
for i := 0; i < n; i += batchN {
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -241,8 +235,7 @@ func TestDB_Put_VeryLarge(t *testing.T) {
|
|||
|
||||
// Ensure that a setting a value on a key with a bucket value returns an error.
|
||||
func TestBucket_Put_IncompatibleValue(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b0, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -264,8 +257,7 @@ func TestBucket_Put_IncompatibleValue(t *testing.T) {
|
|||
|
||||
// Ensure that a setting a value while the transaction is closed returns an error.
|
||||
func TestBucket_Put_Closed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -287,8 +279,7 @@ func TestBucket_Put_Closed(t *testing.T) {
|
|||
|
||||
// Ensure that setting a value on a read-only bucket returns an error.
|
||||
func TestBucket_Put_ReadOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -312,8 +303,7 @@ func TestBucket_Put_ReadOnly(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can delete an existing key.
|
||||
func TestBucket_Delete(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -337,8 +327,7 @@ func TestBucket_Delete(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a large set of keys will work correctly.
|
||||
func TestBucket_Delete_Large(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -388,8 +377,7 @@ func TestBucket_Delete_FreelistOverflow(t *testing.T) {
|
|||
t.Skip("skipping test in short mode.")
|
||||
}
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
k := make([]byte, 16)
|
||||
// The bigger the pages - the more values we need to write.
|
||||
|
@ -436,9 +424,7 @@ func TestBucket_Delete_FreelistOverflow(t *testing.T) {
|
|||
}
|
||||
|
||||
// Free page count should be preserved on reopen.
|
||||
if err := db.DB.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
db.MustClose()
|
||||
db.MustReopen()
|
||||
if reopenFreePages := db.Stats().FreePageN; freePages != reopenFreePages {
|
||||
t.Fatalf("expected %d free pages, got %+v", freePages, db.Stats())
|
||||
|
@ -447,8 +433,7 @@ func TestBucket_Delete_FreelistOverflow(t *testing.T) {
|
|||
|
||||
// Ensure that deleting of non-existing key is a no-op.
|
||||
func TestBucket_Delete_NonExisting(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -480,8 +465,7 @@ func TestBucket_Delete_NonExisting(t *testing.T) {
|
|||
|
||||
// Ensure that accessing and updating nested buckets is ok across transactions.
|
||||
func TestBucket_Nested(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
// Create a widgets bucket.
|
||||
|
@ -567,8 +551,7 @@ func TestBucket_Nested(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket using Delete() returns an error.
|
||||
func TestBucket_Delete_Bucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -588,8 +571,7 @@ func TestBucket_Delete_Bucket(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a key on a read-only bucket returns an error.
|
||||
func TestBucket_Delete_ReadOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -612,8 +594,7 @@ func TestBucket_Delete_ReadOnly(t *testing.T) {
|
|||
|
||||
// Ensure that a deleting value while the transaction is closed returns an error.
|
||||
func TestBucket_Delete_Closed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
|
@ -635,8 +616,7 @@ func TestBucket_Delete_Closed(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket causes nested buckets to be deleted.
|
||||
func TestBucket_DeleteBucket_Nested(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -667,8 +647,7 @@ func TestBucket_DeleteBucket_Nested(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket causes nested buckets to be deleted after they have been committed.
|
||||
func TestBucket_DeleteBucket_Nested2(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -734,8 +713,7 @@ func TestBucket_DeleteBucket_Nested2(t *testing.T) {
|
|||
// Ensure that deleting a child bucket with multiple pages causes all pages to get collected.
|
||||
// NOTE: Consistency check in bolt_test.DB.Close() will panic if pages not freed properly.
|
||||
func TestBucket_DeleteBucket_Large(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -770,8 +748,7 @@ func TestBucket_DeleteBucket_Large(t *testing.T) {
|
|||
|
||||
// Ensure that a simple value retrieved via Bucket() returns a nil.
|
||||
func TestBucket_Bucket_IncompatibleValue(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -793,8 +770,7 @@ func TestBucket_Bucket_IncompatibleValue(t *testing.T) {
|
|||
|
||||
// Ensure that creating a bucket on an existing non-bucket key returns an error.
|
||||
func TestBucket_CreateBucket_IncompatibleValue(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -815,8 +791,7 @@ func TestBucket_CreateBucket_IncompatibleValue(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket on an existing non-bucket key returns an error.
|
||||
func TestBucket_DeleteBucket_IncompatibleValue(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -837,8 +812,7 @@ func TestBucket_DeleteBucket_IncompatibleValue(t *testing.T) {
|
|||
|
||||
// Ensure bucket can set and update its sequence number.
|
||||
func TestBucket_Sequence(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
bkt, err := tx.CreateBucket([]byte("0"))
|
||||
|
@ -879,8 +853,7 @@ func TestBucket_Sequence(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can return an autoincrementing sequence.
|
||||
func TestBucket_NextSequence(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
widgets, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -922,8 +895,7 @@ func TestBucket_NextSequence(t *testing.T) {
|
|||
// the only thing updated on the bucket.
|
||||
// https://github.com/boltdb/bolt/issues/296
|
||||
func TestBucket_NextSequence_Persist(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -958,8 +930,7 @@ func TestBucket_NextSequence_Persist(t *testing.T) {
|
|||
|
||||
// Ensure that retrieving the next sequence on a read-only bucket returns an error.
|
||||
func TestBucket_NextSequence_ReadOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -983,8 +954,7 @@ func TestBucket_NextSequence_ReadOnly(t *testing.T) {
|
|||
|
||||
// Ensure that retrieving the next sequence for a bucket on a closed database return an error.
|
||||
func TestBucket_NextSequence_Closed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -1003,8 +973,7 @@ func TestBucket_NextSequence_Closed(t *testing.T) {
|
|||
|
||||
// Ensure a user can loop over all key/value pairs in a bucket.
|
||||
func TestBucket_ForEach(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
type kv struct {
|
||||
k []byte
|
||||
|
@ -1053,8 +1022,7 @@ func TestBucket_ForEach(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestBucket_ForEachBucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
expectedItems := [][]byte{
|
||||
[]byte("csubbucket"),
|
||||
|
@ -1098,8 +1066,7 @@ func TestBucket_ForEachBucket(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestBucket_ForEachBucket_NoBuckets(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
verifyReads := func(b *bolt.Bucket) {
|
||||
var items [][]byte
|
||||
|
@ -1137,8 +1104,7 @@ func TestBucket_ForEachBucket_NoBuckets(t *testing.T) {
|
|||
|
||||
// Ensure a database can stop iteration early.
|
||||
func TestBucket_ForEach_ShortCircuit(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -1176,8 +1142,7 @@ func TestBucket_ForEach_ShortCircuit(t *testing.T) {
|
|||
|
||||
// Ensure that looping over a bucket on a closed database returns an error.
|
||||
func TestBucket_ForEach_Closed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
|
@ -1200,8 +1165,7 @@ func TestBucket_ForEach_Closed(t *testing.T) {
|
|||
|
||||
// Ensure that an error is returned when inserting with an empty key.
|
||||
func TestBucket_Put_EmptyKey(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -1222,8 +1186,7 @@ func TestBucket_Put_EmptyKey(t *testing.T) {
|
|||
|
||||
// Ensure that an error is returned when inserting with a key that's too large.
|
||||
func TestBucket_Put_KeyTooLarge(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -1245,8 +1208,7 @@ func TestBucket_Put_ValueTooLarge(t *testing.T) {
|
|||
t.Skip("not enough RAM for test")
|
||||
}
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -1268,8 +1230,7 @@ func TestBucket_Stats(t *testing.T) {
|
|||
t.Skip("skipping test in short mode")
|
||||
}
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Add bucket with fewer keys but one big value.
|
||||
bigKey := []byte("really-big-value")
|
||||
|
@ -1361,8 +1322,7 @@ func TestBucket_Stats_RandomFill(t *testing.T) {
|
|||
t.Skip("invalid page size for test")
|
||||
}
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Add a set of values in random order. It will be the same random
|
||||
// order so we can maintain consistency between test runs.
|
||||
|
@ -1423,8 +1383,7 @@ func TestBucket_Stats_RandomFill(t *testing.T) {
|
|||
|
||||
// Ensure a bucket can calculate stats.
|
||||
func TestBucket_Stats_Small(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
// Add a bucket that fits on a single root leaf.
|
||||
|
@ -1487,8 +1446,7 @@ func TestBucket_Stats_Small(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestBucket_Stats_EmptyBucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
// Add a bucket that fits on a single root leaf.
|
||||
|
@ -1547,8 +1505,7 @@ func TestBucket_Stats_EmptyBucket(t *testing.T) {
|
|||
|
||||
// Ensure a bucket can calculate stats.
|
||||
func TestBucket_Stats_Nested(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("foo"))
|
||||
|
@ -1653,8 +1610,7 @@ func TestBucket_Stats_Large(t *testing.T) {
|
|||
t.Skip("skipping test in short mode.")
|
||||
}
|
||||
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var index int
|
||||
for i := 0; i < 100; i++ {
|
||||
|
@ -1731,7 +1687,7 @@ func TestBucket_Put_Single(t *testing.T) {
|
|||
|
||||
index := 0
|
||||
if err := quick.Check(func(items testdata) bool {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.MustClose()
|
||||
|
||||
m := make(map[string][]byte)
|
||||
|
@ -1788,7 +1744,7 @@ func TestBucket_Put_Multiple(t *testing.T) {
|
|||
}
|
||||
|
||||
if err := quick.Check(func(items testdata) bool {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.MustClose()
|
||||
|
||||
// Bulk insert all values.
|
||||
|
@ -1841,7 +1797,7 @@ func TestBucket_Delete_Quick(t *testing.T) {
|
|||
}
|
||||
|
||||
if err := quick.Check(func(items testdata) bool {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.MustClose()
|
||||
|
||||
// Bulk insert all values.
|
||||
|
|
|
@ -4,6 +4,7 @@ import (
|
|||
"bytes"
|
||||
"encoding/binary"
|
||||
"fmt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
"log"
|
||||
"os"
|
||||
"reflect"
|
||||
|
@ -16,8 +17,7 @@ import (
|
|||
|
||||
// Ensure that a cursor can return a reference to the bucket that created it.
|
||||
func TestCursor_Bucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -34,8 +34,7 @@ func TestCursor_Bucket(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can seek to the appropriate keys.
|
||||
func TestCursor_Seek(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -104,8 +103,7 @@ func TestCursor_Seek(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestCursor_Delete(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
const count = 1000
|
||||
|
||||
|
@ -167,8 +165,7 @@ func TestCursor_Delete(t *testing.T) {
|
|||
//
|
||||
// Related: https://github.com/boltdb/bolt/pull/187
|
||||
func TestCursor_Seek_Large(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var count = 10000
|
||||
|
||||
|
@ -231,8 +228,7 @@ func TestCursor_Seek_Large(t *testing.T) {
|
|||
|
||||
// Ensure that a cursor can iterate over an empty bucket without error.
|
||||
func TestCursor_EmptyBucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
return err
|
||||
|
@ -256,8 +252,7 @@ func TestCursor_EmptyBucket(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can reverse iterate over an empty bucket without error.
|
||||
func TestCursor_EmptyBucketReverse(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -281,8 +276,7 @@ func TestCursor_EmptyBucketReverse(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can iterate over a single root with a couple elements.
|
||||
func TestCursor_Iterate_Leaf(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -352,8 +346,7 @@ func TestCursor_Iterate_Leaf(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can iterate in reverse over a single root with a couple elements.
|
||||
func TestCursor_LeafRootReverse(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -416,8 +409,7 @@ func TestCursor_LeafRootReverse(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can restart from the beginning.
|
||||
func TestCursor_Restart(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -462,8 +454,7 @@ func TestCursor_Restart(t *testing.T) {
|
|||
|
||||
// Ensure that a cursor can skip over empty pages that have been deleted.
|
||||
func TestCursor_First_EmptyPages(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Create 1000 keys in the "widgets" bucket.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -509,8 +500,7 @@ func TestCursor_First_EmptyPages(t *testing.T) {
|
|||
|
||||
// Ensure that a cursor can skip over empty pages that have been deleted.
|
||||
func TestCursor_Last_EmptyPages(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Create 1000 keys in the "widgets" bucket.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -557,7 +547,7 @@ func TestCursor_Last_EmptyPages(t *testing.T) {
|
|||
// Ensure that a Tx can iterate over all elements in a bucket.
|
||||
func TestCursor_QuickCheck(t *testing.T) {
|
||||
f := func(items testdata) bool {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.MustClose()
|
||||
|
||||
// Bulk insert all values.
|
||||
|
@ -615,7 +605,7 @@ func TestCursor_QuickCheck(t *testing.T) {
|
|||
// Ensure that a transaction can iterate over all elements in a bucket in reverse.
|
||||
func TestCursor_QuickCheck_Reverse(t *testing.T) {
|
||||
f := func(items testdata) bool {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.MustClose()
|
||||
|
||||
// Bulk insert all values.
|
||||
|
@ -671,8 +661,7 @@ func TestCursor_QuickCheck_Reverse(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can iterate over subbuckets.
|
||||
func TestCursor_QuickCheck_BucketsOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -713,8 +702,7 @@ func TestCursor_QuickCheck_BucketsOnly(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx cursor can reverse iterate over subbuckets.
|
||||
func TestCursor_QuickCheck_BucketsOnly_Reverse(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
|
274
db_test.go
274
db_test.go
|
@ -4,24 +4,22 @@ import (
|
|||
"bytes"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"flag"
|
||||
"fmt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
"hash/fnv"
|
||||
"log"
|
||||
"math/rand"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"sync"
|
||||
"testing"
|
||||
"time"
|
||||
"unsafe"
|
||||
|
||||
bolt "go.etcd.io/bbolt"
|
||||
)
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
var statsFlag = flag.Bool("stats", false, "show performance stats")
|
||||
bolt "go.etcd.io/bbolt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
)
|
||||
|
||||
// pageSize is the size of one page in the data file.
|
||||
const pageSize = 4096
|
||||
|
@ -148,12 +146,11 @@ func TestOpen_ErrVersionMismatch(t *testing.T) {
|
|||
}
|
||||
|
||||
// Create empty database.
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
path := db.Path()
|
||||
defer db.MustClose()
|
||||
|
||||
// Close database.
|
||||
if err := db.DB.Close(); err != nil {
|
||||
if err := db.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
|
@ -185,12 +182,11 @@ func TestOpen_ErrChecksum(t *testing.T) {
|
|||
}
|
||||
|
||||
// Create empty database.
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
path := db.Path()
|
||||
defer db.MustClose()
|
||||
|
||||
// Close database.
|
||||
if err := db.DB.Close(); err != nil {
|
||||
if err := db.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
|
@ -272,9 +268,8 @@ func TestOpen_Size_Large(t *testing.T) {
|
|||
}
|
||||
|
||||
// Open a data file.
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
path := db.Path()
|
||||
defer db.MustClose()
|
||||
|
||||
pagesize := db.Info().PageSize
|
||||
|
||||
|
@ -454,8 +449,7 @@ func TestDB_Open_InitialMmapSize(t *testing.T) {
|
|||
// TestDB_Open_ReadOnly checks a database in read only mode can read but not write.
|
||||
func TestDB_Open_ReadOnly(t *testing.T) {
|
||||
// Create a writable db, write k-v and close it.
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -473,7 +467,7 @@ func TestDB_Open_ReadOnly(t *testing.T) {
|
|||
t.Fatal(err)
|
||||
}
|
||||
|
||||
f := db.f
|
||||
f := db.Path()
|
||||
o := &bolt.Options{ReadOnly: true}
|
||||
readOnlyDB, err := bolt.Open(f, 0666, o)
|
||||
if err != nil {
|
||||
|
@ -510,13 +504,11 @@ func TestDB_Open_ReadOnly(t *testing.T) {
|
|||
func TestOpen_BigPage(t *testing.T) {
|
||||
pageSize := os.Getpagesize()
|
||||
|
||||
db1 := MustOpenWithOption(&bolt.Options{PageSize: pageSize * 2})
|
||||
defer db1.MustClose()
|
||||
db1 := btesting.MustCreateDBWithOption(t, &bolt.Options{PageSize: pageSize * 2})
|
||||
|
||||
db2 := MustOpenWithOption(&bolt.Options{PageSize: pageSize * 4})
|
||||
defer db2.MustClose()
|
||||
db2 := btesting.MustCreateDBWithOption(t, &bolt.Options{PageSize: pageSize * 4})
|
||||
|
||||
if db1sz, db2sz := fileSize(db1.f), fileSize(db2.f); db1sz >= db2sz {
|
||||
if db1sz, db2sz := fileSize(db1.Path()), fileSize(db2.Path()); db1sz >= db2sz {
|
||||
t.Errorf("expected %d < %d", db1sz, db2sz)
|
||||
}
|
||||
}
|
||||
|
@ -525,8 +517,7 @@ func TestOpen_BigPage(t *testing.T) {
|
|||
// write-out after no free list sync will recover the free list
|
||||
// and write it out.
|
||||
func TestOpen_RecoverFreeList(t *testing.T) {
|
||||
db := MustOpenWithOption(&bolt.Options{NoFreelistSync: true})
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, &bolt.Options{NoFreelistSync: true})
|
||||
|
||||
// Write some pages.
|
||||
tx, err := db.Begin(true)
|
||||
|
@ -568,6 +559,7 @@ func TestOpen_RecoverFreeList(t *testing.T) {
|
|||
if err := db.DB.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
db.MustClose()
|
||||
|
||||
// Record freelist count from opening with NoFreelistSync.
|
||||
db.MustReopen()
|
||||
|
@ -578,9 +570,10 @@ func TestOpen_RecoverFreeList(t *testing.T) {
|
|||
if err := db.DB.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
db.MustClose()
|
||||
|
||||
// Check free page count is reconstructed when opened with freelist sync.
|
||||
db.o = &bolt.Options{}
|
||||
db.SetOptions(&bolt.Options{})
|
||||
db.MustReopen()
|
||||
// One less free page for syncing the free list on open.
|
||||
freepages--
|
||||
|
@ -599,33 +592,22 @@ func TestDB_Begin_ErrDatabaseNotOpen(t *testing.T) {
|
|||
|
||||
// Ensure that a read-write transaction can be retrieved.
|
||||
func TestDB_BeginRW(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
} else if tx == nil {
|
||||
t.Fatal("expected tx")
|
||||
}
|
||||
require.NoError(t, err)
|
||||
require.NotNil(t, tx, "expected tx")
|
||||
defer func() { require.NoError(t, tx.Commit()) }()
|
||||
|
||||
if tx.DB() != db.DB {
|
||||
t.Fatal("unexpected tx database")
|
||||
} else if !tx.Writable() {
|
||||
t.Fatal("expected writable tx")
|
||||
}
|
||||
|
||||
if err := tx.Commit(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
require.True(t, tx.Writable(), "expected writable tx")
|
||||
require.Same(t, db.DB, tx.DB())
|
||||
}
|
||||
|
||||
// TestDB_Concurrent_WriteTo checks that issuing WriteTo operations concurrently
|
||||
// with commits does not produce corrupted db files.
|
||||
func TestDB_Concurrent_WriteTo(t *testing.T) {
|
||||
o := &bolt.Options{NoFreelistSync: false}
|
||||
db := MustOpenWithOption(o)
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, o)
|
||||
|
||||
var wg sync.WaitGroup
|
||||
wtxs, rtxs := 5, 5
|
||||
|
@ -646,8 +628,7 @@ func TestDB_Concurrent_WriteTo(t *testing.T) {
|
|||
panic(err)
|
||||
}
|
||||
f.Close()
|
||||
snap := &DB{nil, f.Name(), o}
|
||||
snap.MustReopen()
|
||||
snap := btesting.MustOpenDBWithOption(t, f.Name(), o)
|
||||
defer snap.MustClose()
|
||||
snap.MustCheck()
|
||||
}
|
||||
|
@ -698,7 +679,7 @@ func TestDB_Close_PendingTx_RO(t *testing.T) { testDB_Close_PendingTx(t, false)
|
|||
|
||||
// Ensure that a database cannot close while transactions are open.
|
||||
func testDB_Close_PendingTx(t *testing.T, writable bool) {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Start transaction.
|
||||
tx, err := db.Begin(writable)
|
||||
|
@ -748,8 +729,7 @@ func testDB_Close_PendingTx(t *testing.T, writable bool) {
|
|||
|
||||
// Ensure a database can provide a transactional block.
|
||||
func TestDB_Update(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -797,8 +777,7 @@ func TestDB_Update_Closed(t *testing.T) {
|
|||
|
||||
// Ensure a panic occurs while trying to commit a managed transaction.
|
||||
func TestDB_Update_ManualCommit(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var panicked bool
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -823,8 +802,7 @@ func TestDB_Update_ManualCommit(t *testing.T) {
|
|||
|
||||
// Ensure a panic occurs while trying to rollback a managed transaction.
|
||||
func TestDB_Update_ManualRollback(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var panicked bool
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -849,8 +827,7 @@ func TestDB_Update_ManualRollback(t *testing.T) {
|
|||
|
||||
// Ensure a panic occurs while trying to commit a managed transaction.
|
||||
func TestDB_View_ManualCommit(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var panicked bool
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
|
@ -875,8 +852,7 @@ func TestDB_View_ManualCommit(t *testing.T) {
|
|||
|
||||
// Ensure a panic occurs while trying to rollback a managed transaction.
|
||||
func TestDB_View_ManualRollback(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var panicked bool
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
|
@ -901,8 +877,7 @@ func TestDB_View_ManualRollback(t *testing.T) {
|
|||
|
||||
// Ensure a write transaction that panics does not hold open locks.
|
||||
func TestDB_Update_Panic(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Panic during update but recover.
|
||||
func() {
|
||||
|
@ -945,8 +920,7 @@ func TestDB_Update_Panic(t *testing.T) {
|
|||
|
||||
// Ensure a database can return an error through a read-only transactional block.
|
||||
func TestDB_View_Error(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
return errors.New("xxx")
|
||||
|
@ -957,8 +931,7 @@ func TestDB_View_Error(t *testing.T) {
|
|||
|
||||
// Ensure a read transaction that panics does not hold open locks.
|
||||
func TestDB_View_Panic(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -1000,8 +973,7 @@ func TestDB_View_Panic(t *testing.T) {
|
|||
|
||||
// Ensure that DB stats can be returned.
|
||||
func TestDB_Stats(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
return err
|
||||
|
@ -1021,8 +993,7 @@ func TestDB_Stats(t *testing.T) {
|
|||
|
||||
// Ensure that database pages are in expected order and type.
|
||||
func TestDB_Consistency(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
return err
|
||||
|
@ -1107,8 +1078,7 @@ func TestDBStats_Sub(t *testing.T) {
|
|||
|
||||
// Ensure two functions can perform updates in a single batch.
|
||||
func TestDB_Batch(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
|
@ -1152,8 +1122,7 @@ func TestDB_Batch(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestDB_Batch_Panic(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var sentinel int
|
||||
var bork = &sentinel
|
||||
|
@ -1183,8 +1152,7 @@ func TestDB_Batch_Panic(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestDB_BatchFull(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
return err
|
||||
|
@ -1242,8 +1210,7 @@ func TestDB_BatchFull(t *testing.T) {
|
|||
}
|
||||
|
||||
func TestDB_BatchTime(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("widgets"))
|
||||
return err
|
||||
|
@ -1431,8 +1398,8 @@ func ExampleDB_Begin() {
|
|||
}
|
||||
|
||||
func BenchmarkDBBatchAutomatic(b *testing.B) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(b)
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("bench"))
|
||||
return err
|
||||
|
@ -1476,8 +1443,7 @@ func BenchmarkDBBatchAutomatic(b *testing.B) {
|
|||
}
|
||||
|
||||
func BenchmarkDBBatchSingle(b *testing.B) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(b)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("bench"))
|
||||
return err
|
||||
|
@ -1520,8 +1486,7 @@ func BenchmarkDBBatchSingle(b *testing.B) {
|
|||
}
|
||||
|
||||
func BenchmarkDBBatchManual10x100(b *testing.B) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(b)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("bench"))
|
||||
return err
|
||||
|
@ -1574,7 +1539,7 @@ func BenchmarkDBBatchManual10x100(b *testing.B) {
|
|||
validateBatchBench(b, db)
|
||||
}
|
||||
|
||||
func validateBatchBench(b *testing.B, db *DB) {
|
||||
func validateBatchBench(b *testing.B, db *btesting.DB) {
|
||||
var rollback = errors.New("sentinel error to cause rollback")
|
||||
validate := func(tx *bolt.Tx) error {
|
||||
bucket := tx.Bucket([]byte("bench"))
|
||||
|
@ -1609,149 +1574,6 @@ func validateBatchBench(b *testing.B, db *DB) {
|
|||
}
|
||||
}
|
||||
|
||||
// DB is a test wrapper for bolt.DB.
|
||||
type DB struct {
|
||||
*bolt.DB
|
||||
f string
|
||||
o *bolt.Options
|
||||
}
|
||||
|
||||
// MustOpenDB returns a new, open DB at a temporary location.
|
||||
// Deprecated: Please use btesting.MustCreateDB(...).
|
||||
func MustOpenDB() *DB {
|
||||
return MustOpenWithOption(nil)
|
||||
}
|
||||
|
||||
// MustOpenDBWithOption returns a new, open DB at a temporary location with given options.
|
||||
// Deprecated: Please use btesting.MustCreateWithOption(...).
|
||||
func MustOpenWithOption(o *bolt.Options) *DB {
|
||||
f := tempfile()
|
||||
if o == nil {
|
||||
o = bolt.DefaultOptions
|
||||
}
|
||||
|
||||
freelistType := bolt.FreelistArrayType
|
||||
if env := os.Getenv(bolt.TestFreelistType); env == string(bolt.FreelistMapType) {
|
||||
freelistType = bolt.FreelistMapType
|
||||
}
|
||||
o.FreelistType = freelistType
|
||||
|
||||
db, err := bolt.Open(f, 0666, o)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
return &DB{
|
||||
DB: db,
|
||||
f: f,
|
||||
o: o,
|
||||
}
|
||||
}
|
||||
|
||||
// Closes the DB without removing the file.
|
||||
// Allows for Reopen().
|
||||
func (db *DB) CloseTemporarily() error {
|
||||
if err := db.DB.Close(); err != nil {
|
||||
return err
|
||||
}
|
||||
db.DB = nil
|
||||
return nil
|
||||
}
|
||||
|
||||
// Close closes the database and deletes the underlying file.
|
||||
func (db *DB) Close() error {
|
||||
// Log statistics.
|
||||
if *statsFlag {
|
||||
db.PrintStats()
|
||||
}
|
||||
|
||||
// Check database consistency after every test.
|
||||
db.MustCheck()
|
||||
|
||||
// Close database and remove file.
|
||||
defer os.Remove(db.Path())
|
||||
return db.DB.Close()
|
||||
}
|
||||
|
||||
// MustClose closes the database and deletes the underlying file. Panic on error.
|
||||
func (db *DB) MustClose() {
|
||||
if err := db.Close(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
|
||||
// MustReopen reopen the database. Panic on error.
|
||||
func (db *DB) MustReopen() {
|
||||
indb, err := bolt.Open(db.f, 0666, db.o)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
db.DB = indb
|
||||
}
|
||||
|
||||
// PrintStats prints the database stats
|
||||
func (db *DB) PrintStats() {
|
||||
var stats = db.Stats()
|
||||
fmt.Printf("[db] %-20s %-20s %-20s\n",
|
||||
fmt.Sprintf("pg(%d/%d)", stats.TxStats.PageCount, stats.TxStats.PageAlloc),
|
||||
fmt.Sprintf("cur(%d)", stats.TxStats.CursorCount),
|
||||
fmt.Sprintf("node(%d/%d)", stats.TxStats.NodeCount, stats.TxStats.NodeDeref),
|
||||
)
|
||||
fmt.Printf(" %-20s %-20s %-20s\n",
|
||||
fmt.Sprintf("rebal(%d/%v)", stats.TxStats.Rebalance, truncDuration(stats.TxStats.RebalanceTime)),
|
||||
fmt.Sprintf("spill(%d/%v)", stats.TxStats.Spill, truncDuration(stats.TxStats.SpillTime)),
|
||||
fmt.Sprintf("w(%d/%v)", stats.TxStats.Write, truncDuration(stats.TxStats.WriteTime)),
|
||||
)
|
||||
}
|
||||
|
||||
// MustCheck runs a consistency check on the database and panics if any errors are found.
|
||||
func (db *DB) MustCheck() {
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
// Collect all the errors.
|
||||
var errors []error
|
||||
for err := range tx.Check() {
|
||||
errors = append(errors, err)
|
||||
if len(errors) > 10 {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// If errors occurred, copy the DB and print the errors.
|
||||
if len(errors) > 0 {
|
||||
var path = tempfile()
|
||||
if err := tx.CopyFile(path, 0600); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
// Print errors.
|
||||
fmt.Print("\n\n")
|
||||
fmt.Printf("consistency check failed (%d errors)\n", len(errors))
|
||||
for _, err := range errors {
|
||||
fmt.Println(err)
|
||||
}
|
||||
fmt.Println("")
|
||||
fmt.Println("db saved to:")
|
||||
fmt.Println(path)
|
||||
fmt.Print("\n\n")
|
||||
os.Exit(-1)
|
||||
}
|
||||
|
||||
return nil
|
||||
}); err != nil && err != bolt.ErrDatabaseNotOpen {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
|
||||
// CopyTempFile copies a database to a temporary file.
|
||||
func (db *DB) CopyTempFile() {
|
||||
path := tempfile()
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
return tx.CopyFile(path, 0600)
|
||||
}); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
fmt.Println("db copied to: ", path)
|
||||
}
|
||||
|
||||
// tempfile returns a temporary file path.
|
||||
func tempfile() string {
|
||||
f, err := os.CreateTemp("", "bolt-")
|
||||
|
@ -1774,10 +1596,6 @@ func trunc(b []byte, length int) []byte {
|
|||
return b
|
||||
}
|
||||
|
||||
func truncDuration(d time.Duration) string {
|
||||
return regexp.MustCompile(`^(\d+)(\.\d+)`).ReplaceAllString(d.String(), "$1")
|
||||
}
|
||||
|
||||
func fileSize(path string) int64 {
|
||||
fi, err := os.Stat(path)
|
||||
if err != nil {
|
||||
|
|
|
@ -3,6 +3,7 @@ package btesting
|
|||
import (
|
||||
"flag"
|
||||
"fmt"
|
||||
"github.com/stretchr/testify/require"
|
||||
bolt "go.etcd.io/bbolt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
|
@ -36,15 +37,15 @@ func MustCreateDBWithOption(t testing.TB, o *bolt.Options) *DB {
|
|||
}
|
||||
|
||||
func MustOpenDBWithOption(t testing.TB, f string, o *bolt.Options) *DB {
|
||||
t.Logf("Opening bbolt DB at: %s", f)
|
||||
if o == nil {
|
||||
o = bolt.DefaultOptions
|
||||
}
|
||||
|
||||
freelistType := bolt.FreelistMapType
|
||||
if env := os.Getenv(TestFreelistType); env == string(bolt.FreelistArrayType) {
|
||||
freelistType = bolt.FreelistArrayType
|
||||
freelistType := bolt.FreelistArrayType
|
||||
if env := os.Getenv(TestFreelistType); env == string(bolt.FreelistMapType) {
|
||||
freelistType = bolt.FreelistMapType
|
||||
}
|
||||
o.FreelistType = freelistType
|
||||
|
||||
o.FreelistType = freelistType
|
||||
|
||||
|
@ -77,7 +78,7 @@ func (db *DB) Close() error {
|
|||
if *statsFlag {
|
||||
db.PrintStats()
|
||||
}
|
||||
|
||||
db.t.Logf("Closing bbolt DB at: %s", db.f)
|
||||
err := db.DB.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
|
@ -87,13 +88,18 @@ func (db *DB) Close() error {
|
|||
return nil
|
||||
}
|
||||
|
||||
// MustClose closes the database and deletes the underlying file. Panic on error.
|
||||
// MustClose closes the database but does NOT delete the underlying file.
|
||||
func (db *DB) MustClose() {
|
||||
if err := db.Close(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
|
||||
func (db *DB) MustDeleteFile() {
|
||||
err := os.Remove(db.Path())
|
||||
require.NoError(db.t, err)
|
||||
}
|
||||
|
||||
func (db *DB) SetOptions(o *bolt.Options) {
|
||||
db.o = o
|
||||
}
|
||||
|
@ -103,6 +109,7 @@ func (db *DB) MustReopen() {
|
|||
if db.DB != nil {
|
||||
panic("Please call Close() before MustReopen()")
|
||||
}
|
||||
db.t.Logf("Reopening bbolt DB at: %s", db.f)
|
||||
indb, err := bolt.Open(db.Path(), 0666, db.o)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
|
|
|
@ -3,6 +3,7 @@ package bbolt_test
|
|||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
"math/rand"
|
||||
"sync"
|
||||
"sync/atomic"
|
||||
|
@ -43,8 +44,7 @@ func testSimulate(t *testing.T, openOption *bolt.Options, round, threadCount, pa
|
|||
var versions = make(map[int]*QuickDB)
|
||||
versions[1] = NewQuickDB()
|
||||
|
||||
db := MustOpenWithOption(openOption)
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, openOption)
|
||||
|
||||
var mutex sync.Mutex
|
||||
|
||||
|
@ -146,6 +146,9 @@ func testSimulate(t *testing.T, openOption *bolt.Options, round, threadCount, pa
|
|||
}
|
||||
|
||||
db.MustClose()
|
||||
// I have doubts the DB drop is indented here (as 'versions' is not being reset).
|
||||
// But I'm preserving for now the original behavior.
|
||||
db.MustDeleteFile()
|
||||
db.MustReopen()
|
||||
}
|
||||
|
||||
|
|
82
tx_test.go
82
tx_test.go
|
@ -4,6 +4,7 @@ import (
|
|||
"bytes"
|
||||
"errors"
|
||||
"fmt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
"log"
|
||||
"os"
|
||||
"testing"
|
||||
|
@ -13,7 +14,7 @@ import (
|
|||
|
||||
// TestTx_Check_ReadOnly tests consistency checking on a ReadOnly database.
|
||||
func TestTx_Check_ReadOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
db := btesting.MustCreateDB(t)
|
||||
defer db.Close()
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
|
@ -27,11 +28,11 @@ func TestTx_Check_ReadOnly(t *testing.T) {
|
|||
}); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if err := db.DB.Close(); err != nil {
|
||||
if err := db.Close(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
readOnlyDB, err := bolt.Open(db.f, 0666, &bolt.Options{ReadOnly: true})
|
||||
readOnlyDB, err := bolt.Open(db.Path(), 0666, &bolt.Options{ReadOnly: true})
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
@ -65,8 +66,7 @@ func TestTx_Check_ReadOnly(t *testing.T) {
|
|||
|
||||
// Ensure that committing a closed transaction returns an error.
|
||||
func TestTx_Commit_ErrTxClosed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -87,8 +87,7 @@ func TestTx_Commit_ErrTxClosed(t *testing.T) {
|
|||
|
||||
// Ensure that rolling back a closed transaction returns an error.
|
||||
func TestTx_Rollback_ErrTxClosed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
|
@ -105,8 +104,7 @@ func TestTx_Rollback_ErrTxClosed(t *testing.T) {
|
|||
|
||||
// Ensure that committing a read-only transaction returns an error.
|
||||
func TestTx_Commit_ErrTxNotWritable(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(false)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -123,8 +121,7 @@ func TestTx_Commit_ErrTxNotWritable(t *testing.T) {
|
|||
|
||||
// Ensure that a transaction can retrieve a cursor on the root bucket.
|
||||
func TestTx_Cursor(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -161,8 +158,7 @@ func TestTx_Cursor(t *testing.T) {
|
|||
|
||||
// Ensure that creating a bucket with a read-only transaction returns an error.
|
||||
func TestTx_CreateBucket_ErrTxNotWritable(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
_, err := tx.CreateBucket([]byte("foo"))
|
||||
if err != bolt.ErrTxNotWritable {
|
||||
|
@ -176,8 +172,7 @@ func TestTx_CreateBucket_ErrTxNotWritable(t *testing.T) {
|
|||
|
||||
// Ensure that creating a bucket on a closed transaction returns an error.
|
||||
func TestTx_CreateBucket_ErrTxClosed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -193,8 +188,7 @@ func TestTx_CreateBucket_ErrTxClosed(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx can retrieve a bucket.
|
||||
func TestTx_Bucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket([]byte("widgets")); err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -210,8 +204,7 @@ func TestTx_Bucket(t *testing.T) {
|
|||
|
||||
// Ensure that a Tx retrieving a non-existent key returns nil.
|
||||
func TestTx_Get_NotFound(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -232,8 +225,7 @@ func TestTx_Get_NotFound(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can be created and retrieved.
|
||||
func TestTx_CreateBucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Create a bucket.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -261,8 +253,7 @@ func TestTx_CreateBucket(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can be created if it doesn't already exist.
|
||||
func TestTx_CreateBucketIfNotExists(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
// Create bucket.
|
||||
if b, err := tx.CreateBucketIfNotExists([]byte("widgets")); err != nil {
|
||||
|
@ -296,8 +287,7 @@ func TestTx_CreateBucketIfNotExists(t *testing.T) {
|
|||
|
||||
// Ensure transaction returns an error if creating an unnamed bucket.
|
||||
func TestTx_CreateBucketIfNotExists_ErrBucketNameRequired(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucketIfNotExists([]byte{}); err != bolt.ErrBucketNameRequired {
|
||||
t.Fatalf("unexpected error: %s", err)
|
||||
|
@ -315,8 +305,7 @@ func TestTx_CreateBucketIfNotExists_ErrBucketNameRequired(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket cannot be created twice.
|
||||
func TestTx_CreateBucket_ErrBucketExists(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Create a bucket.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -341,8 +330,7 @@ func TestTx_CreateBucket_ErrBucketExists(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket is created with a non-blank name.
|
||||
func TestTx_CreateBucket_ErrBucketNameRequired(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if _, err := tx.CreateBucket(nil); err != bolt.ErrBucketNameRequired {
|
||||
t.Fatalf("unexpected error: %s", err)
|
||||
|
@ -355,8 +343,7 @@ func TestTx_CreateBucket_ErrBucketNameRequired(t *testing.T) {
|
|||
|
||||
// Ensure that a bucket can be deleted.
|
||||
func TestTx_DeleteBucket(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
// Create a bucket and add a value.
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -402,8 +389,7 @@ func TestTx_DeleteBucket(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket on a closed transaction returns an error.
|
||||
func TestTx_DeleteBucket_ErrTxClosed(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
tx, err := db.Begin(true)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
|
@ -418,8 +404,7 @@ func TestTx_DeleteBucket_ErrTxClosed(t *testing.T) {
|
|||
|
||||
// Ensure that deleting a bucket with a read-only transaction returns an error.
|
||||
func TestTx_DeleteBucket_ReadOnly(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.View(func(tx *bolt.Tx) error {
|
||||
if err := tx.DeleteBucket([]byte("foo")); err != bolt.ErrTxNotWritable {
|
||||
t.Fatalf("unexpected error: %s", err)
|
||||
|
@ -432,8 +417,7 @@ func TestTx_DeleteBucket_ReadOnly(t *testing.T) {
|
|||
|
||||
// Ensure that nothing happens when deleting a bucket that doesn't exist.
|
||||
func TestTx_DeleteBucket_NotFound(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
if err := tx.DeleteBucket([]byte("widgets")); err != bolt.ErrBucketNotFound {
|
||||
t.Fatalf("unexpected error: %s", err)
|
||||
|
@ -447,8 +431,7 @@ func TestTx_DeleteBucket_NotFound(t *testing.T) {
|
|||
// Ensure that no error is returned when a tx.ForEach function does not return
|
||||
// an error.
|
||||
func TestTx_ForEach_NoError(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -471,8 +454,7 @@ func TestTx_ForEach_NoError(t *testing.T) {
|
|||
|
||||
// Ensure that an error is returned when a tx.ForEach function returns an error.
|
||||
func TestTx_ForEach_WithError(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -496,8 +478,7 @@ func TestTx_ForEach_WithError(t *testing.T) {
|
|||
|
||||
// Ensure that Tx commit handlers are called after a transaction successfully commits.
|
||||
func TestTx_OnCommit(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var x int
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -516,8 +497,7 @@ func TestTx_OnCommit(t *testing.T) {
|
|||
|
||||
// Ensure that Tx commit handlers are NOT called after a transaction rolls back.
|
||||
func TestTx_OnCommit_Rollback(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
var x int
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -536,8 +516,7 @@ func TestTx_OnCommit_Rollback(t *testing.T) {
|
|||
|
||||
// Ensure that the database can be copied to a file path.
|
||||
func TestTx_CopyFile(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
|
||||
path := tempfile()
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
@ -607,8 +586,7 @@ func (f *failWriter) Write(p []byte) (n int, err error) {
|
|||
|
||||
// Ensure that Copy handles write errors right.
|
||||
func TestTx_CopyFile_Error_Meta(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -634,8 +612,7 @@ func TestTx_CopyFile_Error_Meta(t *testing.T) {
|
|||
|
||||
// Ensure that Copy handles write errors right.
|
||||
func TestTx_CopyFile_Error_Normal(t *testing.T) {
|
||||
db := MustOpenDB()
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDB(t)
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucket([]byte("widgets"))
|
||||
if err != nil {
|
||||
|
@ -719,8 +696,7 @@ func TestTx_releaseRange(t *testing.T) {
|
|||
// Set initial mmap size well beyond the limit we will hit in this
|
||||
// test, since we are testing with long running read transactions
|
||||
// and will deadlock if db.grow is triggered.
|
||||
db := MustOpenWithOption(&bolt.Options{InitialMmapSize: os.Getpagesize() * 100})
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, &bolt.Options{InitialMmapSize: os.Getpagesize() * 100})
|
||||
|
||||
bucket := "bucket"
|
||||
|
||||
|
|
16
unix_test.go
16
unix_test.go
|
@ -5,6 +5,7 @@ package bbolt_test
|
|||
|
||||
import (
|
||||
"fmt"
|
||||
"go.etcd.io/bbolt/internal/btesting"
|
||||
"testing"
|
||||
|
||||
bolt "go.etcd.io/bbolt"
|
||||
|
@ -15,8 +16,7 @@ func TestMlock_DbOpen(t *testing.T) {
|
|||
// 32KB
|
||||
skipOnMemlockLimitBelow(t, 32*1024)
|
||||
|
||||
db := MustOpenWithOption(&bolt.Options{Mlock: true})
|
||||
defer db.MustClose()
|
||||
btesting.MustCreateDBWithOption(t, &bolt.Options{Mlock: true})
|
||||
}
|
||||
|
||||
// Test change between "empty" (16KB) and "non-empty" db
|
||||
|
@ -24,8 +24,7 @@ func TestMlock_DbCanGrow_Small(t *testing.T) {
|
|||
// 32KB
|
||||
skipOnMemlockLimitBelow(t, 32*1024)
|
||||
|
||||
db := MustOpenWithOption(&bolt.Options{Mlock: true})
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, &bolt.Options{Mlock: true})
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucketIfNotExists([]byte("bucket"))
|
||||
|
@ -58,25 +57,24 @@ func TestMlock_DbCanGrow_Big(t *testing.T) {
|
|||
chunksBefore := 64
|
||||
chunksAfter := 64
|
||||
|
||||
db := MustOpenWithOption(&bolt.Options{Mlock: true})
|
||||
defer db.MustClose()
|
||||
db := btesting.MustCreateDBWithOption(t, &bolt.Options{Mlock: true})
|
||||
|
||||
for chunk := 0; chunk < chunksBefore; chunk++ {
|
||||
insertChunk(t, db, chunk)
|
||||
}
|
||||
dbSize := fileSize(db.f)
|
||||
dbSize := fileSize(db.Path())
|
||||
|
||||
for chunk := 0; chunk < chunksAfter; chunk++ {
|
||||
insertChunk(t, db, chunksBefore+chunk)
|
||||
}
|
||||
newDbSize := fileSize(db.f)
|
||||
newDbSize := fileSize(db.Path())
|
||||
|
||||
if newDbSize <= dbSize {
|
||||
t.Errorf("db didn't grow: %v <= %v", newDbSize, dbSize)
|
||||
}
|
||||
}
|
||||
|
||||
func insertChunk(t *testing.T, db *DB, chunkId int) {
|
||||
func insertChunk(t *testing.T, db *btesting.DB, chunkId int) {
|
||||
chunkSize := 1024
|
||||
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
|
|
Loading…
Reference in New Issue