2021-07-06 18:39:56 -07:00
|
|
|
package blockrepo
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
2021-07-27 09:34:15 -04:00
|
|
|
"github.com/pkg/errors"
|
2021-07-06 18:39:56 -07:00
|
|
|
|
|
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
|
|
|
|
|
|
"github.com/cockroachdb/pebble"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Pebble struct {
|
|
|
|
db *pebble.DB
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPebble(path string) (*Pebble, error) {
|
|
|
|
|
|
|
|
db, err := pebble.Open(path, nil)
|
|
|
|
repo := &Pebble{db: db}
|
|
|
|
|
2021-07-27 09:34:15 -04:00
|
|
|
return repo, errors.Wrapf(err, "unable to open %s", path)
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Pebble) Load() (int32, error) {
|
|
|
|
|
|
|
|
iter := repo.db.NewIter(nil)
|
|
|
|
if !iter.Last() {
|
2021-07-27 09:34:15 -04:00
|
|
|
err := iter.Close()
|
|
|
|
return 0, errors.Wrap(err, "closing iterator with no last")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
height := int32(binary.BigEndian.Uint32(iter.Key()))
|
2021-07-27 09:34:15 -04:00
|
|
|
err := iter.Close()
|
|
|
|
return height, errors.Wrap(err, "closing iterator")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Pebble) Get(height int32) (*chainhash.Hash, error) {
|
|
|
|
|
|
|
|
key := make([]byte, 4)
|
|
|
|
binary.BigEndian.PutUint32(key, uint32(height))
|
|
|
|
|
|
|
|
b, closer, err := repo.db.Get(key)
|
2021-07-27 09:34:15 -04:00
|
|
|
if closer != nil {
|
|
|
|
defer closer.Close()
|
|
|
|
}
|
2021-07-06 18:39:56 -07:00
|
|
|
if err != nil {
|
2021-07-27 09:34:15 -04:00
|
|
|
return nil, errors.Wrap(err, "in get")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
hash, err := chainhash.NewHash(b)
|
2021-07-27 09:34:15 -04:00
|
|
|
return hash, errors.Wrap(err, "creating hash")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Pebble) Set(height int32, hash *chainhash.Hash) error {
|
|
|
|
|
|
|
|
key := make([]byte, 4)
|
|
|
|
binary.BigEndian.PutUint32(key, uint32(height))
|
|
|
|
|
2021-07-27 09:34:15 -04:00
|
|
|
return errors.WithStack(repo.db.Set(key, hash[:], pebble.NoSync))
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (repo *Pebble) Close() error {
|
|
|
|
|
|
|
|
err := repo.db.Flush()
|
|
|
|
if err != nil {
|
2021-07-27 09:34:15 -04:00
|
|
|
// if we fail to close are we going to try again later?
|
|
|
|
return errors.Wrap(err, "on flush")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
err = repo.db.Close()
|
2021-07-27 09:34:15 -04:00
|
|
|
return errors.Wrap(err, "on close")
|
2021-07-06 18:39:56 -07:00
|
|
|
}
|
2021-07-27 15:10:17 -04:00
|
|
|
|
|
|
|
func (repo *Pebble) Flush() error {
|
|
|
|
_, err := repo.db.AsyncFlush()
|
|
|
|
return err
|
|
|
|
}
|