package remotedb import ( "context" "fmt" "github.com/tendermint/tmlibs/db" "github.com/tendermint/tmlibs/grpcdb" protodb "github.com/tendermint/tmlibs/proto" ) type RemoteDB struct { ctx context.Context dc protodb.DBClient } func NewRemoteDB(serverAddr string, serverKey string) (*RemoteDB, error) { return newRemoteDB(grpcdb.NewClient(serverAddr, serverKey)) } func newRemoteDB(gdc protodb.DBClient, err error) (*RemoteDB, error) { if err != nil { return nil, err } return &RemoteDB{dc: gdc, ctx: context.Background()}, nil } type Init struct { Dir string Name string Type string } func (rd *RemoteDB) InitRemote(in *Init) error { _, err := rd.dc.Init(rd.ctx, &protodb.Init{Dir: in.Dir, Type: in.Type, Name: in.Name}) return err } var _ db.DB = (*RemoteDB)(nil) // Close is a noop currently func (rd *RemoteDB) Close() { } func (rd *RemoteDB) Delete(key []byte) { if _, err := rd.dc.Delete(rd.ctx, &protodb.Entity{Key: key}); err != nil { panic(fmt.Sprintf("RemoteDB.Delete: %v", err)) } } func (rd *RemoteDB) DeleteSync(key []byte) { if _, err := rd.dc.DeleteSync(rd.ctx, &protodb.Entity{Key: key}); err != nil { panic(fmt.Sprintf("RemoteDB.DeleteSync: %v", err)) } } func (rd *RemoteDB) Set(key, value []byte) { if _, err := rd.dc.Set(rd.ctx, &protodb.Entity{Key: key, Value: value}); err != nil { panic(fmt.Sprintf("RemoteDB.Set: %v", err)) } } func (rd *RemoteDB) SetSync(key, value []byte) { if _, err := rd.dc.SetSync(rd.ctx, &protodb.Entity{Key: key, Value: value}); err != nil { panic(fmt.Sprintf("RemoteDB.SetSync: %v", err)) } } func (rd *RemoteDB) Get(key []byte) []byte { res, err := rd.dc.Get(rd.ctx, &protodb.Entity{Key: key}) if err != nil { panic(fmt.Sprintf("RemoteDB.Get error: %v", err)) } return res.Value } func (rd *RemoteDB) Has(key []byte) bool { res, err := rd.dc.Has(rd.ctx, &protodb.Entity{Key: key}) if err != nil { panic(fmt.Sprintf("RemoteDB.Has error: %v", err)) } return res.Exists } func (rd *RemoteDB) ReverseIterator(start, end []byte) db.Iterator { dic, err := rd.dc.ReverseIterator(rd.ctx, &protodb.Entity{Start: start, End: end}) if err != nil { panic(fmt.Sprintf("RemoteDB.Iterator error: %v", err)) } return makeReverseIterator(dic) } // TODO: Implement NewBatch func (rd *RemoteDB) NewBatch() db.Batch { panic("Unimplemented") } // TODO: Implement Print when db.DB implements a method // to print to a string and not db.Print to stdout. func (rd *RemoteDB) Print() { panic("Unimplemented") } func (rd *RemoteDB) Stats() map[string]string { stats, err := rd.dc.Stats(rd.ctx, &protodb.Nothing{}) if err != nil { panic(fmt.Sprintf("RemoteDB.Stats error: %v", err)) } if stats == nil { return nil } return stats.Data } func (rd *RemoteDB) Iterator(start, end []byte) db.Iterator { dic, err := rd.dc.Iterator(rd.ctx, &protodb.Entity{Start: start, End: end}) if err != nil { panic(fmt.Sprintf("RemoteDB.Iterator error: %v", err)) } return makeIterator(dic) } func makeIterator(dic protodb.DB_IteratorClient) db.Iterator { return &iterator{dic: dic} } func makeReverseIterator(dric protodb.DB_ReverseIteratorClient) db.Iterator { return &reverseIterator{dric: dric} } type reverseIterator struct { dric protodb.DB_ReverseIteratorClient cur *protodb.Iterator } var _ db.Iterator = (*iterator)(nil) func (rItr *reverseIterator) Valid() bool { return rItr.cur != nil && rItr.cur.Valid } func (rItr *reverseIterator) Domain() (start, end []byte) { if rItr.cur == nil || rItr.cur.Domain == nil { return nil, nil } return rItr.cur.Domain.Start, rItr.cur.Domain.End } // Next advances the current reverseIterator func (rItr *reverseIterator) Next() { var err error rItr.cur, err = rItr.dric.Recv() if err != nil { panic(fmt.Sprintf("RemoteDB.ReverseIterator.Next error: %v", err)) } } func (rItr *reverseIterator) Key() []byte { if rItr.cur == nil { return nil } return rItr.cur.Key } func (rItr *reverseIterator) Value() []byte { if rItr.cur == nil { return nil } return rItr.cur.Value } func (rItr *reverseIterator) Close() { } // iterator implements the db.Iterator by retrieving // streamed iterators from the remote backend as // needed. It is NOT safe for concurrent usage, // matching the behavior of other iterators. type iterator struct { dic protodb.DB_IteratorClient cur *protodb.Iterator } var _ db.Iterator = (*iterator)(nil) func (itr *iterator) Valid() bool { return itr.cur != nil && itr.cur.Valid } func (itr *iterator) Domain() (start, end []byte) { if itr.cur == nil || itr.cur.Domain == nil { return nil, nil } return itr.cur.Domain.Start, itr.cur.Domain.End } // Next advances the current iterator func (itr *iterator) Next() { var err error itr.cur, err = itr.dic.Recv() if err != nil { panic(fmt.Sprintf("RemoteDB.Iterator.Next error: %v", err)) } } func (itr *iterator) Key() []byte { if itr.cur == nil { return nil } return itr.cur.Key } func (itr *iterator) Value() []byte { if itr.cur == nil { return nil } return itr.cur.Value } func (itr *iterator) Close() { // TODO: Shut down the iterator }