mirror of
https://github.com/fluencelabs/tendermint
synced 2025-05-29 06:01:21 +00:00
BREAKING Example response: ```json { "jsonrpc": "2.0", "id": "", "result": { "height": "2109", "txs_results": null, "validator_updates": null, "consensus_param_updates": null } } ``` Old result consisted of ABCIResponses struct and height. Exposing internal ABCI structures (which we store in state package) in RPC seems bad to me for the following reasons: 1) high risk of breaking the API when somebody changes internal structs (HAPPENED HERE!) 2) RPC is aware of ABCI, which I'm not sure we want
431 lines
12 KiB
Go
431 lines
12 KiB
Go
package core
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
cmn "github.com/tendermint/tendermint/libs/common"
|
|
ctypes "github.com/tendermint/tendermint/rpc/core/types"
|
|
rpctypes "github.com/tendermint/tendermint/rpc/lib/types"
|
|
sm "github.com/tendermint/tendermint/state"
|
|
"github.com/tendermint/tendermint/types"
|
|
)
|
|
|
|
// Get block headers for minHeight <= height <= maxHeight.
|
|
// Block headers are returned in descending order (highest first).
|
|
//
|
|
// ```shell
|
|
// curl 'localhost:26657/blockchain?minHeight=10&maxHeight=10'
|
|
// ```
|
|
//
|
|
// ```go
|
|
// client := client.NewHTTP("tcp://0.0.0.0:26657", "/websocket")
|
|
// err := client.Start()
|
|
// if err != nil {
|
|
// // handle error
|
|
// }
|
|
// defer client.Stop()
|
|
// info, err := client.BlockchainInfo(10, 10)
|
|
// ```
|
|
//
|
|
// > The above command returns JSON structured like this:
|
|
//
|
|
// ```json
|
|
// {
|
|
// "error": "",
|
|
// "result": {
|
|
// "block_metas": [
|
|
// {
|
|
// "header": {
|
|
// "app_hash": "",
|
|
// "chain_id": "test-chain-6UTNIN",
|
|
// "height": "10",
|
|
// "time": "2017-05-29T15:05:53.877Z",
|
|
// "num_txs": "0",
|
|
// "last_block_id": {
|
|
// "parts": {
|
|
// "hash": "3C78F00658E06744A88F24FF97A0A5011139F34A",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "F70588DAB36BDA5A953D548A16F7D48C6C2DFD78"
|
|
// },
|
|
// "last_commit_hash": "F31CC4282E50B3F2A58D763D233D76F26D26CABE",
|
|
// "data_hash": "",
|
|
// "validators_hash": "9365FC80F234C967BD233F5A3E2AB2F1E4B0E5AA"
|
|
// },
|
|
// "block_id": {
|
|
// "parts": {
|
|
// "hash": "277A4DBEF91483A18B85F2F5677ABF9694DFA40F",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "96B1D2F2D201BA4BC383EB8224139DB1294944E5"
|
|
// }
|
|
// }
|
|
// ],
|
|
// "last_height": "5493"
|
|
// },
|
|
// "id": "",
|
|
// "jsonrpc": "2.0"
|
|
// }
|
|
// ```
|
|
//
|
|
// <aside class="notice">Returns at most 20 items.</aside>
|
|
func BlockchainInfo(ctx *rpctypes.Context, minHeight, maxHeight int64) (*ctypes.ResultBlockchainInfo, error) {
|
|
|
|
// maximum 20 block metas
|
|
const limit int64 = 20
|
|
var err error
|
|
minHeight, maxHeight, err = filterMinMax(blockStore.Height(), minHeight, maxHeight, limit)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
logger.Debug("BlockchainInfoHandler", "maxHeight", maxHeight, "minHeight", minHeight)
|
|
|
|
blockMetas := []*types.BlockMeta{}
|
|
for height := maxHeight; height >= minHeight; height-- {
|
|
blockMeta := blockStore.LoadBlockMeta(height)
|
|
blockMetas = append(blockMetas, blockMeta)
|
|
}
|
|
|
|
return &ctypes.ResultBlockchainInfo{
|
|
LastHeight: blockStore.Height(),
|
|
BlockMetas: blockMetas}, nil
|
|
}
|
|
|
|
// error if either min or max are negative or min < max
|
|
// if 0, use 1 for min, latest block height for max
|
|
// enforce limit.
|
|
// error if min > max
|
|
func filterMinMax(height, min, max, limit int64) (int64, int64, error) {
|
|
// filter negatives
|
|
if min < 0 || max < 0 {
|
|
return min, max, fmt.Errorf("heights must be non-negative")
|
|
}
|
|
|
|
// adjust for default values
|
|
if min == 0 {
|
|
min = 1
|
|
}
|
|
if max == 0 {
|
|
max = height
|
|
}
|
|
|
|
// limit max to the height
|
|
max = cmn.MinInt64(height, max)
|
|
|
|
// limit min to within `limit` of max
|
|
// so the total number of blocks returned will be `limit`
|
|
min = cmn.MaxInt64(min, max-limit+1)
|
|
|
|
if min > max {
|
|
return min, max, fmt.Errorf("min height %d can't be greater than max height %d", min, max)
|
|
}
|
|
return min, max, nil
|
|
}
|
|
|
|
// Get block at a given height.
|
|
// If no height is provided, it will fetch the latest block.
|
|
//
|
|
// ```shell
|
|
// curl 'localhost:26657/block?height=10'
|
|
// ```
|
|
//
|
|
// ```go
|
|
// client := client.NewHTTP("tcp://0.0.0.0:26657", "/websocket")
|
|
// err := client.Start()
|
|
// if err != nil {
|
|
// // handle error
|
|
// }
|
|
// defer client.Stop()
|
|
// info, err := client.Block(10)
|
|
// ```
|
|
//
|
|
// > The above command returns JSON structured like this:
|
|
//
|
|
// ```json
|
|
// {
|
|
// "error": "",
|
|
// "result": {
|
|
// "block": {
|
|
// "last_commit": {
|
|
// "precommits": [
|
|
// {
|
|
// "signature": {
|
|
// "data": "12C0D8893B8A38224488DC1DE6270DF76BB1A5E9DB1C68577706A6A97C6EC34FFD12339183D5CA8BC2F46148773823DE905B7F6F5862FD564038BB7AE03BF50D",
|
|
// "type": "ed25519"
|
|
// },
|
|
// "block_id": {
|
|
// "parts": {
|
|
// "hash": "3C78F00658E06744A88F24FF97A0A5011139F34A",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "F70588DAB36BDA5A953D548A16F7D48C6C2DFD78"
|
|
// },
|
|
// "type": "2",
|
|
// "round": "0",
|
|
// "height": "9",
|
|
// "validator_index": "0",
|
|
// "validator_address": "E89A51D60F68385E09E716D353373B11F8FACD62"
|
|
// }
|
|
// ],
|
|
// "blockID": {
|
|
// "parts": {
|
|
// "hash": "3C78F00658E06744A88F24FF97A0A5011139F34A",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "F70588DAB36BDA5A953D548A16F7D48C6C2DFD78"
|
|
// }
|
|
// },
|
|
// "data": {
|
|
// "txs": []
|
|
// },
|
|
// "header": {
|
|
// "app_hash": "",
|
|
// "chain_id": "test-chain-6UTNIN",
|
|
// "height": "10",
|
|
// "time": "2017-05-29T15:05:53.877Z",
|
|
// "num_txs": "0",
|
|
// "last_block_id": {
|
|
// "parts": {
|
|
// "hash": "3C78F00658E06744A88F24FF97A0A5011139F34A",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "F70588DAB36BDA5A953D548A16F7D48C6C2DFD78"
|
|
// },
|
|
// "last_commit_hash": "F31CC4282E50B3F2A58D763D233D76F26D26CABE",
|
|
// "data_hash": "",
|
|
// "validators_hash": "9365FC80F234C967BD233F5A3E2AB2F1E4B0E5AA"
|
|
// }
|
|
// },
|
|
// "block_meta": {
|
|
// "header": {
|
|
// "app_hash": "",
|
|
// "chain_id": "test-chain-6UTNIN",
|
|
// "height": "10",
|
|
// "time": "2017-05-29T15:05:53.877Z",
|
|
// "num_txs": "0",
|
|
// "last_block_id": {
|
|
// "parts": {
|
|
// "hash": "3C78F00658E06744A88F24FF97A0A5011139F34A",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "F70588DAB36BDA5A953D548A16F7D48C6C2DFD78"
|
|
// },
|
|
// "last_commit_hash": "F31CC4282E50B3F2A58D763D233D76F26D26CABE",
|
|
// "data_hash": "",
|
|
// "validators_hash": "9365FC80F234C967BD233F5A3E2AB2F1E4B0E5AA"
|
|
// },
|
|
// "block_id": {
|
|
// "parts": {
|
|
// "hash": "277A4DBEF91483A18B85F2F5677ABF9694DFA40F",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "96B1D2F2D201BA4BC383EB8224139DB1294944E5"
|
|
// }
|
|
// }
|
|
// },
|
|
// "id": "",
|
|
// "jsonrpc": "2.0"
|
|
// }
|
|
// ```
|
|
func Block(ctx *rpctypes.Context, heightPtr *int64) (*ctypes.ResultBlock, error) {
|
|
storeHeight := blockStore.Height()
|
|
height, err := getHeight(storeHeight, heightPtr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
blockMeta := blockStore.LoadBlockMeta(height)
|
|
block := blockStore.LoadBlock(height)
|
|
return &ctypes.ResultBlock{BlockMeta: blockMeta, Block: block}, nil
|
|
}
|
|
|
|
// Get block commit at a given height.
|
|
// If no height is provided, it will fetch the commit for the latest block.
|
|
//
|
|
// ```shell
|
|
// curl 'localhost:26657/commit?height=11'
|
|
// ```
|
|
//
|
|
// ```go
|
|
// client := client.NewHTTP("tcp://0.0.0.0:26657", "/websocket")
|
|
// err := client.Start()
|
|
// if err != nil {
|
|
// // handle error
|
|
// }
|
|
// defer client.Stop()
|
|
// info, err := client.Commit(11)
|
|
// ```
|
|
//
|
|
// > The above command returns JSON structured like this:
|
|
//
|
|
// ```json
|
|
// {
|
|
// "error": "",
|
|
// "result": {
|
|
// "canonical": true,
|
|
// "commit": {
|
|
// "precommits": [
|
|
// {
|
|
// "signature": {
|
|
// "data": "00970429FEC652E9E21D106A90AE8C5413759A7488775CEF4A3F44DC46C7F9D941070E4FBE9ED54DF247FA3983359A0C3A238D61DE55C75C9116D72ABC9CF50F",
|
|
// "type": "ed25519"
|
|
// },
|
|
// "block_id": {
|
|
// "parts": {
|
|
// "hash": "9E37CBF266BC044A779E09D81C456E653B89E006",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "CC6E861E31CA4334E9888381B4A9137D1458AB6A"
|
|
// },
|
|
// "type": "2",
|
|
// "round": "0",
|
|
// "height": "11",
|
|
// "validator_index": "0",
|
|
// "validator_address": "E89A51D60F68385E09E716D353373B11F8FACD62"
|
|
// }
|
|
// ],
|
|
// "blockID": {
|
|
// "parts": {
|
|
// "hash": "9E37CBF266BC044A779E09D81C456E653B89E006",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "CC6E861E31CA4334E9888381B4A9137D1458AB6A"
|
|
// }
|
|
// },
|
|
// "header": {
|
|
// "app_hash": "",
|
|
// "chain_id": "test-chain-6UTNIN",
|
|
// "height": "11",
|
|
// "time": "2017-05-29T15:05:54.893Z",
|
|
// "num_txs": "0",
|
|
// "last_block_id": {
|
|
// "parts": {
|
|
// "hash": "277A4DBEF91483A18B85F2F5677ABF9694DFA40F",
|
|
// "total": "1"
|
|
// },
|
|
// "hash": "96B1D2F2D201BA4BC383EB8224139DB1294944E5"
|
|
// },
|
|
// "last_commit_hash": "3CE0C9727CE524BA9CB7C91E28F08E2B94001087",
|
|
// "data_hash": "",
|
|
// "validators_hash": "9365FC80F234C967BD233F5A3E2AB2F1E4B0E5AA"
|
|
// }
|
|
// },
|
|
// "id": "",
|
|
// "jsonrpc": "2.0"
|
|
// }
|
|
// ```
|
|
func Commit(ctx *rpctypes.Context, heightPtr *int64) (*ctypes.ResultCommit, error) {
|
|
storeHeight := blockStore.Height()
|
|
height, err := getHeight(storeHeight, heightPtr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
header := blockStore.LoadBlockMeta(height).Header
|
|
|
|
// If the next block has not been committed yet,
|
|
// use a non-canonical commit
|
|
if height == storeHeight {
|
|
commit := blockStore.LoadSeenCommit(height)
|
|
return ctypes.NewResultCommit(&header, commit, false), nil
|
|
}
|
|
|
|
// Return the canonical commit (comes from the block at height+1)
|
|
commit := blockStore.LoadBlockCommit(height)
|
|
return ctypes.NewResultCommit(&header, commit, true), nil
|
|
}
|
|
|
|
// BlockResults gets ABCIResults at a given height.
|
|
// If no height is provided, it will fetch results for the latest block.
|
|
//
|
|
// Results are for the height of the block containing the txs.
|
|
// Thus response.results.deliver_tx[5] is the results of executing
|
|
// getBlock(h).Txs[5]
|
|
//
|
|
// ```shell
|
|
// curl 'localhost:26657/block_results?height=10'
|
|
// ```
|
|
//
|
|
// ```go
|
|
// client := client.NewHTTP("tcp://0.0.0.0:26657", "/websocket")
|
|
// err := client.Start()
|
|
// if err != nil {
|
|
// // handle error
|
|
// }
|
|
// defer client.Stop()
|
|
// info, err := client.BlockResults(10)
|
|
// ```
|
|
//
|
|
//
|
|
// > The above command returns JSON structured like this:
|
|
//
|
|
// ```json
|
|
// {
|
|
// "jsonrpc": "2.0",
|
|
// "id": "",
|
|
// "result": {
|
|
// "height": "437",
|
|
// "txs_results": [
|
|
// {
|
|
// "gas_wanted": 1,
|
|
// "gas_used": 1,
|
|
// "tags": [
|
|
// {
|
|
// "key": "YXBwLmNyZWF0b3I=",
|
|
// "value": "Q29zbW9zaGkgTmV0b3dva28="
|
|
// },
|
|
// ]
|
|
// },
|
|
// {
|
|
// "code": 1,
|
|
// "codespace": "ibc",
|
|
// "log": "not enough gas",
|
|
// "gas_wanted": 1,
|
|
// "gas_used": 2,
|
|
// "tags": [
|
|
// {
|
|
// "key": "YXBwLmNyZWF0b3I=",
|
|
// "value": "Q29zbW9zaGkgTmV0b3dva28="
|
|
// },
|
|
// ]
|
|
// },
|
|
// ],
|
|
// "validator_updates": null,
|
|
// "consensus_param_updates": null,
|
|
// }
|
|
//}
|
|
// ```
|
|
func BlockResults(ctx *rpctypes.Context, heightPtr *int64) (*ctypes.ResultBlockResults, error) {
|
|
storeHeight := blockStore.Height()
|
|
height, err := getHeight(storeHeight, heightPtr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
results, err := sm.LoadABCIResponses(stateDB, height)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &ctypes.ResultBlockResults{
|
|
Height: height,
|
|
TxsResults: results.DeliverTx,
|
|
ValidatorUpdates: results.EndBlock.ValidatorUpdates,
|
|
ConsensusParamUpdates: results.EndBlock.ConsensusParamUpdates,
|
|
}, nil
|
|
}
|
|
|
|
func getHeight(currentHeight int64, heightPtr *int64) (int64, error) {
|
|
if heightPtr != nil {
|
|
height := *heightPtr
|
|
if height <= 0 {
|
|
return 0, fmt.Errorf("Height must be greater than 0")
|
|
}
|
|
if height > currentHeight {
|
|
return 0, fmt.Errorf("Height must be less than or equal to the current blockchain height")
|
|
}
|
|
return height, nil
|
|
}
|
|
return currentHeight, nil
|
|
}
|