tendermint/abci/client/client.go

130 lines
3.5 KiB
Go
Raw Permalink Normal View History

2017-01-12 12:47:55 -08:00
package abcicli
2016-01-22 15:50:11 -08:00
import (
2016-05-18 15:30:38 -07:00
"fmt"
2016-03-24 10:19:48 -07:00
"sync"
2016-05-18 15:30:38 -07:00
2018-06-21 21:59:02 -07:00
"github.com/tendermint/tendermint/abci/types"
2018-07-01 19:36:49 -07:00
cmn "github.com/tendermint/tendermint/libs/common"
2016-01-22 15:50:11 -08:00
)
2017-11-27 23:47:51 -08:00
const (
dialRetryIntervalSeconds = 3
echoRetryIntervalSeconds = 1
)
// Client defines an interface for an ABCI client.
// All `Async` methods return a `ReqRes` object.
// All `Sync` methods return the appropriate protobuf ResponseXxx struct and an error.
// Note these are client errors, eg. ABCI socket connectivity issues.
// Application-related errors are reflected in response via ABCI error codes and logs.
2016-03-24 10:19:48 -07:00
type Client interface {
2017-01-23 20:26:17 -08:00
cmn.Service
2016-03-24 10:19:48 -07:00
SetResponseCallback(Callback)
Error() error
FlushAsync() *ReqRes
EchoAsync(msg string) *ReqRes
2017-09-22 08:10:39 -07:00
InfoAsync(types.RequestInfo) *ReqRes
SetOptionAsync(types.RequestSetOption) *ReqRes
2017-01-12 12:27:08 -08:00
DeliverTxAsync(tx []byte) *ReqRes
2016-03-24 10:19:48 -07:00
CheckTxAsync(tx []byte) *ReqRes
QueryAsync(types.RequestQuery) *ReqRes
2016-03-24 10:19:48 -07:00
CommitAsync() *ReqRes
InitChainAsync(types.RequestInitChain) *ReqRes
BeginBlockAsync(types.RequestBeginBlock) *ReqRes
EndBlockAsync(types.RequestEndBlock) *ReqRes
2016-03-24 10:19:48 -07:00
FlushSync() error
EchoSync(msg string) (*types.ResponseEcho, error)
InfoSync(types.RequestInfo) (*types.ResponseInfo, error)
SetOptionSync(types.RequestSetOption) (*types.ResponseSetOption, error)
DeliverTxSync(tx []byte) (*types.ResponseDeliverTx, error)
CheckTxSync(tx []byte) (*types.ResponseCheckTx, error)
QuerySync(types.RequestQuery) (*types.ResponseQuery, error)
CommitSync() (*types.ResponseCommit, error)
InitChainSync(types.RequestInitChain) (*types.ResponseInitChain, error)
BeginBlockSync(types.RequestBeginBlock) (*types.ResponseBeginBlock, error)
EndBlockSync(types.RequestEndBlock) (*types.ResponseEndBlock, error)
2016-02-28 19:19:29 -08:00
}
2016-01-22 15:50:11 -08:00
//----------------------------------------
// NewClient returns a new ABCI client of the specified transport type.
// It returns an error if the transport is not "socket" or "grpc"
2016-05-18 15:30:38 -07:00
func NewClient(addr, transport string, mustConnect bool) (client Client, err error) {
switch transport {
case "socket":
client = NewSocketClient(addr, mustConnect)
2016-05-18 15:30:38 -07:00
case "grpc":
client = NewGRPCClient(addr, mustConnect)
2016-05-18 15:30:38 -07:00
default:
2017-01-12 12:47:55 -08:00
err = fmt.Errorf("Unknown abci transport %s", transport)
2016-05-18 15:30:38 -07:00
}
return
}
//----------------------------------------
2016-03-24 10:19:48 -07:00
type Callback func(*types.Request, *types.Response)
2016-01-22 15:50:11 -08:00
//----------------------------------------
2016-02-08 00:50:00 -08:00
type ReqRes struct {
2016-01-30 19:36:33 -08:00
*types.Request
2016-01-22 15:50:11 -08:00
*sync.WaitGroup
2016-01-30 19:36:33 -08:00
*types.Response // Not set atomically, so be sure to use WaitGroup.
2016-02-08 00:50:00 -08:00
mtx sync.Mutex
done bool // Gets set to true once *after* WaitGroup.Done().
cb func(*types.Response) // A single callback that may be set.
2016-01-22 15:50:11 -08:00
}
2016-02-26 23:02:34 -08:00
func NewReqRes(req *types.Request) *ReqRes {
2016-02-08 00:50:00 -08:00
return &ReqRes{
2016-01-22 15:50:11 -08:00
Request: req,
WaitGroup: waitGroup1(),
Response: nil,
2016-02-08 00:50:00 -08:00
done: false,
cb: nil,
2016-01-22 15:50:11 -08:00
}
}
2016-02-08 00:50:00 -08:00
// Sets the callback for this ReqRes atomically.
// If reqRes is already done, calls cb immediately.
// NOTE: reqRes.cb should not change if reqRes.done.
// NOTE: only one callback is supported.
func (reqRes *ReqRes) SetCallback(cb func(res *types.Response)) {
reqRes.mtx.Lock()
if reqRes.done {
reqRes.mtx.Unlock()
cb(reqRes.Response)
return
}
defer reqRes.mtx.Unlock()
reqRes.cb = cb
}
func (reqRes *ReqRes) GetCallback() func(*types.Response) {
reqRes.mtx.Lock()
defer reqRes.mtx.Unlock()
return reqRes.cb
}
// NOTE: it should be safe to read reqRes.cb without locks after this.
func (reqRes *ReqRes) SetDone() {
reqRes.mtx.Lock()
reqRes.done = true
reqRes.mtx.Unlock()
}
2016-01-22 15:50:11 -08:00
func waitGroup1() (wg *sync.WaitGroup) {
wg = &sync.WaitGroup{}
wg.Add(1)
return
}