mirror of https://github.com/etcd-io/bbolt.git
Merge pull request #37 from jpbetz/tx-tests
Improve test coverage for releaseRange rollbacks.pull/46/head
commit
4d3ab93ca9
105
tx_test.go
105
tx_test.go
|
@ -602,6 +602,111 @@ func TestTx_CopyFile_Error_Normal(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
// TestTx_releaseRange ensures db.freePages handles page releases
|
||||
// correctly when there are transaction that are no longer reachable
|
||||
// via any read/write transactions and are "between" ongoing read
|
||||
// transactions, which requires they must be freed by
|
||||
// freelist.releaseRange.
|
||||
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()
|
||||
|
||||
bucket := "bucket"
|
||||
|
||||
put := func(key, value string) {
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucketIfNotExists([]byte(bucket))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
return b.Put([]byte(key), []byte(value))
|
||||
}); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
del := func(key string) {
|
||||
if err := db.Update(func(tx *bolt.Tx) error {
|
||||
b, err := tx.CreateBucketIfNotExists([]byte(bucket))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
return b.Delete([]byte(key))
|
||||
}); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
getWithTxn := func(txn *bolt.Tx, key string) []byte {
|
||||
return txn.Bucket([]byte(bucket)).Get([]byte(key))
|
||||
}
|
||||
|
||||
openReadTxn := func() *bolt.Tx {
|
||||
readTx, err := db.Begin(false)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
return readTx
|
||||
}
|
||||
|
||||
checkWithReadTxn := func(txn *bolt.Tx, key string, wantValue []byte) {
|
||||
value := getWithTxn(txn, key)
|
||||
if !bytes.Equal(value, wantValue) {
|
||||
t.Errorf("Wanted value to be %s for key %s, but got %s", wantValue, key, string(value))
|
||||
}
|
||||
}
|
||||
|
||||
rollback := func(txn *bolt.Tx) {
|
||||
if err := txn.Rollback(); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
put("k1", "v1")
|
||||
rtx1 := openReadTxn()
|
||||
put("k2", "v2")
|
||||
hold1 := openReadTxn()
|
||||
put("k3", "v3")
|
||||
hold2 := openReadTxn()
|
||||
del("k3")
|
||||
rtx2 := openReadTxn()
|
||||
del("k1")
|
||||
hold3 := openReadTxn()
|
||||
del("k2")
|
||||
hold4 := openReadTxn()
|
||||
put("k4", "v4")
|
||||
hold5 := openReadTxn()
|
||||
|
||||
// Close the read transactions we established to hold a portion of the pages in pending state.
|
||||
rollback(hold1)
|
||||
rollback(hold2)
|
||||
rollback(hold3)
|
||||
rollback(hold4)
|
||||
rollback(hold5)
|
||||
|
||||
// Execute a write transaction to trigger a releaseRange operation in the db
|
||||
// that will free multiple ranges between the remaining open read transactions, now that the
|
||||
// holds have been rolled back.
|
||||
put("k4", "v4")
|
||||
|
||||
// Check that all long running reads still read correct values.
|
||||
checkWithReadTxn(rtx1, "k1", []byte("v1"))
|
||||
checkWithReadTxn(rtx2, "k2", []byte("v2"))
|
||||
rollback(rtx1)
|
||||
rollback(rtx2)
|
||||
|
||||
// Check that the final state is correct.
|
||||
rtx7 := openReadTxn()
|
||||
checkWithReadTxn(rtx7, "k1", nil)
|
||||
checkWithReadTxn(rtx7, "k2", nil)
|
||||
checkWithReadTxn(rtx7, "k3", nil)
|
||||
checkWithReadTxn(rtx7, "k4", []byte("v4"))
|
||||
rollback(rtx7)
|
||||
}
|
||||
|
||||
func ExampleTx_Rollback() {
|
||||
// Open the database.
|
||||
db, err := bolt.Open(tempfile(), 0666, nil)
|
||||
|
|
Loading…
Reference in New Issue