github.com/ari-anchor/sei-tendermint@v0.0.0-20230519144642-dc826b7b56bb/abci/client/grpc_client.go (about)

     1  package abciclient
     2  
     3  import (
     4  	"context"
     5  	"errors"
     6  	"fmt"
     7  	"net"
     8  	"sync"
     9  	"time"
    10  
    11  	"github.com/ari-anchor/sei-tendermint/abci/types"
    12  	"github.com/ari-anchor/sei-tendermint/libs/log"
    13  	tmnet "github.com/ari-anchor/sei-tendermint/libs/net"
    14  	"github.com/ari-anchor/sei-tendermint/libs/service"
    15  	"google.golang.org/grpc"
    16  )
    17  
    18  // A gRPC client.
    19  type grpcClient struct {
    20  	service.BaseService
    21  	logger log.Logger
    22  
    23  	mustConnect bool
    24  
    25  	client types.ABCIApplicationClient
    26  	conn   *grpc.ClientConn
    27  
    28  	mtx  sync.Mutex
    29  	addr string
    30  	err  error
    31  }
    32  
    33  var _ Client = (*grpcClient)(nil)
    34  
    35  // NewGRPCClient creates a gRPC client, which will connect to addr upon the
    36  // start. Note Client#Start returns an error if connection is unsuccessful and
    37  // mustConnect is true.
    38  func NewGRPCClient(logger log.Logger, addr string, mustConnect bool) Client {
    39  	cli := &grpcClient{
    40  		logger:      logger,
    41  		addr:        addr,
    42  		mustConnect: mustConnect,
    43  	}
    44  	cli.BaseService = *service.NewBaseService(logger, "grpcClient", cli)
    45  	return cli
    46  }
    47  
    48  func dialerFunc(ctx context.Context, addr string) (net.Conn, error) {
    49  	return tmnet.Connect(addr)
    50  }
    51  
    52  func (cli *grpcClient) OnStart(ctx context.Context) error {
    53  	timer := time.NewTimer(0)
    54  	defer timer.Stop()
    55  
    56  RETRY_LOOP:
    57  	for {
    58  		conn, err := grpc.Dial(cli.addr,
    59  			grpc.WithInsecure(),
    60  			grpc.WithContextDialer(dialerFunc),
    61  		)
    62  		if err != nil {
    63  			if cli.mustConnect {
    64  				return err
    65  			}
    66  			cli.logger.Error(fmt.Sprintf("abci.grpcClient failed to connect to %v.  Retrying...\n", cli.addr), "err", err)
    67  			timer.Reset(time.Second * dialRetryIntervalSeconds)
    68  			select {
    69  			case <-ctx.Done():
    70  				return ctx.Err()
    71  			case <-timer.C:
    72  				continue RETRY_LOOP
    73  			}
    74  		}
    75  
    76  		cli.logger.Info("Dialed server. Waiting for echo.", "addr", cli.addr)
    77  		client := types.NewABCIApplicationClient(conn)
    78  		cli.conn = conn
    79  
    80  	ENSURE_CONNECTED:
    81  		for {
    82  			_, err := client.Echo(ctx, &types.RequestEcho{Message: "hello"}, grpc.WaitForReady(true))
    83  			if err == nil {
    84  				break ENSURE_CONNECTED
    85  			}
    86  			if errors.Is(err, context.Canceled) || errors.Is(err, context.DeadlineExceeded) {
    87  				return err
    88  			}
    89  
    90  			cli.logger.Error("Echo failed", "err", err)
    91  			timer.Reset(time.Second * echoRetryIntervalSeconds)
    92  			select {
    93  			case <-ctx.Done():
    94  				return ctx.Err()
    95  			case <-timer.C:
    96  				continue ENSURE_CONNECTED
    97  			}
    98  		}
    99  
   100  		cli.client = client
   101  		return nil
   102  	}
   103  }
   104  
   105  func (cli *grpcClient) OnStop() {
   106  	cli.mtx.Lock()
   107  	defer cli.mtx.Unlock()
   108  
   109  	if cli.conn != nil {
   110  		cli.err = cli.conn.Close()
   111  	}
   112  }
   113  
   114  func (cli *grpcClient) Error() error {
   115  	cli.mtx.Lock()
   116  	defer cli.mtx.Unlock()
   117  
   118  	return cli.err
   119  }
   120  
   121  //----------------------------------------
   122  
   123  func (cli *grpcClient) Flush(ctx context.Context) error { return nil }
   124  
   125  func (cli *grpcClient) Echo(ctx context.Context, msg string) (*types.ResponseEcho, error) {
   126  	return cli.client.Echo(ctx, types.ToRequestEcho(msg).GetEcho(), grpc.WaitForReady(true))
   127  }
   128  
   129  func (cli *grpcClient) Info(ctx context.Context, params *types.RequestInfo) (*types.ResponseInfo, error) {
   130  	return cli.client.Info(ctx, types.ToRequestInfo(params).GetInfo(), grpc.WaitForReady(true))
   131  }
   132  
   133  func (cli *grpcClient) CheckTx(ctx context.Context, params *types.RequestCheckTx) (*types.ResponseCheckTx, error) {
   134  	return cli.client.CheckTx(ctx, types.ToRequestCheckTx(params).GetCheckTx(), grpc.WaitForReady(true))
   135  }
   136  
   137  func (cli *grpcClient) Query(ctx context.Context, params *types.RequestQuery) (*types.ResponseQuery, error) {
   138  	return cli.client.Query(ctx, types.ToRequestQuery(params).GetQuery(), grpc.WaitForReady(true))
   139  }
   140  
   141  func (cli *grpcClient) Commit(ctx context.Context) (*types.ResponseCommit, error) {
   142  	return cli.client.Commit(ctx, types.ToRequestCommit().GetCommit(), grpc.WaitForReady(true))
   143  }
   144  
   145  func (cli *grpcClient) InitChain(ctx context.Context, params *types.RequestInitChain) (*types.ResponseInitChain, error) {
   146  	return cli.client.InitChain(ctx, types.ToRequestInitChain(params).GetInitChain(), grpc.WaitForReady(true))
   147  }
   148  
   149  func (cli *grpcClient) ListSnapshots(ctx context.Context, params *types.RequestListSnapshots) (*types.ResponseListSnapshots, error) {
   150  	return cli.client.ListSnapshots(ctx, types.ToRequestListSnapshots(params).GetListSnapshots(), grpc.WaitForReady(true))
   151  }
   152  
   153  func (cli *grpcClient) OfferSnapshot(ctx context.Context, params *types.RequestOfferSnapshot) (*types.ResponseOfferSnapshot, error) {
   154  	return cli.client.OfferSnapshot(ctx, types.ToRequestOfferSnapshot(params).GetOfferSnapshot(), grpc.WaitForReady(true))
   155  }
   156  
   157  func (cli *grpcClient) LoadSnapshotChunk(ctx context.Context, params *types.RequestLoadSnapshotChunk) (*types.ResponseLoadSnapshotChunk, error) {
   158  	return cli.client.LoadSnapshotChunk(ctx, types.ToRequestLoadSnapshotChunk(params).GetLoadSnapshotChunk(), grpc.WaitForReady(true))
   159  }
   160  
   161  func (cli *grpcClient) ApplySnapshotChunk(ctx context.Context, params *types.RequestApplySnapshotChunk) (*types.ResponseApplySnapshotChunk, error) {
   162  	return cli.client.ApplySnapshotChunk(ctx, types.ToRequestApplySnapshotChunk(params).GetApplySnapshotChunk(), grpc.WaitForReady(true))
   163  }
   164  
   165  func (cli *grpcClient) PrepareProposal(ctx context.Context, params *types.RequestPrepareProposal) (*types.ResponsePrepareProposal, error) {
   166  	return cli.client.PrepareProposal(ctx, types.ToRequestPrepareProposal(params).GetPrepareProposal(), grpc.WaitForReady(true))
   167  }
   168  
   169  func (cli *grpcClient) ProcessProposal(ctx context.Context, params *types.RequestProcessProposal) (*types.ResponseProcessProposal, error) {
   170  	return cli.client.ProcessProposal(ctx, types.ToRequestProcessProposal(params).GetProcessProposal(), grpc.WaitForReady(true))
   171  }
   172  
   173  func (cli *grpcClient) ExtendVote(ctx context.Context, params *types.RequestExtendVote) (*types.ResponseExtendVote, error) {
   174  	return cli.client.ExtendVote(ctx, types.ToRequestExtendVote(params).GetExtendVote(), grpc.WaitForReady(true))
   175  }
   176  
   177  func (cli *grpcClient) VerifyVoteExtension(ctx context.Context, params *types.RequestVerifyVoteExtension) (*types.ResponseVerifyVoteExtension, error) {
   178  	return cli.client.VerifyVoteExtension(ctx, types.ToRequestVerifyVoteExtension(params).GetVerifyVoteExtension(), grpc.WaitForReady(true))
   179  }
   180  
   181  func (cli *grpcClient) FinalizeBlock(ctx context.Context, params *types.RequestFinalizeBlock) (*types.ResponseFinalizeBlock, error) {
   182  	return cli.client.FinalizeBlock(ctx, types.ToRequestFinalizeBlock(params).GetFinalizeBlock(), grpc.WaitForReady(true))
   183  }
   184  
   185  func (cli *grpcClient) LoadLatest(ctx context.Context, params *types.RequestLoadLatest) (*types.ResponseLoadLatest, error) {
   186  	return cli.client.LoadLatest(ctx, types.ToRequestLoadLatest(params).GetLoadLatest(), grpc.WaitForReady(true))
   187  }