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 }