Newer
Older
// Copyright 2017 Monax Industries Limited
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package rpc
import (
"fmt"
"time"
"encoding/json"
acm "github.com/hyperledger/burrow/account"
"github.com/hyperledger/burrow/account/state"
"github.com/hyperledger/burrow/binary"
bcm "github.com/hyperledger/burrow/blockchain"
"github.com/hyperledger/burrow/consensus/tendermint/query"
"github.com/hyperledger/burrow/event"
"github.com/hyperledger/burrow/execution"
"github.com/hyperledger/burrow/execution/names"
"github.com/hyperledger/burrow/keys"
"github.com/hyperledger/burrow/logging"
"github.com/hyperledger/burrow/logging/structure"
"github.com/hyperledger/burrow/permission"
"github.com/hyperledger/burrow/project"
"github.com/hyperledger/burrow/txs"
tm_types "github.com/tendermint/tendermint/types"
"github.com/tmthrgd/go-hex"
)
// Magic! Should probably be configurable, but not shouldn't be so huge we
// end up DoSing ourselves.
const MaxBlockLookback = 1000
const AccountsRingMutexCount = 100
// Base service that provides implementation for all underlying RPC methods
type Service struct {
ctx context.Context
state state.IterableReader
nameReg names.IterableReader
mempoolAccounts *execution.Accounts
subscribable event.Subscribable
blockchain bcm.BlockchainInfo
transactor *execution.Transactor
logger *logging.Logger
}
func NewService(ctx context.Context, state state.IterableReader, nameReg names.IterableReader,
checker state.Reader, subscribable event.Subscribable, blockchain bcm.BlockchainInfo, keyClient keys.KeyClient,
transactor *execution.Transactor, nodeView *query.NodeView, logger *logging.Logger) *Service {
state: state,
mempoolAccounts: execution.NewAccounts(checker, keyClient, AccountsRingMutexCount),
nameReg: nameReg,
subscribable: subscribable,
blockchain: blockchain,
transactor: transactor,
nodeView: nodeView,
logger: logger.With(structure.ComponentKey, "Service"),
}
}
// Provides a sub-service with only the subscriptions methods
func NewSubscribableService(subscribable event.Subscribable, logger *logging.Logger) *Service {
ctx: context.Background(),
subscribable: subscribable,
logger: logger.With(structure.ComponentKey, "Service"),
}
}
// Get a Transactor providing methods for delegating signing and the core BroadcastTx function for publishing
// transactions to the network
func (s *Service) Transactor() *execution.Transactor {
return s.transactor
}
// By providing certain methods on the Transactor (such as Transact, Send, etc) with the (non-final) MempoolAccounts
// rather than the committed (final) Accounts state the transactor can assign a sequence number based on all of the txs
// it has seen since the last block - provided these transactions are successfully committed (via DeliverTx) then
// subsequent transactions will have valid sequence numbers. This allows Burrow to coordinate sequencing and signing
// for a key it holds or is provided - it is down to the key-holder to manage the mutual information between transactions
// concurrent within a new block window.
// Get pending account state residing in the mempool
func (s *Service) MempoolAccounts() *execution.Accounts {
return s.mempoolAccounts
func (s *Service) State() state.Reader {
return s.state
func (s *Service) BlockchainInfo() bcm.BlockchainInfo {
return s.blockchain
}
func (s *Service) ListUnconfirmedTxs(maxTxs int) (*ResultListUnconfirmedTxs, error) {
// Get all transactions for now
transactions, err := s.nodeView.MempoolTransactions(maxTxs)
if err != nil {
return nil, err
}
wrappedTxs := make([]*txs.Envelope, len(transactions))
for i, tx := range transactions {
}
return &ResultListUnconfirmedTxs{
NumTxs: len(transactions),
Txs: wrappedTxs,
}, nil
}
func (s *Service) Subscribe(ctx context.Context, subscriptionID string, eventID string,
callback func(resultEvent *ResultEvent) (stop bool)) error {
queryBuilder := event.QueryForEventID(eventID)
"query", queryBuilder.String(),
"subscription_id", subscriptionID,
"event_id", eventID)
return event.SubscribeCallback(ctx, s.subscribable, subscriptionID, queryBuilder,
func(message interface{}) (stop bool) {
resultEvent, err := NewResultEvent(eventID, message)
if err != nil {
s.logger.InfoMsg("Received event that could not be mapped to ResultEvent",
structure.ErrorKey, err,
"subscription_id", subscriptionID,
"event_id", eventID)
func (s *Service) Unsubscribe(ctx context.Context, subscriptionID string) error {
"subscription_id", subscriptionID)
err := s.subscribable.UnsubscribeAll(ctx, subscriptionID)
if err != nil {
return fmt.Errorf("error unsubscribing from event with subscriptionID '%s': %v", subscriptionID, err)
}
return nil
}
func (s *Service) Status() (*ResultStatus, error) {
latestHeight := s.blockchain.LastBlockHeight()
var (
latestBlockMeta *tm_types.BlockMeta
latestBlockHash []byte
latestBlockTime int64
)
if latestHeight != 0 {
latestBlockMeta = s.nodeView.BlockStore().LoadBlockMeta(int64(latestHeight))
latestBlockHash = latestBlockMeta.Header.Hash()
latestBlockTime = latestBlockMeta.Header.Time.UnixNano()
}
publicKey, err := s.nodeView.PrivValidatorPublicKey()
if err != nil {
return nil, err
}
return &ResultStatus{
NodeInfo: s.nodeView.NodeInfo(),
GenesisHash: s.blockchain.GenesisHash(),
PubKey: publicKey,
LatestBlockHash: latestBlockHash,
LatestBlockHeight: latestHeight,
LatestBlockTime: latestBlockTime,
NodeVersion: project.History.CurrentVersion().String(),
}
func (s *Service) ChainIdentifiers() (*ResultChainId, error) {
return &ResultChainId{
ChainName: s.blockchain.GenesisDoc().ChainName,
ChainId: s.blockchain.ChainID(),
GenesisHash: s.blockchain.GenesisHash(),
}, nil
}
func (s *Service) Peers() (*ResultPeers, error) {
peers := make([]*Peer, s.nodeView.Peers().Size())
for i, peer := range s.nodeView.Peers().List() {
peers[i] = &Peer{
NodeInfo: peer.NodeInfo(),
IsOutbound: peer.IsOutbound(),
}
}
return &ResultPeers{
Peers: peers,
}, nil
}
func (s *Service) NetInfo() (*ResultNetInfo, error) {
listening := s.nodeView.IsListening()
for _, listener := range s.nodeView.Listeners() {
listeners = append(listeners, listener.String())
}
peers, err := s.Peers()
if err != nil {
return nil, err
}
return &ResultNetInfo{
Listening: listening,
Listeners: listeners,
Peers: peers.Peers,
}, nil
}
func (s *Service) Genesis() (*ResultGenesis, error) {
return &ResultGenesis{
Genesis: s.blockchain.GenesisDoc(),
}, nil
}
// Accounts
func (s *Service) GetAccount(address crypto.Address) (*ResultGetAccount, error) {
acc, err := s.state.GetAccount(address)
if err != nil {
return nil, err
}
return &ResultGetAccount{Account: acm.AsConcreteAccount(acc)}, nil
}
func (s *Service) ListAccounts(predicate func(acm.Account) bool) (*ResultListAccounts, error) {
accounts := make([]*acm.ConcreteAccount, 0)
s.state.IterateAccounts(func(account acm.Account) (stop bool) {
if predicate(account) {
accounts = append(accounts, acm.AsConcreteAccount(account))
}
return
})
return &ResultListAccounts{
BlockHeight: s.blockchain.LastBlockHeight(),
Accounts: accounts,
}, nil
}
func (s *Service) GetStorage(address crypto.Address, key []byte) (*ResultGetStorage, error) {
account, err := s.state.GetAccount(address)
if err != nil {
return nil, err
}
if account == nil {
return nil, fmt.Errorf("UnknownAddress: %s", address)
}
value, err := s.state.GetStorage(address, binary.LeftPadWord256(key))
if err != nil {
return nil, err
}
if value == binary.Zero256 {
return &ResultGetStorage{Key: key, Value: nil}, nil
}
return &ResultGetStorage{Key: key, Value: value.UnpadLeft()}, nil
}
func (s *Service) DumpStorage(address crypto.Address) (*ResultDumpStorage, error) {
account, err := s.state.GetAccount(address)
if err != nil {
return nil, err
}
if account == nil {
return nil, fmt.Errorf("UnknownAddress: %X", address)
}
var storageItems []StorageItem
s.state.IterateStorage(address, func(key, value binary.Word256) (stop bool) {
storageItems = append(storageItems, StorageItem{Key: key.UnpadLeft(), Value: value.UnpadLeft()})
return
})
return &ResultDumpStorage{
StorageRoot: account.StorageRoot(),
StorageItems: storageItems,
}, nil
}
func (s *Service) GetAccountHumanReadable(address crypto.Address) (*ResultGetAccountHumanReadable, error) {
acc, err := s.state.GetAccount(address)
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
if err != nil {
return nil, err
}
if acc == nil {
return &ResultGetAccountHumanReadable{}, nil
}
tokens, err := acc.Code().Tokens()
if acc == nil {
return &ResultGetAccountHumanReadable{}, nil
}
perms, err := permission.BasePermissionsToStringList(acc.Permissions().Base)
if acc == nil {
return &ResultGetAccountHumanReadable{}, nil
}
return &ResultGetAccountHumanReadable{
Account: &AccountHumanReadable{
Address: acc.Address(),
PublicKey: acc.PublicKey(),
Sequence: acc.Sequence(),
Balance: acc.Balance(),
Code: tokens,
StorageRoot: hex.EncodeUpperToString(acc.StorageRoot()),
Permissions: perms,
Roles: acc.Permissions().Roles,
},
}, nil
}
// Name registry
func (s *Service) GetName(name string) (*ResultGetName, error) {
entry, err := s.nameReg.GetNameEntry(name)
if err != nil {
return nil, err
}
if entry == nil {
return nil, fmt.Errorf("name %s not found", name)
}
return &ResultGetName{Entry: entry}, nil
}
func (s *Service) ListNames(predicate func(*names.Entry) bool) (*ResultListNames, error) {
var nms []*names.Entry
s.nameReg.IterateNameEntries(func(entry *names.Entry) (stop bool) {
if predicate(entry) {
}
return
})
return &ResultListNames{
BlockHeight: s.blockchain.LastBlockHeight(),
}, nil
}
func (s *Service) GetBlock(height uint64) (*ResultGetBlock, error) {
return &ResultGetBlock{
Block: &Block{s.nodeView.BlockStore().LoadBlock(int64(height))},
BlockMeta: &BlockMeta{s.nodeView.BlockStore().LoadBlockMeta(int64(height))},
}, nil
}
// Returns the current blockchain height and metadata for a range of blocks
// between minHeight and maxHeight. Only returns maxBlockLookback block metadata
// from the top of the range of blocks.
// Passing 0 for maxHeight sets the upper height of the range to the current
// blockchain height.
func (s *Service) ListBlocks(minHeight, maxHeight uint64) (*ResultListBlocks, error) {
latestHeight := s.blockchain.LastBlockHeight()
if minHeight == 0 {
minHeight = 1
}
if maxHeight == 0 || latestHeight < maxHeight {
maxHeight = latestHeight
}
if maxHeight > minHeight && maxHeight-minHeight > MaxBlockLookback {
minHeight = maxHeight - MaxBlockLookback
}
var blockMetas []*tm_types.BlockMeta
for height := maxHeight; height >= minHeight; height-- {
blockMeta := s.nodeView.BlockStore().LoadBlockMeta(int64(height))
blockMetas = append(blockMetas, blockMeta)
}
return &ResultListBlocks{
LastHeight: latestHeight,
BlockMetas: blockMetas,
}, nil
}
func (s *Service) ListValidators() (*ResultListValidators, error) {
concreteValidators := make([]*acm.ConcreteValidator, 0, s.blockchain.NumValidators())
s.blockchain.IterateValidators(func(publicKey crypto.PublicKey, power uint64) (stop bool) {
concreteValidators = append(concreteValidators, &acm.ConcreteValidator{
Address: publicKey.Address(),
PublicKey: publicKey,
Power: power,
})
return
})
return &ResultListValidators{
BlockHeight: s.blockchain.LastBlockHeight(),
BondedValidators: concreteValidators,
UnbondingValidators: nil,
}, nil
}
func (s *Service) DumpConsensusState() (*ResultDumpConsensusState, error) {
peerRoundState, err := s.nodeView.PeerRoundStates()
if err != nil {
return nil, err
}
return &ResultDumpConsensusState{
RoundState: s.nodeView.RoundState().RoundStateSimple(),
PeerRoundStates: peerRoundState,
}, nil
}
func (s *Service) GeneratePrivateAccount() (*ResultGeneratePrivateAccount, error) {
privateAccount, err := acm.GeneratePrivateAccount()
if err != nil {
return nil, err
}
return &ResultGeneratePrivateAccount{
PrivateAccount: acm.AsConcretePrivateAccount(privateAccount),
}, nil
}
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
func (s *Service) LastBlockInfo(blockWithin string) (*ResultLastBlockInfo, error) {
res := &ResultLastBlockInfo{
LastBlockHeight: s.blockchain.LastBlockHeight(),
LastBlockHash: s.blockchain.LastBlockHash(),
LastBlockTime: s.blockchain.LastBlockTime(),
}
if blockWithin == "" {
return res, nil
}
duration, err := time.ParseDuration(blockWithin)
if err != nil {
return nil, fmt.Errorf("could not parse blockWithin duration to determine whether to throw error: %v", err)
}
// Take neg abs in case caller is counting backwards (not we add later)
if duration > 0 {
duration = -duration
}
blockTimeThreshold := time.Now().Add(duration)
if res.LastBlockTime.After(blockTimeThreshold) {
// We've created blocks recently enough
return res, nil
}
resJSON, err := json.Marshal(res)
if err != nil {
resJSON = []byte("<error: could not marshal last block info>")
}
return nil, fmt.Errorf("no block committed within the last %s (cutoff: %s), last block info: %s",
blockWithin, blockTimeThreshold.Format(time.RFC3339), string(resJSON))
}