2021-05-22 10:00:13 +00:00
|
|
|
package services
|
2020-08-12 13:47:59 +00:00
|
|
|
|
|
|
|
import (
|
2021-03-20 03:52:00 +00:00
|
|
|
"context"
|
2021-03-23 09:00:07 +00:00
|
|
|
"errors"
|
2021-05-22 10:00:13 +00:00
|
|
|
"fmt"
|
2021-03-20 03:52:00 +00:00
|
|
|
"io"
|
|
|
|
|
2021-05-20 18:25:53 +00:00
|
|
|
"github.com/ledgerwatch/erigon/common"
|
2021-05-22 10:00:13 +00:00
|
|
|
"github.com/ledgerwatch/erigon/ethdb/remote/remotedbserver"
|
2021-05-20 18:25:53 +00:00
|
|
|
"github.com/ledgerwatch/erigon/gointerfaces"
|
|
|
|
"github.com/ledgerwatch/erigon/gointerfaces/remote"
|
|
|
|
"github.com/ledgerwatch/erigon/log"
|
2021-05-04 01:37:17 +00:00
|
|
|
"google.golang.org/grpc"
|
2021-03-23 09:00:07 +00:00
|
|
|
"google.golang.org/grpc/status"
|
2021-05-22 10:00:13 +00:00
|
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
2020-08-12 13:47:59 +00:00
|
|
|
)
|
|
|
|
|
2021-03-23 09:00:07 +00:00
|
|
|
// ApiBackend - interface which must be used by API layer
|
|
|
|
// implementation can work with local Ethereum object or with Remote (grpc-based) one
|
|
|
|
// this is reason why all methods are accepting context and returning error
|
|
|
|
type ApiBackend interface {
|
|
|
|
Etherbase(ctx context.Context) (common.Address, error)
|
|
|
|
NetVersion(ctx context.Context) (uint64, error)
|
2021-06-17 21:55:20 +00:00
|
|
|
NetPeerCount(ctx context.Context) (uint64, error)
|
2021-04-24 15:46:29 +00:00
|
|
|
ProtocolVersion(ctx context.Context) (uint64, error)
|
|
|
|
ClientVersion(ctx context.Context) (string, error)
|
2021-03-23 09:00:07 +00:00
|
|
|
Subscribe(ctx context.Context, cb func(*remote.SubscribeReply)) error
|
2020-08-12 13:47:59 +00:00
|
|
|
}
|
|
|
|
|
2021-03-20 03:52:00 +00:00
|
|
|
type RemoteBackend struct {
|
|
|
|
remoteEthBackend remote.ETHBACKENDClient
|
|
|
|
log log.Logger
|
2021-05-22 10:00:13 +00:00
|
|
|
version gointerfaces.Version
|
2021-03-20 03:52:00 +00:00
|
|
|
}
|
|
|
|
|
2021-05-04 01:37:17 +00:00
|
|
|
func NewRemoteBackend(cc grpc.ClientConnInterface) *RemoteBackend {
|
2021-03-20 03:52:00 +00:00
|
|
|
return &RemoteBackend{
|
2021-05-04 01:37:17 +00:00
|
|
|
remoteEthBackend: remote.NewETHBACKENDClient(cc),
|
2021-05-22 10:00:13 +00:00
|
|
|
version: gointerfaces.VersionFromProto(remotedbserver.EthBackendAPIVersion),
|
|
|
|
log: log.New("remote_service", "eth_backend"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (back *RemoteBackend) EnsureVersionCompatibility() bool {
|
|
|
|
versionReply, err := back.remoteEthBackend.Version(context.Background(), &emptypb.Empty{}, grpc.WaitForReady(true))
|
|
|
|
if err != nil {
|
|
|
|
|
|
|
|
back.log.Error("getting Version", "error", err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if !gointerfaces.EnsureVersion(back.version, versionReply) {
|
|
|
|
back.log.Error("incompatible interface versions", "client", back.version.String(),
|
|
|
|
"server", fmt.Sprintf("%d.%d.%d", versionReply.Major, versionReply.Minor, versionReply.Patch))
|
|
|
|
return false
|
2021-03-20 03:52:00 +00:00
|
|
|
}
|
2021-05-22 10:00:13 +00:00
|
|
|
back.log.Info("interfaces compatible", "client", back.version.String(),
|
|
|
|
"server", fmt.Sprintf("%d.%d.%d", versionReply.Major, versionReply.Minor, versionReply.Patch))
|
|
|
|
return true
|
2021-03-20 03:52:00 +00:00
|
|
|
}
|
|
|
|
|
2021-03-23 09:00:07 +00:00
|
|
|
func (back *RemoteBackend) Etherbase(ctx context.Context) (common.Address, error) {
|
|
|
|
res, err := back.remoteEthBackend.Etherbase(ctx, &remote.EtherbaseRequest{})
|
2021-03-20 03:52:00 +00:00
|
|
|
if err != nil {
|
2021-03-23 09:00:07 +00:00
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return common.Address{}, errors.New(s.Message())
|
|
|
|
}
|
2021-03-20 03:52:00 +00:00
|
|
|
return common.Address{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return gointerfaces.ConvertH160toAddress(res.Address), nil
|
|
|
|
}
|
|
|
|
|
2021-03-23 09:00:07 +00:00
|
|
|
func (back *RemoteBackend) NetVersion(ctx context.Context) (uint64, error) {
|
|
|
|
res, err := back.remoteEthBackend.NetVersion(ctx, &remote.NetVersionRequest{})
|
2021-03-20 03:52:00 +00:00
|
|
|
if err != nil {
|
2021-03-23 09:00:07 +00:00
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return 0, errors.New(s.Message())
|
|
|
|
}
|
2021-03-20 03:52:00 +00:00
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return res.Id, nil
|
|
|
|
}
|
|
|
|
|
2021-06-17 21:55:20 +00:00
|
|
|
func (back *RemoteBackend) NetPeerCount(ctx context.Context) (uint64, error) {
|
|
|
|
res, err := back.remoteEthBackend.NetPeerCount(ctx, &remote.NetPeerCountRequest{})
|
|
|
|
if err != nil {
|
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return 0, errors.New(s.Message())
|
|
|
|
}
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2021-06-18 13:34:15 +00:00
|
|
|
return res.Count, nil
|
2021-06-17 21:55:20 +00:00
|
|
|
}
|
|
|
|
|
2021-04-24 15:46:29 +00:00
|
|
|
func (back *RemoteBackend) ProtocolVersion(ctx context.Context) (uint64, error) {
|
|
|
|
res, err := back.remoteEthBackend.ProtocolVersion(ctx, &remote.ProtocolVersionRequest{})
|
|
|
|
if err != nil {
|
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return 0, errors.New(s.Message())
|
|
|
|
}
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return res.Id, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (back *RemoteBackend) ClientVersion(ctx context.Context) (string, error) {
|
|
|
|
res, err := back.remoteEthBackend.ClientVersion(ctx, &remote.ClientVersionRequest{})
|
|
|
|
if err != nil {
|
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return "", errors.New(s.Message())
|
|
|
|
}
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return res.NodeName, nil
|
|
|
|
}
|
|
|
|
|
2021-03-23 09:00:07 +00:00
|
|
|
func (back *RemoteBackend) Subscribe(ctx context.Context, onNewEvent func(*remote.SubscribeReply)) error {
|
2021-05-04 01:37:17 +00:00
|
|
|
subscription, err := back.remoteEthBackend.Subscribe(ctx, &remote.SubscribeRequest{}, grpc.WaitForReady(true))
|
2021-03-20 03:52:00 +00:00
|
|
|
if err != nil {
|
2021-03-23 09:00:07 +00:00
|
|
|
if s, ok := status.FromError(err); ok {
|
|
|
|
return errors.New(s.Message())
|
|
|
|
}
|
2021-03-20 03:52:00 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
event, err := subscription.Recv()
|
|
|
|
if err == io.EOF {
|
|
|
|
log.Info("rpcdaemon: the subscription channel was closed")
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
onNewEvent(event)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|