Lite wallet server https://hush.is
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

143 lines
3.4 KiB

package frontend
import (
"context"
"database/sql"
"encoding/hex"
"errors"
_ "github.com/mattn/go-sqlite3"
"github.com/gtank/ctxd/rpc"
"github.com/gtank/ctxd/storage"
)
var (
ErrNoImpl = errors.New("not yet implemented")
ErrUnspecified = errors.New("request for unspecified identifier")
)
// the service type
type SqlStreamer struct {
db *sql.DB
}
func NewSQLiteStreamer(dbPath string) (rpc.CompactTxStreamerServer, error) {
db, err := sql.Open("sqlite3", dbPath)
if err != nil {
return nil, err
}
// Creates our tables if they don't already exist.
err = storage.CreateTables(db)
if err != nil {
return nil, err
}
return &SqlStreamer{db}, nil
}
func (s *SqlStreamer) GracefulStop() error {
return s.db.Close()
}
func (s *SqlStreamer) GetLatestBlock(ctx context.Context, placeholder *rpc.ChainSpec) (*rpc.BlockID, error) {
// the ChainSpec type is an empty placeholder
height, err := storage.GetCurrentHeight(ctx, s.db)
if err != nil {
return nil, err
}
// TODO: also return block hashes here
return &rpc.BlockID{Height: uint64(height)}, nil
}
func (s *SqlStreamer) GetBlock(ctx context.Context, id *rpc.BlockID) (*rpc.CompactBlock, error) {
if id.Height == 0 && id.Hash == nil {
return nil, ErrUnspecified
}
var blockBytes []byte
var err error
// Precedence: a hash is more specific than a height. If we have it, use it first.
if id.Hash != nil {
leHashString := hex.EncodeToString(id.Hash)
blockBytes, err = storage.GetBlockByHash(ctx, s.db, leHashString)
} else {
blockBytes, err = storage.GetBlock(ctx, s.db, int(id.Height))
}
if err != nil {
return nil, err
}
cBlock := &rpc.CompactBlock{}
err = proto.Unmarshal(blockBytes, cBlock)
return cBlock, err
}
func (s *SqlStreamer) GetBlockRange(span *rpc.BlockRange, resp rpc.CompactTxStreamer_GetBlockRangeServer) error {
blocks := make(chan []byte)
errors := make(chan error)
done := make(chan bool)
timeout := resp.Context().WithTimeout(1 * time.Second)
go GetBlockRange(timeout, s.db, blocks, errors, done, span.Start, span.End)
for {
select {
case <-timeout.Done():
return timeout.Err()
case err := <-errors:
return err
case blockBytes := <-blocks:
cBlock := &rpc.CompactBlock{}
err = proto.Unmarshal(blockBytes, cBlock)
if err != nil {
return err // TODO really need better logging in this whole service
}
err = resp.Send(cBlock)
if err != nil {
return err
}
}
}
return nil
}
func (s *SqlStreamer) GetTransaction(ctx context.Context, txf *rpc.TxFilter) (*rpc.RawTransaction, error) {
var txBytes []byte
var err error
if txf.Hash != nil {
leHashString := hex.EncodeToString(txf.Hash)
txBytes, err = storage.GetFullTxByHash(ctx, s.db, leHashString)
if err != nil {
return nil, err
}
return &rpc.RawTransaction{Data: txBytes}, nil
}
if txf.Block.Hash != nil {
leHashString := hex.EncodeToString(txf.Hash)
txBytes, err = storage.GetFullTxByHashAndIndex(ctx, s.db, leHashString, int(txf.Index))
if err != nil {
return nil, err
}
return &rpc.RawTransaction{Data: txBytes}, nil
}
// A totally unset protobuf will attempt to fetch the genesis coinbase tx.
txBytes, err = storage.GetFullTxByHeightAndIndex(ctx, s.db, int(txf.Block.Height), int(txf.Index))
if err != nil {
return nil, err
}
return &rpc.RawTransaction{Data: txBytes}, nil
}
func (s *SqlStreamer) SendTransaction(ctx context.Context, rawtx *rpc.RawTransaction) (*rpc.SendResponse, error) {
return nil, ErrNoImpl
}