cosmos-sdk/app/app.go

221 lines
5.2 KiB
Go
Raw Normal View History

package app
2016-02-05 23:16:33 -08:00
import (
"fmt"
"strings"
2017-01-14 20:42:47 -08:00
abci "github.com/tendermint/abci/types"
2017-07-03 08:32:01 -07:00
"github.com/tendermint/basecoin"
2016-02-05 23:16:33 -08:00
eyes "github.com/tendermint/merkleeyes/client"
2017-06-21 09:15:11 -07:00
cmn "github.com/tendermint/tmlibs/common"
2017-05-01 07:03:54 -07:00
"github.com/tendermint/tmlibs/log"
2017-07-03 08:32:01 -07:00
"github.com/tendermint/basecoin/errors"
"github.com/tendermint/basecoin/modules/auth"
"github.com/tendermint/basecoin/modules/base"
2017-07-03 08:32:01 -07:00
"github.com/tendermint/basecoin/modules/coin"
"github.com/tendermint/basecoin/stack"
sm "github.com/tendermint/basecoin/state"
"github.com/tendermint/basecoin/version"
2016-02-05 23:16:33 -08:00
)
2017-07-04 20:28:27 -07:00
//nolint
2016-03-27 12:47:50 -07:00
const (
ModuleNameBase = "base"
ChainKey = "chain_id"
2016-03-27 12:47:50 -07:00
)
2016-02-05 23:16:33 -08:00
2017-07-04 20:28:27 -07:00
// Basecoin - The ABCI application
2016-02-16 12:29:54 -08:00
type Basecoin struct {
2016-05-01 13:52:08 -07:00
eyesCli *eyes.Client
state *sm.State
cacheState *sm.State
2017-07-03 08:32:01 -07:00
handler basecoin.Handler
height uint64
2017-05-01 07:03:54 -07:00
logger log.Logger
2016-02-05 23:16:33 -08:00
}
2017-07-06 22:27:29 -07:00
var _ abci.Application = &Basecoin{}
2017-07-04 20:28:27 -07:00
// NewBasecoin - create a new instance of the basecoin application
func NewBasecoin(handler basecoin.Handler, eyesCli *eyes.Client, logger log.Logger) *Basecoin {
state := sm.NewState(eyesCli, logger.With("module", "state"))
2016-02-16 12:29:54 -08:00
return &Basecoin{
2017-07-04 20:28:27 -07:00
handler: handler,
2016-05-01 13:52:08 -07:00
eyesCli: eyesCli,
state: state,
cacheState: nil,
height: 0,
2017-07-04 20:28:27 -07:00
logger: logger,
2016-02-05 23:16:33 -08:00
}
}
2017-07-04 20:28:27 -07:00
// DefaultHandler - placeholder to just handle sendtx
2017-07-03 08:32:01 -07:00
func DefaultHandler() basecoin.Handler {
// use the default stack
h := coin.NewHandler()
d := stack.NewDispatcher(stack.WrapHandler(h))
return stack.New(
base.Logger{},
stack.Recovery{},
auth.Signatures{},
base.Chain{},
).Use(d)
2017-07-03 08:32:01 -07:00
}
2017-07-04 20:28:27 -07:00
// GetState - XXX For testing, not thread safe!
2017-02-03 13:06:12 -08:00
func (app *Basecoin) GetState() *sm.State {
return app.state.CacheWrap()
}
2017-07-04 20:28:27 -07:00
// Info - ABCI
2017-01-14 20:42:47 -08:00
func (app *Basecoin) Info() abci.ResponseInfo {
2017-06-21 09:15:11 -07:00
resp, err := app.eyesCli.InfoSync()
if err != nil {
cmn.PanicCrisis(err)
}
app.height = resp.LastBlockHeight
2017-06-21 09:15:11 -07:00
return abci.ResponseInfo{
2017-07-04 20:28:27 -07:00
Data: fmt.Sprintf("Basecoin v%v", version.Version),
2017-06-21 09:15:11 -07:00
LastBlockHeight: resp.LastBlockHeight,
LastBlockAppHash: resp.LastBlockAppHash,
}
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// SetOption - ABCI
2017-02-13 14:04:49 -08:00
func (app *Basecoin) SetOption(key string, value string) string {
2017-07-04 20:28:27 -07:00
module, key := splitKey(key)
if module == ModuleNameBase {
2017-07-04 20:28:27 -07:00
if key == ChainKey {
app.state.SetChainID(value)
return "Success"
}
return fmt.Sprintf("Error: unknown base option: %s", key)
2017-07-03 09:58:28 -07:00
}
2017-07-04 20:28:27 -07:00
log, err := app.handler.SetOption(app.logger, app.state, module, key, value)
2017-07-03 09:58:28 -07:00
if err == nil {
return log
}
return "Error: " + err.Error()
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// DeliverTx - ABCI
2017-07-03 08:32:01 -07:00
func (app *Basecoin) DeliverTx(txBytes []byte) abci.Result {
tx, err := basecoin.LoadTx(txBytes)
2016-02-05 23:16:33 -08:00
if err != nil {
2017-07-03 08:32:01 -07:00
return errors.Result(err)
2016-02-05 23:16:33 -08:00
}
2017-01-12 12:25:04 -08:00
2017-07-03 08:32:01 -07:00
// TODO: can we abstract this setup and commit logic??
cache := app.state.CacheWrap()
2017-07-03 12:34:08 -07:00
ctx := stack.NewContext(
app.state.GetChainID(),
app.height,
2017-07-03 12:34:08 -07:00
app.logger.With("call", "delivertx"),
)
2017-07-03 08:32:01 -07:00
res, err := app.handler.DeliverTx(ctx, cache, tx)
if err != nil {
// discard the cache...
return errors.Result(err)
2016-02-05 23:16:33 -08:00
}
2017-07-03 08:32:01 -07:00
// commit the cache and return result
cache.CacheSync()
return res.ToABCI()
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// CheckTx - ABCI
2017-07-03 08:32:01 -07:00
func (app *Basecoin) CheckTx(txBytes []byte) abci.Result {
tx, err := basecoin.LoadTx(txBytes)
2016-02-05 23:16:33 -08:00
if err != nil {
2017-07-03 08:32:01 -07:00
return errors.Result(err)
2016-02-05 23:16:33 -08:00
}
2017-01-12 12:25:04 -08:00
2017-07-03 08:32:01 -07:00
// TODO: can we abstract this setup and commit logic??
2017-07-03 12:34:08 -07:00
ctx := stack.NewContext(
app.state.GetChainID(),
app.height,
2017-07-03 12:34:08 -07:00
app.logger.With("call", "checktx"),
)
2017-07-03 08:32:01 -07:00
// checktx generally shouldn't touch the state, but we don't care
// here on the framework level, since the cacheState is thrown away next block
res, err := app.handler.CheckTx(ctx, app.cacheState, tx)
if err != nil {
return errors.Result(err)
2016-02-05 23:16:33 -08:00
}
2017-07-03 08:32:01 -07:00
return res.ToABCI()
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// Query - ABCI
2017-01-28 09:29:32 -08:00
func (app *Basecoin) Query(reqQuery abci.RequestQuery) (resQuery abci.ResponseQuery) {
if len(reqQuery.Data) == 0 {
resQuery.Log = "Query cannot be zero length"
resQuery.Code = abci.CodeType_EncodingError
return
2016-03-27 21:51:04 -07:00
}
2017-01-12 12:25:04 -08:00
2017-01-28 09:29:32 -08:00
resQuery, err := app.eyesCli.QuerySync(reqQuery)
if err != nil {
resQuery.Log = "Failed to query MerkleEyes: " + err.Error()
resQuery.Code = abci.CodeType_InternalError
return
}
return
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// Commit - ABCI
2017-01-14 20:42:47 -08:00
func (app *Basecoin) Commit() (res abci.Result) {
2017-01-12 17:49:51 -08:00
// Commit state
res = app.state.Commit()
2017-01-12 12:25:04 -08:00
// Wrap the committed state in cache for CheckTx
app.cacheState = app.state.CacheWrap()
if res.IsErr() {
2017-06-21 09:15:11 -07:00
cmn.PanicSanity("Error getting hash: " + res.Error())
2016-02-05 23:16:33 -08:00
}
return res
2016-02-05 23:16:33 -08:00
}
2017-07-04 20:28:27 -07:00
// InitChain - ABCI
2017-01-14 20:42:47 -08:00
func (app *Basecoin) InitChain(validators []*abci.Validator) {
2017-07-03 08:32:01 -07:00
// for _, plugin := range app.plugins.GetList() {
// plugin.InitChain(app.state, validators)
// }
2016-03-15 15:01:53 -07:00
}
2017-07-04 20:28:27 -07:00
// BeginBlock - ABCI
func (app *Basecoin) BeginBlock(hash []byte, header *abci.Header) {
app.height++
2017-07-03 08:32:01 -07:00
// for _, plugin := range app.plugins.GetList() {
// plugin.BeginBlock(app.state, hash, header)
// }
2016-03-29 14:25:17 -07:00
}
2017-07-04 20:28:27 -07:00
// EndBlock - ABCI
func (app *Basecoin) EndBlock(height uint64) (res abci.ResponseEndBlock) {
2017-07-03 08:32:01 -07:00
// for _, plugin := range app.plugins.GetList() {
// pluginRes := plugin.EndBlock(app.state, height)
// res.Diffs = append(res.Diffs, pluginRes.Diffs...)
// }
2016-04-18 08:09:19 -07:00
return
2016-03-15 15:01:53 -07:00
}
2017-07-04 20:28:27 -07:00
//TODO move split key to tmlibs?
// Splits the string at the first '/'.
// if there are none, assign default module ("base").
func splitKey(key string) (string, string) {
if strings.Contains(key, "/") {
keyParts := strings.SplitN(key, "/", 2)
return keyParts[0], keyParts[1]
}
return ModuleNameBase, key
}