mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-06-19 21:31:37 +00:00
498 lines
16 KiB
Go
498 lines
16 KiB
Go
// Copyright (c) 2018 XDPoSChain
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package XDPoS
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"errors"
|
|
"math/big"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/XinFinOrg/XDPoSChain/common"
|
|
"github.com/XinFinOrg/XDPoSChain/consensus"
|
|
"github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils"
|
|
"github.com/XinFinOrg/XDPoSChain/core/types"
|
|
"github.com/XinFinOrg/XDPoSChain/log"
|
|
"github.com/XinFinOrg/XDPoSChain/params"
|
|
"github.com/XinFinOrg/XDPoSChain/rlp"
|
|
"github.com/XinFinOrg/XDPoSChain/rpc"
|
|
)
|
|
|
|
// API is a user facing RPC API to allow controlling the signer and voting
|
|
// mechanisms of the proof-of-authority scheme.
|
|
type API struct {
|
|
chain consensus.ChainReader
|
|
XDPoS *XDPoS
|
|
}
|
|
|
|
type V2BlockInfo struct {
|
|
Hash common.Hash
|
|
Round types.Round
|
|
Number *big.Int
|
|
ParentHash common.Hash
|
|
Committed bool
|
|
Miner common.Hash
|
|
Timestamp *big.Int
|
|
EncodedRLP string
|
|
Error string
|
|
}
|
|
|
|
type NetworkInformation struct {
|
|
NetworkId *big.Int
|
|
XDCValidatorAddress common.Address
|
|
RelayerRegistrationAddress common.Address
|
|
XDCXListingAddress common.Address
|
|
XDCZAddress common.Address
|
|
LendingAddress common.Address
|
|
ConsensusConfigs params.XDPoSConfig
|
|
}
|
|
|
|
type SignerTypes struct {
|
|
CurrentNumber int
|
|
CurrentSigners []common.Address
|
|
MissingSigners []common.Address
|
|
}
|
|
|
|
type MasternodesStatus struct {
|
|
Number uint64
|
|
Round types.Round
|
|
MasternodesLen int
|
|
Masternodes []common.Address
|
|
PenaltyLen int
|
|
Penalty []common.Address
|
|
StandbynodesLen int
|
|
Standbynodes []common.Address
|
|
Error error
|
|
}
|
|
|
|
type AccountEpochReward struct {
|
|
EpochBlockNum uint64
|
|
Address common.Address
|
|
AccountStatus AccountRewardStatus
|
|
AccountReward *big.Int
|
|
DelegatedReward map[string]*big.Int
|
|
}
|
|
|
|
type AccountRewardStatus string
|
|
|
|
const (
|
|
statusMasternode AccountRewardStatus = "MasterNode"
|
|
statusProtectornode AccountRewardStatus = "ProtectorNode"
|
|
statusObservernode AccountRewardStatus = "ObserverNode"
|
|
)
|
|
|
|
type MessageStatus map[string]map[string]SignerTypes
|
|
|
|
// GetSnapshot retrieves the state snapshot at a given block.
|
|
func (api *API) GetSnapshot(number *rpc.BlockNumber) (*utils.PublicApiSnapshot, error) {
|
|
// Retrieve the requested block number (or current if none requested)
|
|
var header *types.Header
|
|
if number == nil || *number == rpc.LatestBlockNumber {
|
|
header = api.chain.CurrentHeader()
|
|
} else {
|
|
header = api.chain.GetHeaderByNumber(uint64(number.Int64()))
|
|
}
|
|
// Ensure we have an actually valid block and return its snapshot
|
|
if header == nil {
|
|
return nil, utils.ErrUnknownBlock
|
|
}
|
|
return api.XDPoS.GetSnapshot(api.chain, header)
|
|
}
|
|
|
|
// GetSnapshotAtHash retrieves the state snapshot at a given block.
|
|
func (api *API) GetSnapshotAtHash(hash common.Hash) (*utils.PublicApiSnapshot, error) {
|
|
header := api.chain.GetHeaderByHash(hash)
|
|
if header == nil {
|
|
return nil, utils.ErrUnknownBlock
|
|
}
|
|
return api.XDPoS.GetSnapshot(api.chain, header)
|
|
}
|
|
|
|
// GetSigners retrieves the list of authorized signers at the specified block.
|
|
func (api *API) GetSigners(number *rpc.BlockNumber) ([]common.Address, error) {
|
|
// Retrieve the requested block number (or current if none requested)
|
|
var header *types.Header
|
|
if number == nil || *number == rpc.LatestBlockNumber {
|
|
header = api.chain.CurrentHeader()
|
|
} else {
|
|
header = api.chain.GetHeaderByNumber(uint64(number.Int64()))
|
|
}
|
|
// Ensure we have an actually valid block and return the signers from its snapshot
|
|
if header == nil {
|
|
return nil, utils.ErrUnknownBlock
|
|
}
|
|
|
|
return api.XDPoS.GetAuthorisedSignersFromSnapshot(api.chain, header)
|
|
}
|
|
|
|
// GetSignersAtHash retrieves the state snapshot at a given block.
|
|
func (api *API) GetSignersAtHash(hash common.Hash) ([]common.Address, error) {
|
|
header := api.chain.GetHeaderByHash(hash)
|
|
if header == nil {
|
|
return nil, utils.ErrUnknownBlock
|
|
}
|
|
return api.XDPoS.GetAuthorisedSignersFromSnapshot(api.chain, header)
|
|
}
|
|
|
|
func (api *API) GetMasternodesByNumber(number *rpc.BlockNumber) MasternodesStatus {
|
|
var header *types.Header
|
|
if number == nil || *number == rpc.LatestBlockNumber {
|
|
header = api.chain.CurrentHeader()
|
|
} else if *number == rpc.CommittedBlockNumber {
|
|
hash := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash
|
|
header = api.chain.GetHeaderByHash(hash)
|
|
} else {
|
|
header = api.chain.GetHeaderByNumber(uint64(number.Int64()))
|
|
}
|
|
|
|
round, err := api.XDPoS.EngineV2.GetRoundNumber(header)
|
|
if err != nil {
|
|
return MasternodesStatus{
|
|
Error: err,
|
|
}
|
|
}
|
|
|
|
masterNodes := api.XDPoS.EngineV2.GetMasternodes(api.chain, header)
|
|
penalties := api.XDPoS.EngineV2.GetPenalties(api.chain, header)
|
|
standbynodes := api.XDPoS.EngineV2.GetStandbynodes(api.chain, header)
|
|
|
|
info := MasternodesStatus{
|
|
Number: header.Number.Uint64(),
|
|
Round: round,
|
|
MasternodesLen: len(masterNodes),
|
|
Masternodes: masterNodes,
|
|
PenaltyLen: len(penalties),
|
|
Penalty: penalties,
|
|
StandbynodesLen: len(standbynodes),
|
|
Standbynodes: standbynodes,
|
|
}
|
|
return info
|
|
}
|
|
|
|
// Get current vote pool and timeout pool content and missing messages
|
|
func (api *API) GetLatestPoolStatus() MessageStatus {
|
|
header := api.chain.CurrentHeader()
|
|
masternodes := api.XDPoS.EngineV2.GetMasternodes(api.chain, header)
|
|
|
|
receivedVotes := api.XDPoS.EngineV2.ReceivedVotes()
|
|
receivedTimeouts := api.XDPoS.EngineV2.ReceivedTimeouts()
|
|
info := make(MessageStatus)
|
|
info["vote"] = make(map[string]SignerTypes)
|
|
info["timeout"] = make(map[string]SignerTypes)
|
|
|
|
calculateSigners(info["vote"], receivedVotes, masternodes)
|
|
calculateSigners(info["timeout"], receivedTimeouts, masternodes)
|
|
|
|
return info
|
|
}
|
|
|
|
func (api *API) GetV2BlockByHeader(header *types.Header, uncle bool) *V2BlockInfo {
|
|
committed := false
|
|
latestCommittedBlock := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo()
|
|
if latestCommittedBlock == nil {
|
|
return &V2BlockInfo{
|
|
Hash: header.Hash(),
|
|
Error: "can not find latest committed block from consensus",
|
|
}
|
|
}
|
|
if header.Number.Uint64() <= latestCommittedBlock.Number.Uint64() {
|
|
committed = true && !uncle
|
|
}
|
|
|
|
round, err := api.XDPoS.EngineV2.GetRoundNumber(header)
|
|
|
|
if err != nil {
|
|
return &V2BlockInfo{
|
|
Hash: header.Hash(),
|
|
Error: err.Error(),
|
|
}
|
|
}
|
|
|
|
encodeBytes, err := rlp.EncodeToBytes(header)
|
|
if err != nil {
|
|
return &V2BlockInfo{
|
|
Hash: header.Hash(),
|
|
Error: err.Error(),
|
|
}
|
|
}
|
|
|
|
block := &V2BlockInfo{
|
|
Hash: header.Hash(),
|
|
ParentHash: header.ParentHash,
|
|
Number: header.Number,
|
|
Round: round,
|
|
Committed: committed,
|
|
Miner: header.Coinbase.Hash(),
|
|
Timestamp: header.Time,
|
|
EncodedRLP: base64.StdEncoding.EncodeToString(encodeBytes),
|
|
}
|
|
return block
|
|
}
|
|
|
|
func (api *API) GetV2BlockByNumber(number *rpc.BlockNumber) *V2BlockInfo {
|
|
header := api.getHeaderFromApiBlockNum(number)
|
|
if header == nil {
|
|
return &V2BlockInfo{
|
|
Number: big.NewInt(number.Int64()),
|
|
Error: "can not find block from this number",
|
|
}
|
|
}
|
|
|
|
uncle := false
|
|
return api.GetV2BlockByHeader(header, uncle)
|
|
}
|
|
|
|
// Confirm V2 Block Committed Status
|
|
func (api *API) GetV2BlockByHash(blockHash common.Hash) *V2BlockInfo {
|
|
header := api.chain.GetHeaderByHash(blockHash)
|
|
if header == nil {
|
|
return &V2BlockInfo{
|
|
Hash: blockHash,
|
|
Error: "can not find block from this hash",
|
|
}
|
|
}
|
|
|
|
// confirm this is on the main chain
|
|
chainHeader := api.chain.GetHeaderByNumber(header.Number.Uint64())
|
|
uncle := false
|
|
if header.Hash() != chainHeader.Hash() {
|
|
uncle = true
|
|
}
|
|
|
|
return api.GetV2BlockByHeader(header, uncle)
|
|
}
|
|
|
|
func (api *API) NetworkInformation() NetworkInformation {
|
|
info := NetworkInformation{}
|
|
info.NetworkId = api.chain.Config().ChainId
|
|
info.XDCValidatorAddress = common.MasternodeVotingSMCBinary
|
|
info.LendingAddress = common.LendingRegistrationSMC
|
|
info.RelayerRegistrationAddress = common.RelayerRegistrationSMC
|
|
info.XDCXListingAddress = common.XDCXListingSMC
|
|
info.XDCZAddress = common.TRC21IssuerSMC
|
|
info.ConsensusConfigs = *api.XDPoS.config
|
|
|
|
return info
|
|
}
|
|
|
|
/*
|
|
An API exclusively for V2 consensus, designed to assist in troubleshooting miners by identifying who mined during their allocated term.
|
|
*/
|
|
func (api *API) GetMissedRoundsInEpochByBlockNum(number *rpc.BlockNumber) (*utils.PublicApiMissedRoundsMetadata, error) {
|
|
return api.XDPoS.CalculateMissingRounds(api.chain, api.getHeaderFromApiBlockNum(number))
|
|
}
|
|
|
|
func (api *API) getHeaderFromApiBlockNum(number *rpc.BlockNumber) *types.Header {
|
|
var header *types.Header
|
|
if number == nil || *number == rpc.LatestBlockNumber {
|
|
header = api.chain.CurrentHeader()
|
|
} else if *number == rpc.CommittedBlockNumber {
|
|
hash := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash
|
|
header = api.chain.GetHeaderByHash(hash)
|
|
} else {
|
|
header = api.chain.GetHeaderByNumber(uint64(number.Int64()))
|
|
}
|
|
return header
|
|
}
|
|
|
|
func calculateSigners(message map[string]SignerTypes, pool map[string]map[common.Hash]utils.PoolObj, masternodes []common.Address) {
|
|
for name, objs := range pool {
|
|
var currentSigners []common.Address
|
|
missingSigners := make([]common.Address, len(masternodes))
|
|
copy(missingSigners, masternodes)
|
|
|
|
num := len(objs)
|
|
for _, obj := range objs {
|
|
signer := obj.GetSigner()
|
|
currentSigners = append(currentSigners, signer)
|
|
for i, mn := range missingSigners {
|
|
if mn == signer {
|
|
missingSigners = append(missingSigners[:i], missingSigners[i+1:]...)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
message[name] = SignerTypes{
|
|
CurrentNumber: num,
|
|
CurrentSigners: currentSigners,
|
|
MissingSigners: missingSigners,
|
|
}
|
|
}
|
|
}
|
|
|
|
func (api *API) GetRewardByAccount(account common.Address, begin rpc.BlockNumber, end rpc.BlockNumber) ([]AccountEpochReward, error) {
|
|
epochBlocks, err := api.GetEpochNumbersBetween(&begin, &end)
|
|
if err != nil {
|
|
return []AccountEpochReward{}, err
|
|
}
|
|
epochRewards := []AccountEpochReward{}
|
|
for _, epochBlock := range epochBlocks {
|
|
header := api.chain.GetHeaderByNumber(epochBlock)
|
|
if header == nil {
|
|
log.Error("[GetRewardByAccount] header not found, impossible case, please check or report to XDC", "err", err)
|
|
return []AccountEpochReward{}, err
|
|
}
|
|
epochReward, err := getEpochReward(account, header)
|
|
if err != nil {
|
|
return []AccountEpochReward{}, err
|
|
}
|
|
epochRewards = append(epochRewards, epochReward)
|
|
}
|
|
|
|
return epochRewards, nil
|
|
}
|
|
|
|
func getEpochReward(account common.Address, header *types.Header) (AccountEpochReward, error) {
|
|
var data map[string]interface{}
|
|
path := filepath.Join(common.StoreRewardFolder, header.Number.String()+"."+header.Hash().Hex())
|
|
alternatePath := filepath.Join(common.StoreRewardFolder, header.Number.String()+"."+header.HashNoValidator().Hex())
|
|
file, err := os.Open(path)
|
|
if err != nil {
|
|
file, err := os.Open(alternatePath)
|
|
if err != nil {
|
|
log.Warn("[getEpochReward] rewards file not found", "path", path, "alternatePath", alternatePath)
|
|
return AccountEpochReward{}, err
|
|
}
|
|
defer file.Close()
|
|
decoder := json.NewDecoder(file)
|
|
decoder.UseNumber()
|
|
if err := decoder.Decode(&data); err != nil {
|
|
log.Warn("[getEpochReward] Failed to decode JSON:", "err", err)
|
|
return AccountEpochReward{}, err
|
|
}
|
|
} else {
|
|
defer file.Close()
|
|
decoder := json.NewDecoder(file)
|
|
decoder.UseNumber()
|
|
if err := decoder.Decode(&data); err != nil {
|
|
log.Warn("[getEpochReward] Failed to decode JSON:", "err", err)
|
|
return AccountEpochReward{}, err
|
|
}
|
|
}
|
|
|
|
epochReward := AccountEpochReward{
|
|
Address: account,
|
|
EpochBlockNum: header.Number.Uint64(),
|
|
DelegatedReward: make(map[string]*big.Int),
|
|
}
|
|
epochReward.getRewardAndStatus(strings.ToLower(account.String0x()), data)
|
|
|
|
return epochReward, nil
|
|
}
|
|
|
|
func (rewardObj *AccountEpochReward) getRewardAndStatus(account string, data map[string]interface{}) {
|
|
if val, exists := data["signers"]; exists {
|
|
if val, exists := val.(map[string]interface{})[account]; exists {
|
|
nodeReward := val.(map[string]interface{})["reward"]
|
|
delegatedReward := data["rewards"].(map[string]interface{})[account]
|
|
rewardObj.AccountStatus = statusMasternode
|
|
nodeRewardBigInt, _ := new(big.Int).SetString(nodeReward.(json.Number).String(), 10)
|
|
rewardObj.AccountReward = nodeRewardBigInt
|
|
|
|
for k, v := range delegatedReward.(map[string]interface{}) {
|
|
delegatedBigInt, _ := new(big.Int).SetString(v.(json.Number).String(), 10)
|
|
rewardObj.DelegatedReward[k] = delegatedBigInt
|
|
}
|
|
return
|
|
}
|
|
}
|
|
|
|
if val, exists := data["signersProtector"]; exists {
|
|
if val, exists := val.(map[string]interface{})[account]; exists {
|
|
nodeReward := val.(map[string]interface{})["reward"]
|
|
delegatedReward := data["rewardsProtector"].(map[string]interface{})[account]
|
|
rewardObj.AccountStatus = statusProtectornode
|
|
nodeRewardBigInt, _ := new(big.Int).SetString(nodeReward.(json.Number).String(), 10)
|
|
rewardObj.AccountReward = nodeRewardBigInt
|
|
|
|
for k, v := range delegatedReward.(map[string]interface{}) {
|
|
delegatedBigInt, _ := new(big.Int).SetString(v.(json.Number).String(), 10)
|
|
rewardObj.DelegatedReward[k] = delegatedBigInt
|
|
}
|
|
return
|
|
}
|
|
|
|
}
|
|
|
|
if val, exists := data["signersObserver"]; exists {
|
|
if val, exists := val.(map[string]interface{})[account]; exists {
|
|
nodeReward := val.(map[string]interface{})["reward"]
|
|
delegatedReward := data["rewardsObserver"].(map[string]interface{})[account]
|
|
rewardObj.AccountStatus = statusObservernode
|
|
nodeRewardBigInt, _ := new(big.Int).SetString(nodeReward.(json.Number).String(), 10)
|
|
rewardObj.AccountReward = nodeRewardBigInt
|
|
|
|
for k, v := range delegatedReward.(map[string]interface{}) {
|
|
delegatedBigInt, _ := new(big.Int).SetString(v.(json.Number).String(), 10)
|
|
rewardObj.DelegatedReward[k] = delegatedBigInt
|
|
}
|
|
return
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
func (api *API) GetEpochNumbersBetween(begin, end *rpc.BlockNumber) ([]uint64, error) {
|
|
beginHeader := api.getHeaderFromApiBlockNum(begin)
|
|
if beginHeader == nil {
|
|
return nil, errors.New("illegal begin block number")
|
|
}
|
|
endHeader := api.getHeaderFromApiBlockNum(end)
|
|
if endHeader == nil {
|
|
return nil, errors.New("illegal end block number")
|
|
}
|
|
diff := new(big.Int).Sub(endHeader.Number, beginHeader.Number).Int64()
|
|
if diff < 0 {
|
|
return nil, errors.New("illegal begin and end block number, begin > end")
|
|
}
|
|
if diff > 50_000 {
|
|
return nil, errors.New("block range over limit of 50,000 blocks")
|
|
}
|
|
epochSwitchInfos, err := api.XDPoS.GetEpochSwitchInfoBetween(api.chain, beginHeader, endHeader)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
epochSwitchNumbers := make([]uint64, len(epochSwitchInfos))
|
|
for i, info := range epochSwitchInfos {
|
|
epochSwitchNumbers[i] = info.EpochSwitchBlockInfo.Number.Uint64()
|
|
}
|
|
return epochSwitchNumbers, nil
|
|
}
|
|
|
|
/*
|
|
An API exclusively for V2 consensus, designed to assist in getting rewards of the epoch number.
|
|
Given the epoch number, search the epoch switch block.
|
|
*/
|
|
func (api *API) GetBlockInfoByEpochNum(epochNumber uint64) (*utils.EpochNumInfo, error) {
|
|
thisEpoch, err := api.XDPoS.EngineV2.GetBlockByEpochNumber(api.chain, epochNumber)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
info := &utils.EpochNumInfo{
|
|
EpochBlockHash: thisEpoch.Hash,
|
|
EpochRound: thisEpoch.Round,
|
|
EpochFirstBlockNumber: thisEpoch.Number,
|
|
}
|
|
nextEpoch, err := api.XDPoS.EngineV2.GetBlockByEpochNumber(api.chain, epochNumber+1)
|
|
|
|
if err == nil {
|
|
info.EpochLastBlockNumber = new(big.Int).Sub(nextEpoch.Number, big.NewInt(1))
|
|
}
|
|
return info, nil
|
|
}
|