2019-11-05 12:28:36 +00:00
|
|
|
// Copyright 2014 The go-ethereum Authors
|
|
|
|
// This file is part of the go-ethereum library.
|
|
|
|
//
|
|
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2021-10-12 05:04:04 +00:00
|
|
|
//go:build !js
|
2019-11-05 12:28:36 +00:00
|
|
|
|
2023-10-09 08:47:54 +00:00
|
|
|
package membatch
|
2019-11-05 12:28:36 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-07-27 06:10:50 +00:00
|
|
|
"context"
|
2019-11-05 12:28:36 +00:00
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
2020-07-27 06:10:50 +00:00
|
|
|
"time"
|
2019-11-13 16:43:26 +00:00
|
|
|
|
2021-07-29 11:53:13 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/memdb"
|
2020-07-27 06:10:50 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2019-11-05 12:28:36 +00:00
|
|
|
)
|
|
|
|
|
2021-07-28 02:47:38 +00:00
|
|
|
var testBucket = kv.HashedAccounts
|
2019-11-05 12:28:36 +00:00
|
|
|
var testValues = []string{"a", "1251", "\x00123\x00"}
|
|
|
|
|
2021-05-19 03:47:28 +00:00
|
|
|
func TestPutGet(t *testing.T) {
|
2021-07-28 02:47:38 +00:00
|
|
|
_, tx := memdb.NewTestTx(t)
|
2020-05-30 08:12:21 +00:00
|
|
|
|
2020-08-02 11:53:08 +00:00
|
|
|
//for _, k := range testValues {
|
|
|
|
// err := db.Put(testBucket, []byte(k), []byte{})
|
|
|
|
// if err != nil {
|
|
|
|
// t.Fatalf("put failed: %v", err)
|
|
|
|
// }
|
|
|
|
//}
|
|
|
|
//
|
|
|
|
//for _, k := range testValues {
|
|
|
|
// data, err := db.Get(testBucket, []byte(k))
|
|
|
|
// if err != nil {
|
|
|
|
// t.Fatalf("get failed: %v", err)
|
|
|
|
// }
|
|
|
|
// if len(data) != 0 {
|
|
|
|
// t.Fatalf("get returned wrong result, got %q expected nil", string(data))
|
|
|
|
// }
|
|
|
|
//}
|
2019-11-05 12:28:36 +00:00
|
|
|
|
2021-05-19 03:47:28 +00:00
|
|
|
_, err := tx.GetOne(testBucket, []byte("non-exist-key"))
|
2021-05-17 09:54:29 +00:00
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
|
|
|
|
for _, v := range testValues {
|
2021-05-17 09:54:29 +00:00
|
|
|
err := tx.Put(testBucket, []byte(v), []byte(v))
|
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2021-05-17 09:54:29 +00:00
|
|
|
data, err := tx.GetOne(testBucket, []byte(v))
|
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
if !bytes.Equal(data, []byte(v)) {
|
|
|
|
t.Fatalf("get returned wrong result, got %q expected %q", string(data), v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2021-05-17 09:54:29 +00:00
|
|
|
err := tx.Put(testBucket, []byte(v), []byte("?"))
|
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2021-05-17 09:54:29 +00:00
|
|
|
data, err := tx.GetOne(testBucket, []byte(v))
|
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
if !bytes.Equal(data, []byte("?")) {
|
|
|
|
t.Fatalf("get returned wrong result, got %q expected ?", string(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2022-07-26 05:47:05 +00:00
|
|
|
err := tx.Delete(testBucket, []byte(v))
|
2021-05-17 09:54:29 +00:00
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2021-05-17 09:54:29 +00:00
|
|
|
_, err := tx.GetOne(testBucket, []byte(v))
|
|
|
|
require.NoError(t, err)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-19 03:47:28 +00:00
|
|
|
func TestNoPanicAfterDbClosed(t *testing.T) {
|
2021-07-28 02:47:38 +00:00
|
|
|
db := memdb.NewTestDB(t)
|
2021-05-17 09:54:29 +00:00
|
|
|
tx, err := db.BeginRo(context.Background())
|
2020-07-27 06:10:50 +00:00
|
|
|
require.NoError(t, err)
|
2021-05-19 03:47:28 +00:00
|
|
|
defer tx.Rollback()
|
2021-05-17 09:54:29 +00:00
|
|
|
writeTx, err := db.BeginRw(context.Background())
|
2020-07-27 06:10:50 +00:00
|
|
|
require.NoError(t, err)
|
2021-05-19 03:47:28 +00:00
|
|
|
defer writeTx.Rollback()
|
2020-10-20 21:23:56 +00:00
|
|
|
|
|
|
|
closeCh := make(chan struct{}, 1)
|
2020-07-27 06:10:50 +00:00
|
|
|
go func() {
|
|
|
|
require.NotPanics(t, func() {
|
2020-10-20 21:23:56 +00:00
|
|
|
<-closeCh
|
2020-07-27 06:10:50 +00:00
|
|
|
db.Close()
|
|
|
|
})
|
|
|
|
}()
|
|
|
|
time.Sleep(time.Millisecond) // wait to check that db.Close doesn't panic, but wait when read tx finished
|
2021-07-28 03:43:51 +00:00
|
|
|
err = writeTx.Put(kv.ChaindataTables[0], []byte{1}, []byte{1})
|
2020-07-27 06:10:50 +00:00
|
|
|
require.NoError(t, err)
|
2021-04-03 06:26:00 +00:00
|
|
|
err = writeTx.Commit()
|
2020-07-27 06:10:50 +00:00
|
|
|
require.NoError(t, err)
|
2021-07-28 03:43:51 +00:00
|
|
|
_, err = tx.GetOne(kv.ChaindataTables[0], []byte{1})
|
2020-07-27 06:10:50 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
tx.Rollback()
|
|
|
|
|
|
|
|
db.Close() // close db from 2nd goroutine
|
2020-10-20 21:23:56 +00:00
|
|
|
close(closeCh)
|
2020-07-27 06:10:50 +00:00
|
|
|
|
|
|
|
// after db closed, methods must not panic but return some error
|
2021-05-17 09:54:29 +00:00
|
|
|
//require.NotPanics(t, func() {
|
|
|
|
// _, err := tx.GetOne(testBucket, []byte{11})
|
|
|
|
// require.Error(t, err)
|
|
|
|
// err = writeTx.Put(testBucket, []byte{11}, []byte{11})
|
|
|
|
// require.Error(t, err)
|
|
|
|
//})
|
2020-07-27 06:10:50 +00:00
|
|
|
}
|
|
|
|
|
2021-05-19 03:47:28 +00:00
|
|
|
func TestParallelPutGet(t *testing.T) {
|
2021-07-28 02:47:38 +00:00
|
|
|
db := memdb.NewTestDB(t)
|
2019-11-05 12:28:36 +00:00
|
|
|
|
|
|
|
const n = 8
|
|
|
|
var pending sync.WaitGroup
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2021-07-28 02:47:38 +00:00
|
|
|
_ = db.Update(context.Background(), func(tx kv.RwTx) error {
|
2021-05-17 09:54:29 +00:00
|
|
|
err := tx.Put(testBucket, []byte(key), []byte("v"+key))
|
|
|
|
if err != nil {
|
|
|
|
panic("put failed: " + err.Error())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2019-11-05 12:28:36 +00:00
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2021-07-28 02:47:38 +00:00
|
|
|
_ = db.View(context.Background(), func(tx kv.Tx) error {
|
2021-05-17 09:54:29 +00:00
|
|
|
data, err := tx.GetOne(testBucket, []byte(key))
|
|
|
|
if err != nil {
|
|
|
|
panic("get failed: " + err.Error())
|
|
|
|
}
|
|
|
|
if !bytes.Equal(data, []byte("v"+key)) {
|
|
|
|
panic(fmt.Sprintf("get failed, got %q expected %q", data, []byte("v"+key)))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2019-11-05 12:28:36 +00:00
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2021-07-28 02:47:38 +00:00
|
|
|
_ = db.Update(context.Background(), func(tx kv.RwTx) error {
|
2022-07-26 05:47:05 +00:00
|
|
|
err := tx.Delete(testBucket, []byte(key))
|
2021-05-17 09:54:29 +00:00
|
|
|
if err != nil {
|
|
|
|
panic("delete failed: " + err.Error())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2019-11-05 12:28:36 +00:00
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2021-07-28 02:47:38 +00:00
|
|
|
_ = db.Update(context.Background(), func(tx kv.RwTx) error {
|
2021-05-17 09:54:29 +00:00
|
|
|
v, err := tx.GetOne(testBucket, []byte(key))
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
if v != nil {
|
|
|
|
panic("get returned something")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2019-11-05 12:28:36 +00:00
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
}
|