erigon-pulse/ethdb/kv_lmdb.go
Alex Sharov df05a581dd
Add tx pool stage to transaction (#981)
* add tx_pool stage to tx

* clean

* don't call commit in read tx

* don't call commit in read tx

* clean

* more logs

* fix tx pool race

* more logs

* clean
2020-08-28 07:27:48 +01:00

1091 lines
23 KiB
Go

package ethdb
import (
"bytes"
"context"
"fmt"
"io/ioutil"
"os"
"path"
"runtime"
"sync"
"time"
"github.com/c2h5oh/datasize"
"github.com/ledgerwatch/lmdb-go/lmdb"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/log"
)
const (
NonExistingDBI = 999_999_999
)
var (
LMDBMapSize = 2 * datasize.TB
)
type lmdbOpts struct {
path string
inMem bool
readOnly bool
}
func (opts lmdbOpts) Path(path string) lmdbOpts {
opts.path = path
return opts
}
func (opts lmdbOpts) InMem() lmdbOpts {
opts.inMem = true
return opts
}
func (opts lmdbOpts) ReadOnly() lmdbOpts {
opts.readOnly = true
return opts
}
func (opts lmdbOpts) Open() (KV, error) {
env, err := lmdb.NewEnv()
if err != nil {
return nil, err
}
err = env.SetMaxDBs(100)
if err != nil {
return nil, err
}
var logger log.Logger
if opts.inMem {
err = env.SetMapSize(64 << 20) // 64MB
logger = log.New("lmdb", "inMem")
if err != nil {
return nil, err
}
opts.path, _ = ioutil.TempDir(os.TempDir(), "lmdb")
} else {
err = env.SetMapSize(int64(LMDBMapSize.Bytes()))
logger = log.New("lmdb", path.Base(opts.path))
if err != nil {
return nil, err
}
}
if err = os.MkdirAll(opts.path, 0744); err != nil {
return nil, fmt.Errorf("could not create dir: %s, %w", opts.path, err)
}
var flags uint = lmdb.NoReadahead
if opts.readOnly {
flags |= lmdb.Readonly
}
if opts.inMem {
flags |= lmdb.NoMetaSync
}
flags |= lmdb.NoSync
err = env.Open(opts.path, flags, 0664)
if err != nil {
return nil, fmt.Errorf("%w, path: %s", err, opts.path)
}
db := &LmdbKV{
opts: opts,
env: env,
log: logger,
wg: &sync.WaitGroup{},
buckets: map[string]lmdb.DBI{},
}
// Open or create buckets
if opts.readOnly {
if err := db.View(context.Background(), func(tx Tx) error {
for _, name := range dbutils.Buckets {
if err := tx.(BucketMigrator).CreateBucket(name); err != nil {
return err
}
}
return nil
}); err != nil {
return nil, err
}
} else {
if err := db.Update(context.Background(), func(tx Tx) error {
for _, name := range dbutils.Buckets {
if err := tx.(BucketMigrator).CreateBucket(name); err != nil {
return err
}
}
return nil
}); err != nil {
return nil, err
}
}
// Open deprecated buckets if they exist, don't create
if err := env.View(func(tx *lmdb.Txn) error {
for _, name := range dbutils.DeprecatedBuckets {
dbi, createErr := tx.OpenDBI(name, 0)
if createErr != nil {
if lmdb.IsNotFound(createErr) {
db.buckets[name] = NonExistingDBI // some non-existing DBI
continue // if deprecated bucket couldn't be open - then it's deleted and it's fine
} else {
return createErr
}
}
db.buckets[name] = dbi
}
return nil
}); err != nil {
return nil, err
}
if !opts.inMem {
if staleReaders, err := db.env.ReaderCheck(); err != nil {
db.log.Error("failed ReaderCheck", "err", err)
} else if staleReaders > 0 {
db.log.Debug("cleared reader slots from dead processes", "amount", staleReaders)
}
}
return db, nil
}
func (opts lmdbOpts) MustOpen() KV {
db, err := opts.Open()
if err != nil {
panic(fmt.Errorf("fail to open lmdb: %w", err))
}
return db
}
type LmdbKV struct {
opts lmdbOpts
env *lmdb.Env
log log.Logger
buckets map[string]lmdb.DBI
stopStaleReadsCheck context.CancelFunc
wg *sync.WaitGroup
}
func NewLMDB() lmdbOpts {
return lmdbOpts{}
}
// Close closes db
// All transactions must be closed before closing the database.
func (db *LmdbKV) Close() {
if db.env != nil {
db.wg.Wait()
}
if db.env != nil {
env := db.env
db.env = nil
time.Sleep(10 * time.Millisecond) // TODO: remove after consensus/ethash/consensus.go:VerifyHeaders will spawn controllable goroutines
if err := env.Close(); err != nil {
db.log.Warn("failed to close DB", "err", err)
} else {
db.log.Info("database closed (LMDB)")
}
}
if db.opts.inMem {
if err := os.RemoveAll(db.opts.path); err != nil {
db.log.Warn("failed to remove in-mem db file", "err", err)
}
}
}
func (db *LmdbKV) DiskSize(_ context.Context) (uint64, error) {
stats, err := db.env.Stat()
if err != nil {
return 0, fmt.Errorf("could not read database size: %w", err)
}
return uint64(stats.PSize) * (stats.LeafPages + stats.BranchPages + stats.OverflowPages), nil
}
func (db *LmdbKV) Begin(ctx context.Context, parent Tx, writable bool) (Tx, error) {
if db.env == nil {
return nil, fmt.Errorf("db closed")
}
isSubTx := parent != nil
if !isSubTx {
runtime.LockOSThread()
db.wg.Add(1)
}
flags := uint(0)
if !writable {
flags |= lmdb.Readonly
}
var parentTx *lmdb.Txn
if parent != nil {
parentTx = parent.(*lmdbTx).tx
}
tx, err := db.env.BeginTxn(parentTx, flags)
if err != nil {
if !isSubTx {
runtime.UnlockOSThread() // unlock only in case of error. normal flow is "defer .Rollback()"
}
return nil, err
}
tx.RawRead = true
return &lmdbTx{
db: db,
ctx: ctx,
tx: tx,
isSubTx: isSubTx,
}, nil
}
type lmdbTx struct {
isSubTx bool
tx *lmdb.Txn
ctx context.Context
db *LmdbKV
cursors []*lmdb.Cursor
}
type LmdbCursor struct {
ctx context.Context
tx *lmdbTx
bucketName string
dbi lmdb.DBI
bucketCfg *dbutils.BucketConfigItem
prefix []byte
cursor *lmdb.Cursor
}
func (db *LmdbKV) Env() *lmdb.Env {
return db.env
}
func (db *LmdbKV) AllDBI() map[string]lmdb.DBI {
return db.buckets
}
// All buckets stored as keys of un-named bucket
func (tx *lmdbTx) ExistingBuckets() ([]string, error) {
var res []string
rawTx := tx.tx
root, err := rawTx.OpenRoot(0)
if err != nil {
return nil, err
}
c, err := rawTx.OpenCursor(root)
if err != nil {
return nil, err
}
for k, _, _ := c.Get(nil, nil, lmdb.First); k != nil; k, _, _ = c.Get(nil, nil, lmdb.Next) {
res = append(res, string(k))
}
return res, nil
}
func (db *LmdbKV) View(ctx context.Context, f func(tx Tx) error) (err error) {
if db.env == nil {
return fmt.Errorf("db closed")
}
db.wg.Add(1)
defer db.wg.Done()
// can't use db.evn.View method - because it calls commit for read transactions - it conflicts with write transactions.
tx, err := db.Begin(ctx, nil, false)
if err != nil {
return err
}
defer tx.Rollback()
return f(tx)
}
func (db *LmdbKV) Update(ctx context.Context, f func(tx Tx) error) (err error) {
if db.env == nil {
return fmt.Errorf("db closed")
}
db.wg.Add(1)
defer db.wg.Done()
tx, err := db.Begin(ctx, nil, true)
if err != nil {
return err
}
defer tx.Rollback()
err = f(tx)
if err != nil {
return err
}
err = tx.Commit(ctx)
if err != nil {
return err
}
return nil
}
func (tx *lmdbTx) CreateBucket(name string) error {
var flags uint = 0
if !tx.db.opts.readOnly {
flags |= lmdb.Create
}
if dbutils.BucketsCfg[name].IsDupSort {
flags |= lmdb.DupSort
}
dbi, err := tx.tx.OpenDBI(name, flags)
if err != nil {
return err
}
tx.db.buckets[name] = dbi
return nil
}
func (tx *lmdbTx) dropEvenIfBucketIsNotDeprecated(name string) error {
dbi := tx.db.buckets[name]
// if bucket was not open on db start, then it's may be deprecated
// try to open it now without `Create` flag, and if fail then nothing to drop
if dbi == NonExistingDBI {
var err error
dbi, err = tx.tx.OpenDBI(name, 0)
if err != nil {
if lmdb.IsNotFound(err) {
return nil // DBI doesn't exists means no drop needed
}
return err
}
}
if err := tx.tx.Drop(dbi, true); err != nil {
return err
}
tx.db.buckets[name] = NonExistingDBI
return nil
}
func (tx *lmdbTx) ClearBucket(bucket string) error {
if err := tx.dropEvenIfBucketIsNotDeprecated(bucket); err != nil {
return nil
}
return tx.CreateBucket(bucket)
}
func (tx *lmdbTx) DropBucket(name string) error {
for i := range dbutils.Buckets {
if dbutils.Buckets[i] == name {
return fmt.Errorf("%w, bucket: %s", ErrAttemptToDeleteNonDeprecatedBucket, name)
}
}
return tx.dropEvenIfBucketIsNotDeprecated(name)
}
func (tx *lmdbTx) ExistsBucket(name string) bool {
return tx.db.buckets[name] != NonExistingDBI
}
func (tx *lmdbTx) Commit(ctx context.Context) error {
if tx.db.env == nil {
return fmt.Errorf("db closed")
}
if tx.tx == nil {
return nil
}
defer func() {
tx.tx = nil
if !tx.isSubTx {
tx.db.wg.Done()
runtime.UnlockOSThread()
}
}()
tx.closeCursors()
commitTimer := time.Now()
if err := tx.tx.Commit(); err != nil {
return err
}
commitTook := time.Since(commitTimer)
if commitTook > 20*time.Second {
log.Info("Batch", "commit", commitTook)
}
if !tx.isSubTx { // call fsync only after main transaction commit
fsyncTimer := time.Now()
if err := tx.db.env.Sync(true); err != nil {
log.Warn("fsync after commit failed: \n", err)
}
fsyncTook := time.Since(fsyncTimer)
if fsyncTook > 20*time.Second {
log.Info("Batch", "fsync", fsyncTook)
}
}
return nil
}
func (tx *lmdbTx) Rollback() {
if tx.db.env == nil {
return
}
if tx.tx == nil {
return
}
defer func() {
tx.tx = nil
if !tx.isSubTx {
tx.db.wg.Done()
runtime.UnlockOSThread()
}
}()
tx.closeCursors()
tx.tx.Abort()
}
func (tx *lmdbTx) get(dbi lmdb.DBI, key []byte) ([]byte, error) {
return tx.tx.Get(dbi, key)
}
func (tx *lmdbTx) closeCursors() {
for _, c := range tx.cursors {
if c != nil {
c.Close()
}
}
tx.cursors = []*lmdb.Cursor{}
}
func (c *LmdbCursor) Prefix(v []byte) Cursor {
c.prefix = v
return c
}
func (c *LmdbCursor) Prefetch(v uint) Cursor {
//c.cursorOpts.PrefetchSize = int(v)
return c
}
func (tx *lmdbTx) Get(bucket string, key []byte) ([]byte, error) {
dbi := tx.db.buckets[bucket]
cfg := dbutils.BucketsCfg[bucket]
if cfg.IsDupSort {
return tx.getDupSort(bucket, dbi, cfg, key)
}
val, err := tx.get(dbi, key)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
return val, nil
}
func (tx *lmdbTx) getDupSort(bucket string, dbi lmdb.DBI, cfg *dbutils.BucketConfigItem, key []byte) ([]byte, error) {
from, to := cfg.DupFromLen, cfg.DupToLen
if len(key) == from {
c := tx.Cursor(bucket).(*LmdbCursor)
if err := c.initCursor(); err != nil {
return nil, err
}
_, v, err := c.getBothRange(key[:to], key[to:])
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
if !bytes.Equal(key[to:], v[:from-to]) {
return nil, nil
}
return v[from-to:], nil
}
val, err := tx.get(dbi, key)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
return val, nil
}
func (tx *lmdbTx) BucketSize(name string) (uint64, error) {
st, err := tx.tx.Stat(tx.db.buckets[name])
if err != nil {
return 0, err
}
return (st.LeafPages + st.BranchPages + st.OverflowPages) * uint64(os.Getpagesize()), nil
}
func (tx *lmdbTx) Cursor(bucket string) Cursor {
return &LmdbCursor{bucketName: bucket, ctx: tx.ctx, tx: tx, bucketCfg: dbutils.BucketsCfg[bucket], dbi: tx.db.buckets[bucket]}
}
func (tx *lmdbTx) NoValuesCursor(bucket string) NoValuesCursor {
return &lmdbNoValuesCursor{LmdbCursor: tx.Cursor(bucket).(*LmdbCursor)}
}
func (c *LmdbCursor) initCursor() error {
if c.cursor != nil {
return nil
}
tx := c.tx
var err error
c.cursor, err = tx.tx.OpenCursor(c.tx.db.buckets[c.bucketName])
if err != nil {
panic("su-tx")
return err
}
// add to auto-cleanup on end of transactions
if tx.cursors == nil {
tx.cursors = make([]*lmdb.Cursor, 0, 1)
}
tx.cursors = append(tx.cursors, c.cursor)
return nil
}
func (c *LmdbCursor) First() ([]byte, []byte, error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return []byte{}, nil, err
}
}
return c.Seek(c.prefix)
}
func (c *LmdbCursor) Last() ([]byte, []byte, error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return []byte{}, nil, err
}
}
if c.prefix != nil {
return []byte{}, nil, fmt.Errorf(".Last doesn't support c.prefix yet")
}
k, v, err := c.last()
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
err = fmt.Errorf("failed LmdbKV cursor.Last(): %w, bucket: %s", err, c.bucketName)
return []byte{}, nil, err
}
if c.bucketCfg.IsDupSort {
if k == nil {
return k, v, nil
}
k, v, err = c.lastDup(k)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
err = fmt.Errorf("failed LmdbKV cursor.Last(): %w, bucket: %s", err, c.bucketName)
return []byte{}, nil, err
}
}
return k, v, nil
}
func (c *LmdbCursor) Seek(seek []byte) (k, v []byte, err error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return []byte{}, nil, err
}
}
if c.bucketCfg.IsDupSort {
return c.seekDupSort(seek)
}
if len(seek) == 0 {
k, v, err = c.cursor.Get(nil, nil, lmdb.First)
} else {
k, v, err = c.setRange(seek)
}
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
err = fmt.Errorf("failed LmdbKV cursor.Seek(): %w, bucket: %s, key: %x", err, c.bucketName, seek)
return []byte{}, nil, err
}
if c.prefix != nil && !bytes.HasPrefix(k, c.prefix) {
k, v = nil, nil
}
return k, v, nil
}
func (c *LmdbCursor) seekDupSort(seek []byte) (k, v []byte, err error) {
b := c.bucketCfg
from, to := b.DupFromLen, b.DupToLen
if len(seek) == 0 {
k, v, err = c.cursor.Get(nil, nil, lmdb.First)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, err
}
if c.prefix != nil && !bytes.HasPrefix(k, c.prefix) {
k, v = nil, nil
}
return k, v, nil
}
var seek1, seek2 []byte
if len(seek) > to {
seek1, seek2 = seek[:to], seek[to:]
} else {
seek1 = seek
}
k, v, err = c.setRange(seek1)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, err
}
if seek2 != nil && bytes.Equal(seek1, k) {
k, v, err = c.getBothRange(seek1, seek2)
if err != nil && lmdb.IsNotFound(err) {
k, v, err = c.next()
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, err
}
} else if err != nil {
return []byte{}, nil, err
}
}
if len(k) == to {
k2 := make([]byte, 0, len(k)+from-to)
k2 = append(append(k2, k...), v[:from-to]...)
v = v[from-to:]
k = k2
}
if c.prefix != nil && !bytes.HasPrefix(k, c.prefix) {
k, v = nil, nil
}
return k, v, nil
}
func (c *LmdbCursor) Next() (k, v []byte, err error) {
select {
case <-c.ctx.Done():
return []byte{}, nil, c.ctx.Err()
default:
}
if c.bucketCfg.IsDupSort {
return c.nextDupSort()
}
if c.cursor == nil {
if err = c.initCursor(); err != nil {
log.Error("init cursor", "err", err)
}
}
k, v, err = c.cursor.Get(nil, nil, lmdb.Next)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, fmt.Errorf("failed LmdbKV cursor.Next(): %w", err)
}
if c.prefix != nil && !bytes.HasPrefix(k, c.prefix) {
k, v = nil, nil
}
return k, v, nil
}
func (c *LmdbCursor) nextDupSort() (k, v []byte, err error) {
b := c.bucketCfg
from, to := b.DupFromLen, b.DupToLen
k, v, err = c.cursor.Get(nil, nil, lmdb.NextDup)
if err != nil && lmdb.IsNotFound(err) {
k, v, err = c.cursor.Get(nil, nil, lmdb.Next)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, fmt.Errorf("failed LmdbKV cursor.Next(): %w", err)
}
} else if err != nil {
return nil, nil, err
}
if len(k) == to {
k = append(k, v[:from-to]...)
v = v[from-to:]
}
if c.prefix != nil && !bytes.HasPrefix(k, c.prefix) {
k, v = nil, nil
}
return k, v, nil
}
func (c *LmdbCursor) Delete(key []byte) error {
select {
case <-c.ctx.Done():
return c.ctx.Err()
default:
}
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return err
}
}
if c.bucketCfg.IsDupSort {
return c.deleteDupSort(key)
}
_, _, err := c.cursor.Get(key, nil, lmdb.Set)
if err != nil {
if lmdb.IsNotFound(err) {
return nil
}
return err
}
return c.cursor.Del(0)
}
func (c *LmdbCursor) deleteDupSort(key []byte) error {
b := c.bucketCfg
from, to := b.DupFromLen, b.DupToLen
if len(key) != from && len(key) >= to {
return fmt.Errorf("dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x", c.bucketName, from, to, key)
}
if len(key) == from {
_, v, err := c.cursor.Get(key[:to], key[to:], lmdb.GetBothRange)
if err != nil { // if key not found, or found another one - then nothing to delete
if lmdb.IsNotFound(err) {
return nil
}
return err
}
if !bytes.Equal(v[:from-to], key[to:]) {
return nil
}
return c.cursor.Del(0)
}
_, _, err := c.cursor.Get(key, nil, lmdb.Set)
if err != nil {
if lmdb.IsNotFound(err) {
return nil
}
return err
}
return c.cursor.Del(0)
}
func (c *LmdbCursor) Put(key []byte, value []byte) error {
if len(key) == 0 {
return fmt.Errorf("lmdb doesn't support empty keys. bucket: %s", c.bucketName)
}
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return err
}
}
if c.bucketCfg.IsDupSort {
return c.putDupSort(key, value)
}
return c.put(key, value)
}
func (c *LmdbCursor) PutCurrent(key []byte, value []byte) error {
if len(key) == 0 {
return fmt.Errorf("lmdb doesn't support empty keys. bucket: %s", c.bucketName)
}
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return err
}
}
return c.putCurrent(key, value)
}
func (c *LmdbCursor) putDupSort(key []byte, value []byte) error {
b := c.bucketCfg
from, to := b.DupFromLen, b.DupToLen
if len(key) != from && len(key) >= to {
return fmt.Errorf("dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x", c.bucketName, from, to, key)
}
if len(key) != from {
_, _, err := c.set(key)
if err != nil {
if lmdb.IsNotFound(err) {
return c.put(key, value)
}
return err
}
return c.putCurrent(key, value)
}
newValue := make([]byte, 0, from-to+len(value))
newValue = append(append(newValue, key[to:]...), value...)
key = key[:to]
_, v, err := c.getBothRange(key, newValue[:from-to])
if err != nil { // if key not found, or found another one - then just insert
if lmdb.IsNotFound(err) {
return c.put(key, newValue)
}
return err
}
if bytes.Equal(v[:from-to], newValue[:from-to]) {
if len(v) == len(newValue) { // in DupSort case lmdb.Current works only with values of same length
return c.putCurrent(key, newValue)
}
err = c.delCurrent()
if err != nil {
return err
}
return c.put(key, newValue)
}
return c.put(key, newValue)
}
func (c *LmdbCursor) SeekExact(key []byte) ([]byte, error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return nil, err
}
}
if c.bucketCfg.IsDupSort {
return c.getDupSort(key)
}
_, v, err := c.set(key)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
return v, nil
}
func (c *LmdbCursor) getDupSort(key []byte) ([]byte, error) {
from, to := c.bucketCfg.DupFromLen, c.bucketCfg.DupToLen
if len(key) == from {
_, v, err := c.getBothRange(key[:to], key[to:])
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
if !bytes.Equal(key[to:], v[:from-to]) {
return nil, nil
}
return v[from-to:], nil
}
_, val, err := c.set(key)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil
}
return nil, err
}
return val, nil
}
func (c *LmdbCursor) SeekBothRange(key, value []byte) ([]byte, []byte, error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return []byte{}, nil, err
}
}
k, v, err := c.getBothRange(key, value)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, err
}
return k, v, nil
}
func (c *LmdbCursor) SeekBothExact(key, value []byte) ([]byte, []byte, error) {
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return []byte{}, nil, err
}
}
k, v, err := c.getBoth(key, value)
if err != nil {
if lmdb.IsNotFound(err) {
return nil, nil, nil
}
return []byte{}, nil, err
}
return k, v, nil
}
func (c *LmdbCursor) set(key []byte) ([]byte, []byte, error) {
return c.cursor.Get(key, nil, lmdb.Set)
}
func (c *LmdbCursor) setRange(key []byte) ([]byte, []byte, error) {
return c.cursor.Get(key, nil, lmdb.SetRange)
}
func (c *LmdbCursor) next() ([]byte, []byte, error) {
return c.cursor.Get(nil, nil, lmdb.Next)
}
func (c *LmdbCursor) getBothRange(key []byte, value []byte) ([]byte, []byte, error) {
k, v, err := c.cursor.Get(key, value, lmdb.GetBothRange)
if err != nil {
return []byte{}, nil, err
}
return k, v, nil
}
func (c *LmdbCursor) getBoth(key []byte, value []byte) ([]byte, []byte, error) {
k, v, err := c.cursor.Get(key, value, lmdb.GetBoth)
if err != nil {
return []byte{}, nil, err
}
return k, v, nil
}
func (c *LmdbCursor) last() ([]byte, []byte, error) {
return c.cursor.Get(nil, nil, lmdb.Last)
}
func (c *LmdbCursor) lastDup(key []byte) ([]byte, []byte, error) {
return c.cursor.Get(key, nil, lmdb.LastDup)
}
func (c *LmdbCursor) delCurrent() error {
return c.cursor.Del(0)
}
func (c *LmdbCursor) put(key []byte, value []byte) error {
return c.cursor.Put(key, value, 0)
}
func (c *LmdbCursor) putCurrent(key []byte, value []byte) error {
return c.cursor.Put(key, value, lmdb.Current)
}
func (c *LmdbCursor) append(key []byte, value []byte) error {
return c.cursor.Put(key, value, lmdb.Append)
}
func (c *LmdbCursor) appendDup(key []byte, value []byte) error {
return c.cursor.Put(key, value, lmdb.AppendDup)
}
// Append - speedy feature of lmdb which is not part of KV interface.
// Cast your cursor to *LmdbCursor to use this method.
// Danger: if provided data will not sorted (or bucket have old records which mess with new in sorting manner) - db will corrupt.
func (c *LmdbCursor) Append(key []byte, value []byte) error {
if len(key) == 0 {
return fmt.Errorf("lmdb doesn't support empty keys. bucket: %s", c.bucketName)
}
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return err
}
}
b := c.bucketCfg
from, to := b.DupFromLen, b.DupToLen
if b.IsDupSort {
if len(key) != from && len(key) >= to {
return fmt.Errorf("dupsort bucket: %s, can have keys of len==%d and len<%d. key: %x", c.bucketName, from, to, key)
}
if len(key) == from {
newValue := make([]byte, 0, from-to+len(value))
newValue = append(append(newValue, key[to:]...), value...)
key = key[:to]
return c.appendDup(key, newValue)
}
return c.append(key, value)
}
return c.append(key, value)
}
func (c *LmdbCursor) AppendDup(key []byte, value []byte) error {
if len(key) == 0 {
return fmt.Errorf("lmdb doesn't support empty keys. bucket: %s", c.bucketName)
}
if c.cursor == nil {
if err := c.initCursor(); err != nil {
return err
}
}
return c.appendDup(key, value)
}
func (c *LmdbCursor) Close() error {
if c.cursor != nil {
c.cursor.Close()
c.cursor = nil
}
return nil
}
type lmdbNoValuesCursor struct {
*LmdbCursor
}
func (c *lmdbNoValuesCursor) First() (k []byte, v uint32, err error) {
return c.Seek(c.prefix)
}
func (c *lmdbNoValuesCursor) Seek(seek []byte) (k []byte, vSize uint32, err error) {
k, v, err := c.LmdbCursor.Seek(seek)
if err != nil {
return []byte{}, 0, err
}
return k, uint32(len(v)), err
}
func (c *lmdbNoValuesCursor) Next() (k []byte, vSize uint32, err error) {
k, v, err := c.LmdbCursor.Next()
if err != nil {
return []byte{}, 0, err
}
return k, uint32(len(v)), err
}