mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 17:31:11 +00:00
c5ffc971c5
* badger v2 investigation * buf pool - use native New method and avoid double checks * db.Open prototype * db.Tx/Bucket/Cursor prototypes * Chained config * Item concept added * save changes to test on master * make hack resumable * Design document v0 * Cursor concept * less brackets syntax of cursor builder * benchmarks * cleanup fs * test for context cancelations * test for context cancelations * test for cursor.Prefix option * add ForEachKey method * add ForEachKey method * add naming explanation * experiment of non-pointers cursor/bucket * .Bucket() and .Cursor() doesn't returns error * .Bucket() and .Cursor() doesn't returns error * .Bucket() and .Cursor() doesn't returns error * remove CursorOpts concept * more test-cases * simplify open api * Tx, Bucket, Cursor - now are interfaces * Tx, Bucket, Cursor - now are interfaces * switch to interfaces * rebase master Co-authored-by: alex.sharov <alex.sharov@lazada.com>
919 lines
24 KiB
Go
919 lines
24 KiB
Go
// Copyright 2019 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/>.
|
|
|
|
package remote
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb/codecpool"
|
|
"github.com/ledgerwatch/turbo-geth/log"
|
|
"github.com/ugorji/go/codec"
|
|
)
|
|
|
|
// Version is the current version of the remote db protocol. If the protocol changes in a non backwards compatible way,
|
|
// this constant needs to be increased
|
|
const Version uint64 = 2
|
|
|
|
// Command is the type of command in the boltdb remote protocol
|
|
type Command uint8
|
|
type ResponseCode uint8
|
|
|
|
const (
|
|
// ResponseOk
|
|
// successful response to client's request
|
|
ResponseOk ResponseCode = iota
|
|
// ResponseErr : errorMessage
|
|
// returns error to client
|
|
ResponseErr
|
|
)
|
|
|
|
const (
|
|
// CmdVersion : version
|
|
// is sent from client to server to ask about the version of protocol the server supports
|
|
// it is also to be used to be sent periodically to make sure the connection stays open
|
|
CmdVersion Command = iota
|
|
// CmdBeginTx
|
|
// request starting a new transaction (read-only). It returns transaction's handle (uint64), or 0
|
|
// if there was an error. If 0 is returned, the corresponding
|
|
CmdBeginTx
|
|
// CmdEndTx ()
|
|
// request the end of the transaction (rollback)
|
|
CmdEndTx
|
|
// CmdBucket (name): bucketHandle
|
|
// requests opening a bucket with given name. It returns bucket's handle (uint64)
|
|
CmdBucket
|
|
// CmdGet (bucketHandle, key): value
|
|
// requests a value for a key from given bucket.
|
|
CmdGet
|
|
// CmdCursor (bucketHandle): cursorHandle
|
|
// request creating a cursor for the given bucket. It returns cursor's handle (uint64)
|
|
CmdCursor
|
|
// CmdCursorSeek (cursorHandle, seekKey): (key, value)
|
|
// Moves given cursor to the seekKey, or to the next key after seekKey
|
|
CmdCursorSeek
|
|
// CmdCursorNext (cursorHandle, number of keys): [(key, value)]
|
|
// Moves given cursor over the next given number of keys and streams back the (key, value) pairs
|
|
// Pair with key == nil signifies the end of the stream
|
|
CmdCursorNext
|
|
// CmdCursorFirst (cursorHandle, number of keys): [(key, value)]
|
|
// Moves given cursor to bucket start and streams back the (key, value) pairs
|
|
// Pair with key == nil signifies the end of the stream
|
|
CmdCursorFirst
|
|
// CmdCursorSeekTo (cursorHandle, seekKey): (key, value)
|
|
// Moves given cursor to the seekKey, or to the next key after seekKey
|
|
CmdCursorSeekTo
|
|
// CmdGetAsOf (bucket, hBucket, key []byte, timestamp uint64): (value)
|
|
CmdGetAsOf
|
|
// CmdCursorFirstKey (cursorHandle, number of keys): [(key, valueIsEmpty)]
|
|
// Moves given cursor to bucket start and streams back the (key, valueIsEmpty) pairs
|
|
// Pair with key == nil signifies the end of the stream
|
|
CmdCursorFirstKey
|
|
// CmdCursorNextKey (cursorHandle, number of keys): [(key, valueIsEmpty)]
|
|
// Moves given cursor over the next given number of keys and streams back the (key, valueIsEmpty) pairs
|
|
// Pair with key == nil signifies the end of the stream
|
|
CmdCursorNextKey
|
|
)
|
|
|
|
const DefaultCursorBatchSize uint64 = 1
|
|
const CursorMaxBatchSize uint64 = 1 * 1000 * 1000
|
|
const ClientMaxConnections uint64 = 128
|
|
|
|
var logger = log.New("database", "remote")
|
|
|
|
func decodeKey(decoder *codec.Decoder, key *[]byte, valueIsEmpty *bool) error {
|
|
if err := decoder.Decode(key); err != nil {
|
|
return err
|
|
}
|
|
if err := decoder.Decode(valueIsEmpty); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func decodeKeyValue(decoder *codec.Decoder, key *[]byte, value *[]byte) (err error) {
|
|
if err := decoder.Decode(key); err != nil {
|
|
return err
|
|
}
|
|
if err := decoder.Decode(value); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func decodeErr(decoder *codec.Decoder, responseCode ResponseCode) error {
|
|
if responseCode != ResponseErr {
|
|
return fmt.Errorf("unknown response code: %d", responseCode)
|
|
}
|
|
|
|
var errorMessage string
|
|
if err := decoder.Decode(&errorMessage); err != nil {
|
|
return fmt.Errorf("can't decode errorMessage: %w", err)
|
|
}
|
|
|
|
return errors.New(errorMessage)
|
|
}
|
|
|
|
type conn struct {
|
|
in io.Reader
|
|
out io.Writer
|
|
closer io.Closer
|
|
}
|
|
|
|
type DbOpts struct {
|
|
dialAddress string
|
|
DialFunc DialFunc
|
|
DialTimeout time.Duration
|
|
PingTimeout time.Duration
|
|
RetryDialAfter time.Duration
|
|
PingEvery time.Duration
|
|
MaxConnections uint64
|
|
}
|
|
|
|
var DefaultOpts = DbOpts{
|
|
MaxConnections: ClientMaxConnections,
|
|
DialTimeout: 3 * time.Second,
|
|
PingTimeout: 500 * time.Millisecond,
|
|
RetryDialAfter: 1 * time.Second,
|
|
PingEvery: 1 * time.Second,
|
|
}
|
|
|
|
func (opts DbOpts) Addr(v string) DbOpts {
|
|
opts.dialAddress = v
|
|
return opts
|
|
}
|
|
|
|
func defaultDialFunc(ctx context.Context, dialAddress string) (in io.Reader, out io.Writer, closer io.Closer, err error) {
|
|
dialer := net.Dialer{}
|
|
conn, err := dialer.DialContext(ctx, "tcp", dialAddress)
|
|
if err != nil {
|
|
return nil, nil, nil, fmt.Errorf("could not connect to remoteDb. addr: %s. err: %w", dialAddress, err)
|
|
}
|
|
return conn, conn, conn, err
|
|
}
|
|
|
|
// DB mimicks the interface of the bolt.DB,
|
|
// but it works via a pair (Reader, Writer)
|
|
type DB struct {
|
|
opts DbOpts
|
|
connectionPool chan *conn
|
|
doDial chan struct{}
|
|
doPing <-chan time.Time
|
|
}
|
|
|
|
type DialFunc func(ctx context.Context) (in io.Reader, out io.Writer, closer io.Closer, err error)
|
|
|
|
// Pool of connections to server
|
|
func (db *DB) getConnection(ctx context.Context) (io.Reader, io.Writer, io.Closer, error) {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, nil, nil, ctx.Err()
|
|
case conn := <-db.connectionPool:
|
|
return conn.in, conn.out, conn.closer, nil
|
|
}
|
|
}
|
|
|
|
func (db *DB) returnConn(ctx context.Context, in io.Reader, out io.Writer, closer io.Closer) {
|
|
select {
|
|
case db.connectionPool <- &conn{in: in, out: out, closer: closer}:
|
|
case <-ctx.Done():
|
|
}
|
|
}
|
|
|
|
func (db *DB) ping(ctx context.Context) (err error) {
|
|
in, out, closer, err := db.getConnection(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
if closeErr := closer.Close(); closeErr != nil {
|
|
logger.Error("can't close connection", "err", closeErr)
|
|
}
|
|
return
|
|
}
|
|
|
|
db.returnConn(ctx, in, out, closer)
|
|
}()
|
|
|
|
decoder := codecpool.Decoder(in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(out)
|
|
defer codecpool.Return(encoder)
|
|
// Check version
|
|
if err := encoder.Encode(CmdVersion); err != nil {
|
|
return fmt.Errorf("could not encode CmdVersion: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode ResponseCode of CmdVersion: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
return decodeErr(decoder, responseCode)
|
|
}
|
|
|
|
var v uint64
|
|
if err := decoder.Decode(&v); err != nil {
|
|
return err
|
|
}
|
|
if v != Version {
|
|
return fmt.Errorf("server protocol version %d, expected %d", v, Version)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type notifyOnClose struct {
|
|
internal io.Closer
|
|
notifyCh chan struct{}
|
|
}
|
|
|
|
func (closer notifyOnClose) Close() error {
|
|
closer.notifyCh <- struct{}{}
|
|
if closer.internal == nil {
|
|
return nil
|
|
}
|
|
|
|
return closer.internal.Close()
|
|
}
|
|
|
|
func Open(parentCtx context.Context, opts DbOpts) (*DB, error) {
|
|
if opts.DialFunc == nil {
|
|
opts.DialFunc = func(ctx context.Context) (in io.Reader, out io.Writer, closer io.Closer, err error) {
|
|
if opts.dialAddress == "" {
|
|
return nil, nil, nil, fmt.Errorf("please set opts.dialAddress or opts.DialFunc")
|
|
}
|
|
return defaultDialFunc(ctx, opts.dialAddress)
|
|
}
|
|
}
|
|
|
|
db := &DB{
|
|
opts: opts,
|
|
connectionPool: make(chan *conn, ClientMaxConnections),
|
|
doDial: make(chan struct{}, ClientMaxConnections),
|
|
}
|
|
|
|
for i := uint64(0); i < ClientMaxConnections; i++ {
|
|
db.doDial <- struct{}{}
|
|
}
|
|
|
|
ctx, cancelConnections := context.WithCancel(context.Background())
|
|
go func() {
|
|
<-parentCtx.Done()
|
|
cancelConnections()
|
|
}()
|
|
|
|
traceTicker := time.NewTicker(3 * time.Second)
|
|
pingTicker := time.NewTicker(db.opts.PingEvery)
|
|
db.doPing = pingTicker.C
|
|
go func() {
|
|
defer pingTicker.Stop()
|
|
defer traceTicker.Stop()
|
|
|
|
for {
|
|
select {
|
|
default:
|
|
case <-ctx.Done():
|
|
return
|
|
case <-traceTicker.C:
|
|
logger.Trace("connections in pool", "amount", len(db.connectionPool))
|
|
}
|
|
|
|
db.autoReconnect(ctx)
|
|
}
|
|
}()
|
|
|
|
return db, nil
|
|
}
|
|
|
|
func (db *DB) autoReconnect(ctx context.Context) {
|
|
select {
|
|
case <-db.doDial:
|
|
dialCtx, cancel := context.WithTimeout(ctx, db.opts.DialTimeout)
|
|
defer cancel()
|
|
newIn, newOut, newCloser, err := db.opts.DialFunc(dialCtx)
|
|
if err != nil {
|
|
logger.Warn("dial failed", "err", err)
|
|
db.doDial <- struct{}{}
|
|
time.Sleep(db.opts.RetryDialAfter)
|
|
return
|
|
}
|
|
|
|
notifyCloser := notifyOnClose{notifyCh: db.doDial, internal: newCloser}
|
|
db.returnConn(ctx, newIn, newOut, notifyCloser)
|
|
case <-db.doPing:
|
|
// periodically ping to close broken connections
|
|
pingCtx, cancel := context.WithTimeout(ctx, db.opts.PingTimeout)
|
|
defer cancel()
|
|
if err := db.ping(pingCtx); err != nil {
|
|
if !errors.Is(err, io.EOF) { // io.EOF means server gone
|
|
logger.Warn("ping failed", "err", err)
|
|
return
|
|
}
|
|
|
|
// if server gone, then need re-check all connections by ping. It will remove broken connections from pool.
|
|
for i := uint64(0); i < ClientMaxConnections-1; i++ {
|
|
pingCtx, cancel := context.WithTimeout(ctx, db.opts.PingTimeout)
|
|
_ = db.ping(pingCtx)
|
|
cancel()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Close closes DB by using the closer field
|
|
func (db *DB) Close() error {
|
|
return nil
|
|
}
|
|
|
|
// Tx mimicks the interface of bolt.Tx
|
|
type Tx struct {
|
|
ctx context.Context
|
|
in io.Reader
|
|
out io.Writer
|
|
}
|
|
|
|
func (db *DB) endTx(ctx context.Context, encoder *codec.Encoder, decoder *codec.Decoder) error {
|
|
_ = ctx
|
|
var responseCode ResponseCode
|
|
|
|
if err := encoder.Encode(CmdEndTx); err != nil {
|
|
return fmt.Errorf("could not encode CmdEndTx: %w", err)
|
|
}
|
|
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode ResponseCode for CmdEndTx: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return fmt.Errorf("could not decode errorMessage for CmdEndTx: %w", err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (db *DB) CmdGetAsOf(ctx context.Context, bucket, hBucket, key []byte, timestamp uint64) ([]byte, error) {
|
|
var err error
|
|
var responseCode ResponseCode
|
|
in, out, closer, err := db.getConnection(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
defer func() {
|
|
if err != nil {
|
|
closer.Close()
|
|
return
|
|
}
|
|
db.returnConn(ctx, in, out, closer)
|
|
}()
|
|
|
|
decoder := codecpool.Decoder(in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
err = encoder.Encode(CmdGetAsOf)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGetAsOf: %w", err)
|
|
}
|
|
err = encoder.Encode(&bucket)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGetAsOf: %w", err)
|
|
}
|
|
err = encoder.Encode(&hBucket)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGetAsOf: %w", err)
|
|
}
|
|
err = encoder.Encode(&key)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGetAsOf: %w", err)
|
|
}
|
|
err = encoder.Encode(timestamp)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGetAsOf: %w", err)
|
|
}
|
|
|
|
err = decoder.Decode(&responseCode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
err = decodeErr(decoder, responseCode)
|
|
return nil, err
|
|
}
|
|
|
|
var val []byte
|
|
err = decoder.Decode(&val)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return val, nil
|
|
}
|
|
|
|
// View performs read-only transaction on the remote database
|
|
// NOTE: not thread-safe
|
|
func (db *DB) View(ctx context.Context, f func(tx *Tx) error) (err error) {
|
|
var opErr error
|
|
var endTxErr error
|
|
|
|
var responseCode ResponseCode
|
|
|
|
in, out, closer, err := db.getConnection(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
defer func() {
|
|
if err != nil || endTxErr != nil || opErr != nil {
|
|
if closeErr := closer.Close(); closeErr != nil {
|
|
logger.Error("can't close connection", "err", closeErr)
|
|
}
|
|
return
|
|
}
|
|
db.returnConn(ctx, in, out, closer)
|
|
}()
|
|
|
|
decoder := codecpool.Decoder(in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err = encoder.Encode(CmdBeginTx); err != nil {
|
|
return fmt.Errorf("could not encode CmdBeginTx: %w", err)
|
|
}
|
|
|
|
if err = decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode response code of CmdBeginTx: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
return decodeErr(decoder, responseCode)
|
|
}
|
|
|
|
tx := &Tx{ctx: ctx, in: in, out: out}
|
|
opErr = f(tx)
|
|
|
|
endTxErr = db.endTx(ctx, encoder, decoder)
|
|
if endTxErr != nil {
|
|
logger.Warn("could not finish tx", "err", err)
|
|
}
|
|
|
|
return opErr
|
|
}
|
|
|
|
// Bucket mimicks the interface of bolt.Bucket
|
|
type Bucket struct {
|
|
ctx context.Context
|
|
in io.Reader
|
|
out io.Writer
|
|
bucketHandle uint64
|
|
|
|
name []byte
|
|
initialized bool
|
|
tx *Tx
|
|
}
|
|
|
|
type Cursor struct {
|
|
ctx context.Context
|
|
in io.Reader
|
|
out io.Writer
|
|
opts CursorOpts
|
|
cursorHandle uint64
|
|
cacheLastIdx uint64
|
|
cacheIdx uint64
|
|
cacheKeys [][]byte
|
|
cacheValues [][]byte
|
|
cacheValueIsEmpty []bool
|
|
|
|
bucket *Bucket
|
|
initialized bool
|
|
}
|
|
|
|
type CursorOpts struct {
|
|
prefetchSize uint64
|
|
prefetchValues bool
|
|
}
|
|
|
|
var DefaultCursorOpts = CursorOpts{
|
|
prefetchSize: DefaultCursorBatchSize,
|
|
prefetchValues: false,
|
|
}
|
|
|
|
func (opts CursorOpts) PrefetchSize(v uint64) CursorOpts {
|
|
opts.prefetchSize = v
|
|
return opts
|
|
}
|
|
|
|
func (opts CursorOpts) PrefetchValues(v bool) CursorOpts {
|
|
opts.prefetchValues = v
|
|
return opts
|
|
}
|
|
|
|
// Bucket returns the handle to the bucket in remote DB
|
|
func (tx *Tx) Bucket(name []byte) *Bucket {
|
|
return &Bucket{tx: tx, ctx: tx.ctx, in: tx.in, out: tx.out, name: name}
|
|
}
|
|
|
|
func (b *Bucket) init() error {
|
|
decoder := codecpool.Decoder(b.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(b.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(CmdBucket); err != nil {
|
|
return fmt.Errorf("could not encode CmdBucket: %w", err)
|
|
}
|
|
if err := encoder.Encode(&b.name); err != nil {
|
|
return fmt.Errorf("could not encode name for CmdBucket: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode ResponseCode for CmdBucket: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return fmt.Errorf("could not decode errorMessage for CmdBucket: %w", err)
|
|
}
|
|
}
|
|
|
|
var bucketHandle uint64
|
|
if err := decoder.Decode(&bucketHandle); err != nil {
|
|
return fmt.Errorf("could not decode bucketHandle for CmdBucket: %w", err)
|
|
}
|
|
if bucketHandle == 0 {
|
|
return fmt.Errorf("unexpected bucketHandle: 0")
|
|
}
|
|
|
|
b.bucketHandle = bucketHandle
|
|
return nil
|
|
}
|
|
|
|
// Get reads a value corresponding to the given key, from the bucket
|
|
// return nil if they key is not present
|
|
func (b *Bucket) Get(key []byte) ([]byte, error) {
|
|
if !b.initialized {
|
|
if err := b.init(); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
select {
|
|
default:
|
|
case <-b.ctx.Done():
|
|
return nil, b.ctx.Err()
|
|
}
|
|
|
|
decoder := codecpool.Decoder(b.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(b.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(CmdGet); err != nil {
|
|
return nil, fmt.Errorf("could not encode CmdGet: %w", err)
|
|
}
|
|
if err := encoder.Encode(b.bucketHandle); err != nil {
|
|
return nil, fmt.Errorf("could not encode bucketHandle for CmdGet: %w", err)
|
|
}
|
|
if err := encoder.Encode(&key); err != nil {
|
|
return nil, fmt.Errorf("could not encode key for CmdGet: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return nil, fmt.Errorf("could not decode ResponseCode for CmdGet: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return nil, fmt.Errorf("could not decode errorMessage for CmdGet: %w", err)
|
|
}
|
|
}
|
|
|
|
var value []byte
|
|
if err := decoder.Decode(&value); err != nil {
|
|
return nil, fmt.Errorf("could not decode value for CmdGet: %w", err)
|
|
}
|
|
return value, nil
|
|
}
|
|
|
|
// Cursor iterating over bucket keys
|
|
func (b *Bucket) Cursor(opts CursorOpts) *Cursor {
|
|
return &Cursor{
|
|
ctx: b.ctx,
|
|
opts: opts,
|
|
in: b.in,
|
|
out: b.out,
|
|
}
|
|
}
|
|
|
|
func (c *Cursor) init() error {
|
|
if !c.bucket.initialized {
|
|
if err := c.bucket.init(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
decoder := codecpool.Decoder(c.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(c.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(CmdCursor); err != nil {
|
|
return fmt.Errorf("could not encode CmdCursor: %w", err)
|
|
}
|
|
if err := encoder.Encode(c.bucket.bucketHandle); err != nil {
|
|
return fmt.Errorf("could not encode bucketHandle for CmdCursor: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode ResponseCode for CmdCursor: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return fmt.Errorf("could not decode errorMessage for CmdCursor: %w", err)
|
|
}
|
|
}
|
|
|
|
var cursorHandle uint64
|
|
if err := decoder.Decode(&cursorHandle); err != nil {
|
|
return fmt.Errorf("could not decode cursorHandle for CmdCursor: %w", err)
|
|
}
|
|
|
|
if cursorHandle == 0 { // Retrieve the error
|
|
return fmt.Errorf("unexpected bucketHandle: 0")
|
|
}
|
|
|
|
c.cursorHandle = cursorHandle
|
|
return nil
|
|
}
|
|
|
|
func (c *Cursor) First() (key []byte, value []byte, err error) {
|
|
if !c.initialized {
|
|
if err := c.init(); err != nil {
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
|
|
cmd := CmdCursorFirstKey
|
|
if c.opts.prefetchValues {
|
|
cmd = CmdCursorFirst
|
|
}
|
|
|
|
if err := c.fetchPage(cmd); err != nil {
|
|
return nil, nil, err
|
|
}
|
|
c.cacheIdx = 0
|
|
|
|
k, v := c.cacheKeys[c.cacheIdx], c.cacheValues[c.cacheIdx]
|
|
c.cacheIdx++
|
|
return k, v, nil
|
|
|
|
}
|
|
|
|
func (c *Cursor) FirstKey() (key []byte, vIsEmpty bool, err error) {
|
|
if !c.initialized {
|
|
if err := c.init(); err != nil {
|
|
return nil, false, err
|
|
}
|
|
}
|
|
|
|
if err := c.fetchPage(CmdCursorFirstKey); err != nil {
|
|
return nil, false, err
|
|
}
|
|
c.cacheIdx = 0
|
|
|
|
k, v := c.cacheKeys[c.cacheIdx], c.cacheValueIsEmpty[c.cacheIdx]
|
|
c.cacheIdx++
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *Cursor) SeekKey(seek []byte) (key []byte, vIsEmpty bool, err error) {
|
|
key, v, err := c.Seek(seek)
|
|
vIsEmpty = len(v) == 0
|
|
return key, vIsEmpty, err
|
|
}
|
|
|
|
func (c *Cursor) Seek(seek []byte) (key []byte, value []byte, err error) {
|
|
if !c.initialized {
|
|
if err := c.init(); err != nil {
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
|
|
c.cacheLastIdx = 0 // .Next() cache is invalid after .Seek() and .SeekTo() calls
|
|
|
|
select {
|
|
default:
|
|
case <-c.ctx.Done():
|
|
return nil, nil, c.ctx.Err()
|
|
}
|
|
|
|
decoder := codecpool.Decoder(c.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(c.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(CmdCursorSeek); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode CmdCursorSeek: %w", err)
|
|
}
|
|
if err := encoder.Encode(c.cursorHandle); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode cursorHandle for CmdCursorSeek: %w", err)
|
|
}
|
|
if err := encoder.Encode(&seek); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode key for CmdCursorSeek: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode ResponseCode for CmdCursorSeek: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode errorMessage for CmdCursorSeek: %w", err)
|
|
}
|
|
}
|
|
|
|
if err := decoder.Decode(&key); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode key for CmdCursorSeek: %w", err)
|
|
}
|
|
|
|
if err := decoder.Decode(&value); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode value for CmdCursorSeek: %w", err)
|
|
}
|
|
|
|
return key, value, nil
|
|
}
|
|
|
|
func (c *Cursor) SeekTo(seek []byte) (key []byte, value []byte, err error) {
|
|
if !c.initialized {
|
|
if err := c.init(); err != nil {
|
|
return nil, nil, err
|
|
}
|
|
}
|
|
|
|
c.cacheLastIdx = 0 // .Next() cache is invalid after .Seek() and .SeekTo() calls
|
|
|
|
select {
|
|
default:
|
|
case <-c.ctx.Done():
|
|
return nil, nil, c.ctx.Err()
|
|
}
|
|
|
|
decoder := codecpool.Decoder(c.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(c.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(CmdCursorSeekTo); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode CmdCursorSeekTo: %w", err)
|
|
}
|
|
if err := encoder.Encode(c.cursorHandle); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode cursorHandle for CmdCursorSeekTo: %w", err)
|
|
}
|
|
if err := encoder.Encode(&seek); err != nil {
|
|
return nil, nil, fmt.Errorf("could not encode key for CmdCursorSeekTo: %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode ResponseCode for CmdCursorSeekTo: %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode errorMessage for CmdCursorSeekTo: %w", err)
|
|
}
|
|
}
|
|
|
|
if err := decodeKeyValue(decoder, &key, &value); err != nil {
|
|
return nil, nil, fmt.Errorf("could not decode (key, value) for CmdCursorSeekTo: %w", err)
|
|
}
|
|
|
|
return key, value, nil
|
|
}
|
|
|
|
func (c *Cursor) needFetchNextPage() bool {
|
|
res := c.cacheLastIdx == 0 || // cache is empty
|
|
c.cacheIdx == c.cacheLastIdx // all cache read
|
|
return res
|
|
}
|
|
|
|
func (c *Cursor) Next() (keys []byte, values []byte, err error) {
|
|
if c.needFetchNextPage() {
|
|
err := c.fetchPage(CmdCursorNext)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
c.cacheIdx = 0
|
|
}
|
|
|
|
k, v := c.cacheKeys[c.cacheIdx], c.cacheValues[c.cacheIdx]
|
|
c.cacheIdx++
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *Cursor) NextKey() (keys []byte, vIsEmpty bool, err error) {
|
|
if c.needFetchNextPage() {
|
|
err := c.fetchPage(CmdCursorNextKey)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
c.cacheIdx = 0
|
|
}
|
|
|
|
k, v := c.cacheKeys[c.cacheIdx], c.cacheValueIsEmpty[c.cacheIdx]
|
|
c.cacheIdx++
|
|
|
|
return k, v, nil
|
|
}
|
|
|
|
func (c *Cursor) fetchPage(cmd Command) error {
|
|
if c.cacheKeys == nil {
|
|
c.cacheKeys = make([][]byte, c.opts.prefetchSize)
|
|
c.cacheValues = make([][]byte, c.opts.prefetchSize)
|
|
c.cacheValueIsEmpty = make([]bool, c.opts.prefetchSize)
|
|
}
|
|
|
|
decoder := codecpool.Decoder(c.in)
|
|
defer codecpool.Return(decoder)
|
|
encoder := codecpool.Encoder(c.out)
|
|
defer codecpool.Return(encoder)
|
|
|
|
if err := encoder.Encode(cmd); err != nil {
|
|
return fmt.Errorf("could not encode command %d. %w", cmd, err)
|
|
}
|
|
if err := encoder.Encode(c.cursorHandle); err != nil {
|
|
return fmt.Errorf("could not encode cursorHandle. %w", err)
|
|
}
|
|
|
|
if err := encoder.Encode(c.opts.prefetchSize); err != nil {
|
|
return fmt.Errorf("could not encode c.batchSize. %w", err)
|
|
}
|
|
|
|
var responseCode ResponseCode
|
|
if err := decoder.Decode(&responseCode); err != nil {
|
|
return fmt.Errorf("could not decode ResponseCode. %w", err)
|
|
}
|
|
|
|
if responseCode != ResponseOk {
|
|
if err := decodeErr(decoder, responseCode); err != nil {
|
|
return fmt.Errorf("could not decode errorMessage. %w", err)
|
|
}
|
|
}
|
|
|
|
for c.cacheLastIdx = uint64(0); c.cacheLastIdx < c.opts.prefetchSize; c.cacheLastIdx++ {
|
|
select {
|
|
default:
|
|
case <-c.ctx.Done():
|
|
return c.ctx.Err()
|
|
}
|
|
|
|
switch cmd {
|
|
case CmdCursorFirst, CmdCursorNext:
|
|
if err := decodeKeyValue(decoder, &c.cacheKeys[c.cacheLastIdx], &c.cacheValues[c.cacheLastIdx]); err != nil {
|
|
return fmt.Errorf("could not decode (key, value) for cmd %d: %w", cmd, err)
|
|
}
|
|
case CmdCursorFirstKey, CmdCursorNextKey:
|
|
if err := decodeKey(decoder, &c.cacheKeys[c.cacheLastIdx], &c.cacheValueIsEmpty[c.cacheLastIdx]); err != nil {
|
|
return fmt.Errorf("could not decode (key, valueIsEmpty) for cmd %d: %w", cmd, err)
|
|
}
|
|
}
|
|
|
|
if c.cacheKeys[c.cacheLastIdx] == nil {
|
|
break
|
|
}
|
|
}
|
|
return nil
|
|
}
|