2020-07-27 12:15:48 +00:00
|
|
|
package remotedbserver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
grpc_middleware "github.com/grpc-ecosystem/go-grpc-middleware"
|
|
|
|
grpc_recovery "github.com/grpc-ecosystem/go-grpc-middleware/recovery"
|
|
|
|
grpc_prometheus "github.com/grpc-ecosystem/go-grpc-prometheus"
|
2020-08-11 21:09:30 +00:00
|
|
|
"google.golang.org/grpc"
|
2020-09-11 20:17:37 +00:00
|
|
|
"google.golang.org/grpc/credentials"
|
2020-08-11 21:09:30 +00:00
|
|
|
|
2020-07-27 12:15:48 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
2020-08-11 21:09:30 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/core"
|
2020-07-27 12:15:48 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb/remote"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/log"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/metrics"
|
|
|
|
)
|
|
|
|
|
2020-09-07 05:47:08 +00:00
|
|
|
const MaxTxTTL = 30 * time.Second
|
2020-07-27 12:15:48 +00:00
|
|
|
|
|
|
|
type KvServer struct {
|
|
|
|
remote.UnimplementedKVServer // must be embedded to have forward compatible implementations.
|
|
|
|
|
|
|
|
kv ethdb.KV
|
|
|
|
}
|
|
|
|
|
2020-09-11 20:17:37 +00:00
|
|
|
func StartGrpc(kv ethdb.KV, eth core.Backend, addr string, creds *credentials.TransportCredentials) {
|
2020-07-27 12:15:48 +00:00
|
|
|
log.Info("Starting private RPC server", "on", addr)
|
|
|
|
lis, err := net.Listen("tcp", addr)
|
|
|
|
if err != nil {
|
2020-08-14 06:41:18 +00:00
|
|
|
log.Error("Could not create listener", "address", addr, "err", err)
|
2020-07-27 12:15:48 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
kvSrv := NewKvServer(kv)
|
2020-07-29 04:31:46 +00:00
|
|
|
dbSrv := NewDBServer(kv)
|
2020-08-12 13:47:59 +00:00
|
|
|
ethBackendSrv := NewEthBackendServer(eth)
|
2020-07-27 12:15:48 +00:00
|
|
|
var (
|
|
|
|
streamInterceptors []grpc.StreamServerInterceptor
|
|
|
|
unaryInterceptors []grpc.UnaryServerInterceptor
|
|
|
|
)
|
|
|
|
if metrics.Enabled {
|
|
|
|
streamInterceptors = append(streamInterceptors, grpc_prometheus.StreamServerInterceptor)
|
|
|
|
unaryInterceptors = append(unaryInterceptors, grpc_prometheus.UnaryServerInterceptor)
|
|
|
|
}
|
|
|
|
streamInterceptors = append(streamInterceptors, grpc_recovery.StreamServerInterceptor())
|
|
|
|
unaryInterceptors = append(unaryInterceptors, grpc_recovery.UnaryServerInterceptor())
|
2020-09-11 20:17:37 +00:00
|
|
|
var grpcServer *grpc.Server
|
|
|
|
if creds == nil {
|
|
|
|
grpcServer = grpc.NewServer(
|
|
|
|
grpc.NumStreamWorkers(20), // reduce amount of goroutines
|
|
|
|
grpc.WriteBufferSize(1024), // reduce buffers to save mem
|
|
|
|
grpc.ReadBufferSize(1024),
|
|
|
|
grpc.MaxConcurrentStreams(40), // to force clients reduce concurency level
|
|
|
|
grpc.StreamInterceptor(grpc_middleware.ChainStreamServer(streamInterceptors...)),
|
|
|
|
grpc.UnaryInterceptor(grpc_middleware.ChainUnaryServer(unaryInterceptors...)),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
grpcServer = grpc.NewServer(
|
|
|
|
grpc.NumStreamWorkers(20), // reduce amount of goroutines
|
|
|
|
grpc.WriteBufferSize(1024), // reduce buffers to save mem
|
|
|
|
grpc.ReadBufferSize(1024),
|
|
|
|
grpc.MaxConcurrentStreams(40), // to force clients reduce concurency level
|
|
|
|
grpc.StreamInterceptor(grpc_middleware.ChainStreamServer(streamInterceptors...)),
|
|
|
|
grpc.UnaryInterceptor(grpc_middleware.ChainUnaryServer(unaryInterceptors...)),
|
|
|
|
grpc.Creds(*creds),
|
|
|
|
)
|
|
|
|
}
|
2020-07-27 12:15:48 +00:00
|
|
|
remote.RegisterKVServer(grpcServer, kvSrv)
|
2020-07-29 04:31:46 +00:00
|
|
|
remote.RegisterDBServer(grpcServer, dbSrv)
|
2020-08-12 13:47:59 +00:00
|
|
|
remote.RegisterETHBACKENDServer(grpcServer, ethBackendSrv)
|
2020-07-27 12:15:48 +00:00
|
|
|
|
|
|
|
if metrics.Enabled {
|
|
|
|
grpc_prometheus.Register(grpcServer)
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
if err := grpcServer.Serve(lis); err != nil {
|
2020-08-14 06:41:18 +00:00
|
|
|
log.Error("private RPC server fail", "err", err)
|
2020-07-27 12:15:48 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewKvServer(kv ethdb.KV) *KvServer {
|
|
|
|
return &KvServer{kv: kv}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *KvServer) Seek(stream remote.KV_SeekServer) error {
|
|
|
|
in, recvErr := stream.Recv()
|
|
|
|
if recvErr != nil {
|
|
|
|
return recvErr
|
|
|
|
}
|
|
|
|
|
2020-08-11 10:35:59 +00:00
|
|
|
tx, err := s.kv.Begin(context.Background(), nil, false)
|
2020-07-27 12:15:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rollback := func() {
|
|
|
|
tx.Rollback()
|
|
|
|
}
|
|
|
|
defer rollback()
|
|
|
|
|
|
|
|
bucketName, prefix := in.BucketName, in.Prefix // 'in' value will cahnge, but this params will immutable
|
|
|
|
|
2020-08-14 06:41:18 +00:00
|
|
|
c := tx.Cursor(bucketName).Prefix(prefix)
|
2020-07-27 12:15:48 +00:00
|
|
|
|
2020-09-11 20:44:24 +00:00
|
|
|
txTicker := time.NewTicker(MaxTxTTL)
|
|
|
|
defer txTicker.Stop()
|
2020-09-07 05:47:08 +00:00
|
|
|
|
2020-07-27 12:15:48 +00:00
|
|
|
// send all items to client, if k==nil - stil send it to client and break loop
|
|
|
|
for k, v, err := c.Seek(in.SeekKey); ; k, v, err = c.Next() {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = stream.Send(&remote.Pair{Key: common.CopyBytes(k), Value: common.CopyBytes(v)})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if k == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// if client not requested stream then wait signal from him before send any item
|
|
|
|
if !in.StartSreaming {
|
|
|
|
in, err = stream.Recv()
|
|
|
|
if err != nil {
|
|
|
|
if err == io.EOF {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-07 05:47:08 +00:00
|
|
|
//TODO: protect against client - which doesn't send any requests
|
|
|
|
select {
|
|
|
|
default:
|
2020-09-11 20:44:24 +00:00
|
|
|
case <-txTicker.C:
|
2020-07-27 12:15:48 +00:00
|
|
|
tx.Rollback()
|
2020-08-11 10:35:59 +00:00
|
|
|
tx, err = s.kv.Begin(context.Background(), nil, false)
|
2020-07-27 12:15:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-14 06:41:18 +00:00
|
|
|
c = tx.Cursor(bucketName).Prefix(prefix)
|
2020-07-27 12:15:48 +00:00
|
|
|
_, _, _ = c.Seek(k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|