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/>.
|
|
|
|
|
|
|
|
// +build !js
|
|
|
|
|
|
|
|
package ethdb
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strconv"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
2019-11-13 16:43:26 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
2020-05-30 08:12:21 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
2019-11-13 16:43:26 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2019-11-05 12:28:36 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func newTestBoltDB() (*BoltDatabase, func()) {
|
|
|
|
dirname, err := ioutil.TempDir(os.TempDir(), "ethdb_test_")
|
|
|
|
if err != nil {
|
|
|
|
panic("failed to create test file: " + err.Error())
|
|
|
|
}
|
|
|
|
db, err := NewBoltDatabase(path.Join(dirname, "db"))
|
|
|
|
if err != nil {
|
|
|
|
panic("failed to create test database: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return db, func() {
|
|
|
|
db.Close()
|
|
|
|
os.RemoveAll(dirname)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTestBadgerDB() (*BadgerDatabase, func()) {
|
2019-11-21 15:12:38 +00:00
|
|
|
db, err := NewEphemeralBadger()
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
panic("failed to create test database: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return db, func() {
|
|
|
|
db.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-30 08:12:21 +00:00
|
|
|
func newTestLmdb() *ObjectDatabase {
|
2020-06-12 09:31:21 +00:00
|
|
|
return NewObjectDatabase(NewLMDB().InMem().MustOpen())
|
2020-05-30 08:12:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var testBucket = dbutils.CurrentStateBucket
|
2019-11-05 12:28:36 +00:00
|
|
|
var testValues = []string{"a", "1251", "\x00123\x00"}
|
|
|
|
|
|
|
|
func TestBoltDB_PutGet(t *testing.T) {
|
|
|
|
db, remove := newTestBoltDB()
|
|
|
|
defer remove()
|
|
|
|
testPutGet(db, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMemoryDB_PutGet(t *testing.T) {
|
2020-06-05 09:25:33 +00:00
|
|
|
db := NewMemDatabase()
|
|
|
|
defer db.Close()
|
|
|
|
testPutGet(db, t)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestBadgerDB_PutGet(t *testing.T) {
|
|
|
|
db, remove := newTestBadgerDB()
|
|
|
|
defer remove()
|
|
|
|
testPutGet(db, t)
|
|
|
|
}
|
|
|
|
|
2020-05-30 08:12:21 +00:00
|
|
|
func TestLMDB_PutGet(t *testing.T) {
|
|
|
|
db := newTestLmdb()
|
|
|
|
defer db.Close()
|
|
|
|
testPutGet(db, t)
|
|
|
|
}
|
|
|
|
|
2019-11-06 10:24:48 +00:00
|
|
|
func testPutGet(db MinDatabase, t *testing.T) {
|
2019-11-05 12:28:36 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
for _, k := range testValues {
|
2020-06-05 09:25:33 +00:00
|
|
|
err := db.Put(testBucket, []byte(k), []byte{})
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("put failed: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, k := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
data, err := db.Get(testBucket, []byte(k))
|
2019-11-05 12:28:36 +00:00
|
|
|
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))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-11 11:02:37 +00:00
|
|
|
_, err := db.Get(testBucket, []byte("non-exist-key"))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("expect to return a not found error")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Put(testBucket, []byte(v), []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("put failed: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
data, err := db.Get(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("get failed: %v", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(data, []byte(v)) {
|
|
|
|
t.Fatalf("get returned wrong result, got %q expected %q", string(data), v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Put(testBucket, []byte(v), []byte("?"))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("put override failed: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
data, err := db.Get(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("get failed: %v", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(data, []byte("?")) {
|
|
|
|
t.Fatalf("get returned wrong result, got %q expected ?", string(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
orig, err := db.Get(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("get failed: %v", err)
|
|
|
|
}
|
|
|
|
orig[0] = byte(0xff)
|
2020-03-11 11:02:37 +00:00
|
|
|
data, err := db.Get(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("get failed: %v", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(data, []byte("?")) {
|
|
|
|
t.Fatalf("get returned wrong result, got %q expected ?", string(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Delete(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("delete %q failed: %v", v, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range testValues {
|
2020-03-11 11:02:37 +00:00
|
|
|
_, err := db.Get(testBucket, []byte(v))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("got deleted value %q", v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBoltDB_ParallelPutGet(t *testing.T) {
|
|
|
|
db, remove := newTestBoltDB()
|
|
|
|
defer remove()
|
|
|
|
testParallelPutGet(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMemoryDB_ParallelPutGet(t *testing.T) {
|
2020-06-06 10:29:11 +00:00
|
|
|
db := NewMemDatabase()
|
|
|
|
defer db.Close()
|
|
|
|
testParallelPutGet(db)
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 08:12:21 +00:00
|
|
|
func TestLMDB_ParallelPutGet(t *testing.T) {
|
|
|
|
db := newTestLmdb()
|
|
|
|
defer db.Close()
|
|
|
|
testParallelPutGet(db)
|
|
|
|
}
|
|
|
|
|
2019-11-05 12:28:36 +00:00
|
|
|
func TestBadgerDB_ParallelPutGet(t *testing.T) {
|
|
|
|
db, remove := newTestBadgerDB()
|
|
|
|
defer remove()
|
|
|
|
testParallelPutGet(db)
|
|
|
|
}
|
2019-11-06 10:24:48 +00:00
|
|
|
func testParallelPutGet(db MinDatabase) {
|
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()
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Put(testBucket, []byte(key), []byte("v"+key))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
panic("put failed: " + err.Error())
|
|
|
|
}
|
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2020-03-11 11:02:37 +00:00
|
|
|
data, err := db.Get(testBucket, []byte(key))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
panic("get failed: " + err.Error())
|
|
|
|
}
|
|
|
|
if !bytes.Equal(data, []byte("v"+key)) {
|
2019-11-06 10:24:48 +00:00
|
|
|
panic(fmt.Sprintf("get failed, got %q expected %q", data, []byte("v"+key)))
|
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()
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Delete(testBucket, []byte(key))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err != nil {
|
|
|
|
panic("delete failed: " + err.Error())
|
|
|
|
}
|
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
|
|
|
|
pending.Add(n)
|
|
|
|
for i := 0; i < n; i++ {
|
|
|
|
go func(key string) {
|
|
|
|
defer pending.Done()
|
2020-03-11 11:02:37 +00:00
|
|
|
_, err := db.Get(testBucket, []byte(key))
|
2019-11-05 12:28:36 +00:00
|
|
|
if err == nil {
|
2019-11-06 10:30:02 +00:00
|
|
|
panic("get succeeded")
|
2019-11-05 12:28:36 +00:00
|
|
|
}
|
|
|
|
}(strconv.Itoa(i))
|
|
|
|
}
|
|
|
|
pending.Wait()
|
|
|
|
}
|
2019-11-13 16:43:26 +00:00
|
|
|
|
|
|
|
func TestMemoryDB_Walk(t *testing.T) {
|
|
|
|
testWalk(NewMemDatabase(), t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBoltDB_Walk(t *testing.T) {
|
|
|
|
db, remove := newTestBoltDB()
|
|
|
|
defer remove()
|
|
|
|
testWalk(db, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestBadgerDB_Walk(t *testing.T) {
|
|
|
|
db, remove := newTestBadgerDB()
|
|
|
|
defer remove()
|
|
|
|
testWalk(db, t)
|
|
|
|
}
|
|
|
|
|
2020-05-30 08:12:21 +00:00
|
|
|
func TestLMDB_Walk(t *testing.T) {
|
|
|
|
db := newTestLmdb()
|
|
|
|
defer db.Close()
|
|
|
|
testWalk(db, t)
|
|
|
|
}
|
|
|
|
|
2019-11-13 16:43:26 +00:00
|
|
|
var hexEntries = map[string]string{
|
|
|
|
"6b": "89c6",
|
|
|
|
"91": "c476",
|
|
|
|
"a8": "0a514e",
|
|
|
|
"bb": "7a",
|
|
|
|
"bd": "fe76",
|
|
|
|
"c0": "12",
|
|
|
|
}
|
|
|
|
|
|
|
|
var startKey = common.FromHex("a0")
|
2020-05-11 04:46:07 +00:00
|
|
|
var fixedBits int = 3
|
2019-11-13 16:43:26 +00:00
|
|
|
|
|
|
|
var keysInRange = [][]byte{common.FromHex("a8"), common.FromHex("bb"), common.FromHex("bd")}
|
|
|
|
|
2019-11-14 12:00:38 +00:00
|
|
|
func testWalk(db Database, t *testing.T) {
|
2019-11-13 16:43:26 +00:00
|
|
|
for k, v := range hexEntries {
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Put(testBucket, common.FromHex(k), common.FromHex(v))
|
2019-11-13 16:43:26 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("put failed: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var gotKeys [][]byte
|
|
|
|
|
2020-03-11 11:02:37 +00:00
|
|
|
err := db.Walk(testBucket, startKey, fixedBits, func(key, val []byte) (bool, error) {
|
2020-04-11 13:28:15 +00:00
|
|
|
gotKeys = append(gotKeys, common.CopyBytes(key))
|
2019-11-13 16:43:26 +00:00
|
|
|
return true, nil
|
|
|
|
})
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, keysInRange, gotKeys)
|
|
|
|
}
|