2220 lines
73 KiB
Go
2220 lines
73 KiB
Go
// Copyright 2017-2021 DERO Project. All rights reserved.
|
|
// Use of this source code in any form is governed by RESEARCH license.
|
|
// license can be found in the LICENSE file.
|
|
// GPG: 0F39 E425 8C65 3947 702A 8234 08B2 0360 A03A 9DE8
|
|
//
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
package blockchain
|
|
|
|
// This file runs the core consensus protocol
|
|
// please think before randomly editing for after effects
|
|
// We must not call any packages that can call panic
|
|
// NO Panics or FATALs please
|
|
|
|
/*
|
|
import "os"
|
|
|
|
|
|
*/
|
|
import "fmt"
|
|
import "sync"
|
|
import "time"
|
|
import "runtime/debug"
|
|
import "bytes"
|
|
import "sort"
|
|
import "math/big"
|
|
|
|
//import "runtime"
|
|
//import "bufio"
|
|
import "golang.org/x/crypto/sha3"
|
|
import "github.com/romana/rlog"
|
|
|
|
import "sync/atomic"
|
|
|
|
import log "github.com/sirupsen/logrus"
|
|
|
|
import "github.com/golang/groupcache/lru"
|
|
import hashicorp_lru "github.com/hashicorp/golang-lru"
|
|
|
|
import "github.com/deroproject/derohe/config"
|
|
import "github.com/deroproject/derohe/crypto"
|
|
import "github.com/deroproject/derohe/errormsg"
|
|
import "github.com/prometheus/client_golang/prometheus"
|
|
|
|
//import "github.com/deroproject/derosuite/address"
|
|
import "github.com/deroproject/derohe/block"
|
|
import "github.com/deroproject/derohe/globals"
|
|
import "github.com/deroproject/derohe/transaction"
|
|
import "github.com/deroproject/derohe/blockchain/mempool"
|
|
import "github.com/deroproject/derohe/blockchain/regpool"
|
|
|
|
/*
|
|
import "github.com/deroproject/derosuite/emission"
|
|
|
|
import "github.com/deroproject/derosuite/storage"
|
|
import "github.com/deroproject/derosuite/crypto/ringct"
|
|
|
|
|
|
import "github.com/deroproject/derosuite/checkpoints"
|
|
import "github.com/deroproject/derosuite/metrics"
|
|
|
|
import "github.com/deroproject/derosuite/blockchain/inputmaturity"
|
|
*/
|
|
|
|
import "github.com/deroproject/graviton"
|
|
|
|
// all components requiring access to blockchain must use , this struct to communicate
|
|
// this structure must be update while mutex
|
|
type Blockchain struct {
|
|
Store storage // interface to storage layer
|
|
Height int64 // chain height is always 1 more than block
|
|
height_seen int64 // height seen on peers
|
|
Top_ID crypto.Hash // id of the top block
|
|
|
|
Tips map[crypto.Hash]crypto.Hash // current tips
|
|
|
|
dag_unsettled map[crypto.Hash]bool // current unsettled dag
|
|
dag_past_unsettled_cache *lru.Cache
|
|
dag_future_unsettled_cache *lru.Cache
|
|
lrucache_workscore *lru.Cache
|
|
lrucache_fullorder *lru.Cache // keeps full order for tips upto a certain height
|
|
|
|
MINING_BLOCK bool // used to pause mining
|
|
|
|
Difficulty uint64 // current cumulative difficulty
|
|
Median_Block_Size uint64 // current median block size
|
|
Mempool *mempool.Mempool // normal tx pool
|
|
Regpool *regpool.Regpool // registration pool
|
|
Exit_Event chan bool // blockchain is shutting down and we must quit ASAP
|
|
|
|
Top_Block_Median_Size uint64 // median block size of current top block
|
|
Top_Block_Base_Reward uint64 // top block base reward
|
|
|
|
checkpints_disabled bool // are checkpoints disabled
|
|
simulator bool // is simulator mode
|
|
|
|
P2P_Block_Relayer func(*block.Complete_Block, uint64) // tell p2p to broadcast any block this daemon hash found
|
|
|
|
sync.RWMutex
|
|
}
|
|
|
|
var logger *log.Entry
|
|
|
|
//var Exit_Event = make(chan bool) // causes all threads to exit
|
|
|
|
// All blockchain activity is store in a single
|
|
|
|
/* do initialisation , setup storage, put genesis block and chain in store
|
|
This is the first component to get up
|
|
Global parameters are picked up from the config package
|
|
*/
|
|
|
|
func Blockchain_Start(params map[string]interface{}) (*Blockchain, error) {
|
|
|
|
var err error
|
|
var chain Blockchain
|
|
|
|
_ = err
|
|
|
|
logger = globals.Logger.WithFields(log.Fields{"com": "CORE"})
|
|
logger.Infof("Initialising blockchain")
|
|
//init_static_checkpoints() // init some hard coded checkpoints
|
|
//checkpoints.LoadCheckPoints(logger) // load checkpoints from file if provided
|
|
|
|
chain.Store.Initialize(params)
|
|
chain.Tips = map[crypto.Hash]crypto.Hash{}
|
|
|
|
//chain.Tips = map[crypto.Hash]crypto.Hash{} // initialize Tips map
|
|
chain.lrucache_workscore = lru.New(8191) // temporary cache for work caclculation
|
|
chain.lrucache_fullorder = lru.New(20480) // temporary cache for fullorder caclculation
|
|
|
|
if globals.Arguments["--disable-checkpoints"] != nil {
|
|
chain.checkpints_disabled = globals.Arguments["--disable-checkpoints"].(bool)
|
|
}
|
|
|
|
if params["--simulator"] == true {
|
|
chain.simulator = true // enable simulator mode, this will set hard coded difficulty to 1
|
|
}
|
|
|
|
chain.Exit_Event = make(chan bool) // init exit channel
|
|
|
|
// init mempool before chain starts
|
|
chain.Mempool, err = mempool.Init_Mempool(params)
|
|
chain.Regpool, err = regpool.Init_Regpool(params)
|
|
|
|
if !chain.Store.IsBalancesIntialized() {
|
|
logger.Debugf("Genesis block not in store, add it now")
|
|
var complete_block block.Complete_Block
|
|
bl := Generate_Genesis_Block()
|
|
complete_block.Bl = &bl
|
|
|
|
if err, ok := chain.Add_Complete_Block(&complete_block); !ok {
|
|
logger.Fatalf("Failed to add genesis block, we can no longer continue. err %s", err)
|
|
}
|
|
}
|
|
/*
|
|
|
|
// genesis block not in chain, add it to chain, together with its miner tx
|
|
// make sure genesis is in the store
|
|
|
|
//if !chain.Block_Exists(globals.Config.Genesis_Block_Hash) {
|
|
if !chain.Block_Exists(nil, bl.GetHash()) {
|
|
//chain.Store_TOP_ID(globals.Config.Genesis_Block_Hash) // store top id , exception of genesis block
|
|
|
|
|
|
|
|
|
|
logger.Infof("Added block successfully")
|
|
|
|
//chain.store_Block_Settled(bl.GetHash(),true) // genesis block is always settled
|
|
|
|
|
|
|
|
bl_current_hash := bl.GetHash()
|
|
// store total reward
|
|
//dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_MINERTX_REWARD, bl.Miner_TX.Vout[0].Amount)
|
|
|
|
// store base reward
|
|
//dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_BASEREWARD, bl.Miner_TX.Vout[0].Amount)
|
|
|
|
// store total generated coins
|
|
// this is hardcoded at initial chain import, keeping original emission schedule
|
|
if globals.IsMainnet(){
|
|
//dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_ALREADY_GENERATED_COINS, config.MAINNET_HARDFORK_1_TOTAL_SUPPLY)
|
|
}else{
|
|
//dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_ALREADY_GENERATED_COINS, config.TESTNET_HARDFORK_1_TOTAL_SUPPLY)
|
|
}
|
|
|
|
|
|
//chain.Store_Block_Topological_order(dbtx, bl.GetHash(), 0) // genesis block is the lowest
|
|
//chain.Store_TOPO_HEIGHT(dbtx, 0) //
|
|
//chain.Store_TOP_HEIGHT(dbtx, 0)
|
|
|
|
//chain.store_TIPS(dbtx, []crypto.Hash{bl.GetHash()})
|
|
|
|
//dbtx.Commit()
|
|
|
|
}
|
|
*/
|
|
|
|
//fmt.Printf("Genesis Block should be present at height 0\n")
|
|
//blocks := chain.Get_Blocks_At_Height(0)
|
|
// fmt.Printf("blocks at height 0 %+v\n", blocks)
|
|
|
|
// fmt.Printf("Past of genesis %+v\n", chain.Get_Block_Past(bl.GetHash()))
|
|
// fmt.Printf("Future of genesis %+v\n", chain.Get_Block_Future(bl.GetHash()))
|
|
|
|
// fmt.Printf("Future of zero block %+v\n", chain.Get_Block_Future(ZERO_HASH))
|
|
|
|
// hard forks must be initialized asap
|
|
init_hard_forks(params)
|
|
|
|
// load the chain from the disk
|
|
chain.Initialise_Chain_From_DB()
|
|
|
|
// logger.Fatalf("Testing complete quitting")
|
|
|
|
//go clean_up_valid_cache() // clean up valid cache
|
|
|
|
/* txlist := chain.Mempool.Mempool_List_TX()
|
|
for i := range txlist {
|
|
// if fmt.Sprintf("%s", txlist[i]) == "0fe0e7270ba911956e91d9ea099e4d12aa1bce2473d4064e239731bc37acfd86"{
|
|
logger.Infof("Verifying tx %s %+v", txlist[i], chain.Verify_Transaction_NonCoinbase(chain.Mempool.Mempool_Get_TX(txlist[i])))
|
|
|
|
//}
|
|
//p2p.Broadcast_Tx(chain.Mempool.Mempool_Get_TX(txlist[i]))
|
|
}
|
|
|
|
|
|
if chain.checkpints_disabled {
|
|
logger.Infof("Internal Checkpoints are disabled")
|
|
} else {
|
|
logger.Debugf("Internal Checkpoints are enabled")
|
|
}
|
|
|
|
_ = err
|
|
|
|
*/
|
|
|
|
/*
|
|
// register the metrics with the metrics registry
|
|
metrics.Registry.MustRegister(blockchain_tx_counter)
|
|
metrics.Registry.MustRegister(mempool_tx_counter)
|
|
metrics.Registry.MustRegister(mempool_tx_count)
|
|
metrics.Registry.MustRegister(block_size)
|
|
metrics.Registry.MustRegister(transaction_size)
|
|
metrics.Registry.MustRegister(block_tx_count)
|
|
metrics.Registry.MustRegister(block_processing_time)
|
|
*/
|
|
atomic.AddUint32(&globals.Subsystem_Active, 1) // increment subsystem
|
|
|
|
return &chain, nil
|
|
}
|
|
|
|
// this is the only entrypoint for new / old blocks even for genesis block
|
|
// this will add the entire block atomically to the chain
|
|
// this is the only function which can add blocks to the chain
|
|
// this is exported, so ii can be fed new blocks by p2p layer
|
|
// genesis block is no different
|
|
// TODO: we should stop mining while adding the new block
|
|
func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err error, result bool) {
|
|
|
|
var block_hash crypto.Hash
|
|
chain.Lock()
|
|
defer chain.Unlock()
|
|
result = false
|
|
|
|
chain.MINING_BLOCK = true
|
|
|
|
processing_start := time.Now()
|
|
|
|
//old_top := chain.Load_TOP_ID() // store top as it may change
|
|
defer func() {
|
|
|
|
// safety so if anything wrong happens, verification fails
|
|
if r := recover(); r != nil {
|
|
logger.Warnf("Recovered while adding new block, Stack trace below block_hash %s", block_hash)
|
|
logger.Warnf("Stack trace \n%s", debug.Stack())
|
|
result = false
|
|
err = errormsg.ErrPanic
|
|
}
|
|
|
|
chain.MINING_BLOCK = false
|
|
|
|
if result == true { // block was successfully added, commit it atomically
|
|
rlog.Infof("Block successfully acceppted by chain %s", block_hash)
|
|
|
|
// gracefully try to instrument
|
|
func() {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
rlog.Warnf("Recovered while instrumenting")
|
|
rlog.Warnf("Stack trace \n%s", debug.Stack())
|
|
|
|
}
|
|
}()
|
|
blockchain_tx_counter.Add(float64(len(cbl.Bl.Tx_hashes)))
|
|
block_tx_count.Observe(float64(len(cbl.Bl.Tx_hashes)))
|
|
block_processing_time.Observe(float64(time.Now().Sub(processing_start).Round(time.Millisecond) / 1000000))
|
|
|
|
// tracks counters for tx_size
|
|
|
|
{
|
|
complete_block_size := 0
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
tx_size := len(cbl.Txs[i].Serialize())
|
|
complete_block_size += tx_size
|
|
transaction_size.Observe(float64(tx_size))
|
|
}
|
|
block_size.Observe(float64(complete_block_size))
|
|
}
|
|
}()
|
|
|
|
//dbtx.Sync() // sync the DB to disk after every execution of this function
|
|
|
|
//if old_top != chain.Load_TOP_ID() { // if top has changed, discard mining templates and start afresh
|
|
// TODO discard mining templates or something else, if top chnages requires some action
|
|
|
|
//}
|
|
} else {
|
|
//dbtx.Rollback() // if block could not be added, rollback all changes to previous block
|
|
rlog.Infof("Block rejected by chain %s err %s", block_hash, err)
|
|
}
|
|
}()
|
|
|
|
bl := cbl.Bl // small pointer to block
|
|
|
|
// first of all lets do some quick checks
|
|
// before doing extensive checks
|
|
result = false
|
|
|
|
block_hash = bl.GetHash()
|
|
block_logger := logger.WithFields(log.Fields{"blid": block_hash})
|
|
|
|
// check if block already exist skip it
|
|
if chain.Is_Block_Topological_order(block_hash) {
|
|
block_logger.Debugf("block already in chain skipping it ")
|
|
return errormsg.ErrAlreadyExists, false
|
|
}
|
|
|
|
// only 3 tips allowed in block
|
|
if len(bl.Tips) >= 4 {
|
|
rlog.Warnf("More than 3 tips present in block %s rejecting", block_hash)
|
|
return errormsg.ErrPastMissing, false
|
|
}
|
|
|
|
// check whether the tips exist in our chain, if not reject
|
|
for i := range bl.Tips { //if !chain.Is_Block_Topological_order(bl.Tips[i]) {
|
|
if !chain.Block_Exists(bl.Tips[i]) { // alt-tips might not have a topo order at this point, so make sure they exist on disk
|
|
rlog.Warnf("Tip %s is NOT present in chain current block %s, skipping it till we get a parent", bl.Tips[i], block_hash)
|
|
return errormsg.ErrPastMissing, false
|
|
}
|
|
}
|
|
|
|
block_height := chain.Calculate_Height_At_Tips(bl.Tips)
|
|
|
|
if block_height == 0 && int64(bl.Height) == block_height && len(bl.Tips) != 0 {
|
|
block_logger.Warnf("Genesis block cannot have tips. len of tips(%d)", len(bl.Tips))
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
if len(bl.Tips) >= 1 && bl.Height == 0 {
|
|
block_logger.Warnf("Genesis block can only be at height 0. len of tips(%d)", len(bl.Tips))
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
if block_height != 0 && block_height < chain.Get_Stable_Height() {
|
|
rlog.Warnf("Block %s rejected since it is stale stable height %d block height %d", bl.GetHash(), chain.Get_Stable_Height(), block_height)
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
// use checksum to quick jump
|
|
/*
|
|
if chain.checkpints_disabled == false && checkpoints.IsCheckSumKnown(chain.BlockCheckSum(cbl)) {
|
|
rlog.Debugf("Skipping Deep Checks for block %s ", block_hash)
|
|
goto skip_checks
|
|
} else {
|
|
rlog.Debugf("Deep Checks for block %s ", block_hash)
|
|
}
|
|
*/
|
|
|
|
// make sure time is NOT into future, we do not have any margin here
|
|
// some OS have trouble syncing with more than 1 sec granularity
|
|
// if clock diff is more than 1 secs, reject the block
|
|
if bl.Timestamp > (uint64(time.Now().UTC().Unix())) {
|
|
block_logger.Warnf("Rejecting Block, timestamp is too much into future, make sure that system clock is correct")
|
|
return errormsg.ErrFutureTimestamp, false
|
|
}
|
|
|
|
// verify that the clock is not being run in reverse
|
|
// the block timestamp cannot be less than any of the parents
|
|
for i := range bl.Tips {
|
|
if chain.Load_Block_Timestamp(bl.Tips[i]) > bl.Timestamp {
|
|
block_logger.Warnf("Block timestamp is less than its parent, rejecting block %x ",bl.Serialize() )
|
|
return errormsg.ErrInvalidTimestamp, false
|
|
}
|
|
}
|
|
|
|
//logger.Infof("current version %d height %d", chain.Get_Current_Version_at_Height( 2500), chain.Calculate_Height_At_Tips(dbtx, bl.Tips))
|
|
// check whether the major version ( hard fork) is valid
|
|
if !chain.Check_Block_Version(bl) {
|
|
block_logger.Warnf("Rejecting !! Block has invalid fork version actual %d expected %d", bl.Major_Version, chain.Get_Current_Version_at_Height(chain.Calculate_Height_At_Tips( bl.Tips)))
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
// verify whether the tips are unreachable from one another
|
|
if !chain.VerifyNonReachability(bl) {
|
|
block_logger.Warnf("Rejecting !! Block has invalid reachability")
|
|
return errormsg.ErrInvalidBlock, false
|
|
|
|
}
|
|
|
|
// if the block is referencing any past tip too distant into main chain discard now
|
|
// TODO FIXME this need to computed
|
|
for i := range bl.Tips {
|
|
rusty_tip_base_distance := chain.calculate_mainchain_distance(bl.Tips[i])
|
|
|
|
// tips of deviation >= 8 will rejected
|
|
if (int64(chain.Get_Height()) - rusty_tip_base_distance) >= config.STABLE_LIMIT {
|
|
block_logger.Warnf("Rusty TIP mined by ROGUE miner discarding block %s best height %d deviation %d rusty_tip %d", bl.Tips[i], chain.Get_Height(), (int64(chain.Get_Height()) - rusty_tip_base_distance), rusty_tip_base_distance)
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
}
|
|
|
|
// verify difficulty of tips provided
|
|
if len(bl.Tips) > 1 {
|
|
best_tip := chain.find_best_tip_cumulative_difficulty(bl.Tips)
|
|
for i := range bl.Tips {
|
|
if best_tip != bl.Tips[i] {
|
|
if !chain.validate_tips(best_tip, bl.Tips[i]) { // reference is first
|
|
block_logger.Warnf("Rusty tip mined by ROGUE miner, discarding block")
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// check whether the block crosses the size limit
|
|
// block size is calculate by adding all the txs
|
|
// block header/miner tx is excluded, only tx size if calculated
|
|
{
|
|
block_size := 0
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
block_size += len(cbl.Txs[i].Serialize())
|
|
if uint64(block_size) >= config.STARGATE_HE_MAX_BLOCK_SIZE {
|
|
block_logger.Warnf("Block is bigger than max permitted, Rejecting it Actual %d MAX %d ", block_size, config.STARGATE_HE_MAX_BLOCK_SIZE)
|
|
return errormsg.ErrInvalidSize, false
|
|
}
|
|
}
|
|
}
|
|
|
|
//logger.Infof("pow hash %s height %d", bl.GetPoWHash(), block_height)
|
|
|
|
// Verify Blocks Proof-Of-Work
|
|
// check if the PoW is satisfied
|
|
if !chain.VerifyPoW(bl) { // if invalid Pow, reject the bloc
|
|
block_logger.Warnf("Block has invalid PoW, rejecting it %x", bl.Serialize())
|
|
return errormsg.ErrInvalidPoW, false
|
|
}
|
|
|
|
{ // miner TX checks are here
|
|
if bl.Height == 0 && !bl.Miner_TX.IsPremine() { // genesis block contain premine tx a
|
|
block_logger.Warnf("Miner tx failed verification for genesis rejecting ")
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
if bl.Height != 0 && !bl.Miner_TX.IsCoinbase() { // all blocks except genesis block contain coinbase TX
|
|
block_logger.Warnf("Miner tx failed it is not coinbase ")
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
// always check whether the coin base tx is okay
|
|
if bl.Height != 0 && !chain.Verify_Transaction_Coinbase(cbl, &bl.Miner_TX) { // if miner address is not registered give error
|
|
block_logger.Warnf("Miner address is not registered")
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
// TODO we need to verify address whether they are valid points on curve or not
|
|
}
|
|
|
|
// now we need to verify each and every tx in detail
|
|
// we need to verify each and every tx contained in the block, sanity check everything
|
|
// first of all check, whether all the tx contained in the block, match their hashes
|
|
{
|
|
if len(bl.Tx_hashes) != len(cbl.Txs) {
|
|
block_logger.Warnf("Block says it has %d txs , however complete block contained %d txs", len(bl.Tx_hashes), len(cbl.Txs))
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
// first check whether the complete block contains any diplicate hashes
|
|
tx_checklist := map[crypto.Hash]bool{}
|
|
for i := 0; i < len(bl.Tx_hashes); i++ {
|
|
tx_checklist[bl.Tx_hashes[i]] = true
|
|
}
|
|
|
|
if len(tx_checklist) != len(bl.Tx_hashes) { // block has duplicate tx, reject
|
|
block_logger.Warnf("Block has %d duplicate txs, reject it", len(bl.Tx_hashes)-len(tx_checklist))
|
|
return errormsg.ErrInvalidBlock, false
|
|
|
|
}
|
|
// now lets loop through complete block, matching each tx
|
|
// detecting any duplicates using txid hash
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
tx_hash := cbl.Txs[i].GetHash()
|
|
if _, ok := tx_checklist[tx_hash]; !ok {
|
|
// tx is NOT found in map, RED alert reject the block
|
|
block_logger.Warnf("Block says it has tx %s, but complete block does not have it", tx_hash)
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
}
|
|
}
|
|
|
|
// another check, whether the block contains any duplicate registration within the block
|
|
// block wide duplicate input detector
|
|
{
|
|
nonce_map := map[string]bool{}
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
|
|
if cbl.Txs[i].TransactionType == transaction.REGISTRATION {
|
|
if _, ok := nonce_map[string(cbl.Txs[i].MinerAddress[:])]; ok {
|
|
block_logger.Warnf("Double Registration within block %s", cbl.Txs[i].GetHash())
|
|
return errormsg.ErrTXDoubleSpend, false
|
|
}
|
|
nonce_map[string(cbl.Txs[i].MinerAddress[:])] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
// another check, whether the tx is build with the latest snapshot of balance tree
|
|
{
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
if cbl.Txs[i].TransactionType == transaction.NORMAL {
|
|
if cbl.Txs[i].Height+1 != cbl.Bl.Height {
|
|
block_logger.Warnf("invalid tx mined %s", cbl.Txs[i].GetHash())
|
|
return errormsg.ErrTXDoubleSpend, false
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
// another check, whether the tx contains any duplicate nonces within the block
|
|
// block wide duplicate input detector
|
|
{
|
|
nonce_map := map[crypto.Hash]bool{}
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
|
|
if cbl.Txs[i].TransactionType == transaction.NORMAL {
|
|
if _, ok := nonce_map[cbl.Txs[i].Proof.Nonce()]; ok {
|
|
block_logger.Warnf("Double Spend attack within block %s", cbl.Txs[i].GetHash())
|
|
return errormsg.ErrTXDoubleSpend, false
|
|
}
|
|
nonce_map[cbl.Txs[i].Proof.Nonce()] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
// we also need to reject if the the immediately reachable history, has spent the nonce
|
|
// both the checks works on the basis of nonces and not on the basis of txhash
|
|
/*
|
|
{
|
|
reachable_nonces := chain.BuildReachabilityNonces(bl)
|
|
for i := 0; i < len(cbl.Txs); i++ { // loop through all the TXs
|
|
if cbl.Txs[i].TransactionType == transaction.NORMAL {
|
|
if _, ok := reachable_nonces[cbl.Txs[i].Proof.Nonce()]; ok {
|
|
block_logger.Warnf("Double spend attack tx %s is already mined, rejecting ", cbl.Txs[i].GetHash())
|
|
return errormsg.ErrTXDoubleSpend, false
|
|
}
|
|
}
|
|
}
|
|
}*/
|
|
|
|
// we need to anyways verify the TXS since proofs are not covered by checksum
|
|
{
|
|
fail_count := int32(0)
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(len(cbl.Txs)) // add total number of tx as work
|
|
|
|
hf_version := chain.Get_Current_Version_at_Height(chain.Calculate_Height_At_Tips(bl.Tips))
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
go func(j int) {
|
|
if !chain.Verify_Transaction_NonCoinbase(hf_version, cbl.Txs[j]) { // transaction verification failed
|
|
atomic.AddInt32(&fail_count, 1) // increase fail count by 1
|
|
block_logger.Warnf("Block verification failed rejecting since TX %s verification failed", cbl.Txs[j].GetHash())
|
|
}
|
|
wg.Done()
|
|
}(i)
|
|
}
|
|
|
|
wg.Wait() // wait for verifications to finish
|
|
if fail_count > 0 { // check the result
|
|
block_logger.Warnf("Block verification failed rejecting since TX verification failed ")
|
|
return errormsg.ErrInvalidTX, false
|
|
}
|
|
|
|
}
|
|
|
|
// we are here means everything looks good, proceed and save to chain
|
|
//skip_checks:
|
|
|
|
// save all the txs
|
|
// and then save the block
|
|
{ // first lets save all the txs, together with their link to this block as height
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
if err = chain.Store.Block_tx_store.WriteTX(bl.Tx_hashes[i], cbl.Txs[i].Serialize()); err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
chain.StoreBlock(bl)
|
|
|
|
// if the block is on a lower height tip, the block will not increase chain height
|
|
height := chain.Load_Height_for_BL_ID(block_hash)
|
|
if height > chain.Get_Height() || height == 0 { // exception for genesis block
|
|
atomic.StoreInt64(&chain.Height, height)
|
|
//chain.Store_TOP_HEIGHT(dbtx, height)
|
|
rlog.Infof("Chain extended new height %d blid %s", chain.Height, block_hash)
|
|
|
|
} else {
|
|
rlog.Infof("Chain extended but height is same %d blid %s", chain.Height, block_hash)
|
|
|
|
}
|
|
|
|
// calculate new set of tips
|
|
// this is done by removing all known tips which are in the past
|
|
// and add this block as tip
|
|
|
|
//past := chain.Get_Block_Past( bl.GetHash())
|
|
|
|
tips := chain.Get_TIPS()
|
|
tips = tips[:0]
|
|
//chain.Tips[bl.GetHash()] = bl.GetHash() // add this new block as tip
|
|
|
|
for k := range chain.Tips {
|
|
for i := range bl.Tips {
|
|
if bl.Tips[i] == k {
|
|
goto skip_tip
|
|
}
|
|
|
|
}
|
|
tips = append(tips, k)
|
|
skip_tip:
|
|
}
|
|
|
|
tips = append(tips, bl.GetHash()) // add current block as new tip
|
|
|
|
// find the biggest tip in terms of work
|
|
{
|
|
|
|
base, base_height := chain.find_common_base(tips)
|
|
best := chain.find_best_tip(tips, base, base_height)
|
|
|
|
// we only generate full order for the biggest tip
|
|
|
|
//gbl := Generate_Genesis_Block()
|
|
// full_order := chain.Generate_Full_Order( bl.GetHash(), gbl.GetHash(), 0,0)
|
|
//base_topo_index := chain.Load_Block_Topological_order(gbl.GetHash())
|
|
|
|
full_order := chain.Generate_Full_Order(best, base, base_height, 0)
|
|
base_topo_index := chain.Load_Block_Topological_order(base)
|
|
|
|
// we will directly use graviton to mov in to history
|
|
rlog.Infof("Full order %+v base %s base topo pos %d", full_order, base, base_topo_index)
|
|
|
|
if len(bl.Tips) == 0 {
|
|
base_topo_index = 0
|
|
}
|
|
|
|
// any blocks which have not changed their topo will be skipped using graviton trick
|
|
for i := int64(0); i < int64(len(full_order)); i++ {
|
|
|
|
// check whether the new block is at the same position at the last position
|
|
current_topo_block := i + base_topo_index
|
|
if current_topo_block < chain.Store.Topo_store.Count() {
|
|
toporecord, err := chain.Store.Topo_store.Read(current_topo_block)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
if full_order[i] == toporecord.BLOCK_ID { // skip reprocessing if not required
|
|
continue
|
|
}
|
|
|
|
}
|
|
|
|
// TODO we must run smart contracts and TXs in this order
|
|
// basically client protocol must run here
|
|
// even if the HF has triggered we may still accept, old blocks for some time
|
|
// so hf is detected block-wise and processed as such
|
|
|
|
bl_current_hash := full_order[i]
|
|
bl_current, err1 := chain.Load_BL_FROM_ID(bl_current_hash)
|
|
if err1 != nil {
|
|
block_logger.Debugf("Cannot load block %s for client protocol,probably DB corruption", bl_current_hash)
|
|
return errormsg.ErrInvalidBlock, false
|
|
}
|
|
|
|
//fmt.Printf("\ni %d bl %+v\n",i, bl_current)
|
|
|
|
height_current := chain.Calculate_Height_At_Tips(bl_current.Tips)
|
|
hard_fork_version_current := chain.Get_Current_Version_at_Height(height_current)
|
|
|
|
// this version does not require client protocol as of now
|
|
// run full client protocol and find valid transactions
|
|
// rlog.Debugf("running client protocol for %s minertx %s topo %d", bl_current_hash, bl_current.Miner_TX.GetHash(), highest_topo)
|
|
|
|
// generate miner TX rewards as per client protocol
|
|
if hard_fork_version_current == 1 {
|
|
|
|
//CommitElGamal
|
|
/*
|
|
|
|
// hf 2 or later generate miner TX rewards as per client protocol
|
|
|
|
past_coins_generated := chain.Load_Already_Generated_Coins_for_Topo_Index(dbtx, highest_topo-1)
|
|
|
|
base_reward := emission.GetBlockReward_Atlantis(hard_fork_version_current, past_coins_generated)
|
|
|
|
// base reward is only 90%, rest 10 % is pushed back
|
|
if globals.IsMainnet(){
|
|
base_reward = (base_reward * 9) / 10
|
|
}
|
|
|
|
// lower reward for byzantine behaviour
|
|
// for as many block as added
|
|
if chain.isblock_SideBlock(dbtx, bl_current_hash, highest_topo) { // lost race (or byzantine behaviour)
|
|
if hard_fork_version_current == 2 {
|
|
base_reward = (base_reward * 67) / 100 // give only 67 % reward
|
|
}else{
|
|
base_reward = (base_reward * 8) / 100 // give only 8 % reward
|
|
}
|
|
}
|
|
|
|
// logger.Infof("past coins generated %d base reward %d", past_coins_generated, base_reward)
|
|
|
|
// the total reward must be given to the miner TX, since it contains 0, we patch only the output
|
|
// and leave the original TX untouched
|
|
total_reward := base_reward + total_fees
|
|
|
|
// store total reward
|
|
dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_MINERTX_REWARD, total_reward)
|
|
|
|
// store base reward
|
|
dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_BASEREWARD, base_reward)
|
|
|
|
// store total generated coins
|
|
dbtx.StoreUint64(BLOCKCHAIN_UNIVERSE, GALAXY_BLOCK, bl_current_hash[:], PLANET_ALREADY_GENERATED_COINS, past_coins_generated+base_reward)
|
|
|
|
//logger.Infof("base reward %s total generated %s",globals.FormatMoney12(base_reward), globals.FormatMoney12(past_coins_generated+base_reward))
|
|
*/
|
|
|
|
}
|
|
|
|
var balance_tree *graviton.Tree
|
|
//
|
|
if bl_current.Height == 0 { // if it's genesis block
|
|
ss, err := chain.Store.Balance_store.LoadSnapshot(0)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
balance_tree, err = ss.GetTree(BALANCE_TREE)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
} else { // we already have a block before us, use it
|
|
previous_topo_block := i + base_topo_index - 1
|
|
|
|
record_version := uint64(0)
|
|
|
|
if previous_topo_block >= 0 {
|
|
toporecord, err := chain.Store.Topo_store.Read(previous_topo_block)
|
|
|
|
//fmt.Printf("current block %d previous topo %d record %+v err %s\n", i+base_topo_index, i+base_topo_index-1, toporecord,err)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
record_version = toporecord.State_Version
|
|
}
|
|
|
|
ss, err := chain.Store.Balance_store.LoadSnapshot(record_version)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
balance_tree, err = ss.GetTree(BALANCE_TREE)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
}
|
|
|
|
fees_collected := uint64(0)
|
|
|
|
// side blocks only represent chain strenth , else they are are ignored
|
|
// this means they donot get any reward , 0 reward
|
|
// their transactions are ignored
|
|
|
|
//chain.Store.Topo_store.Write(i+base_topo_index, full_order[i],0, int64(bl_current.Height)) // write entry so as sideblock could work
|
|
if !chain.isblock_SideBlock_internal(full_order[i], i+base_topo_index, int64(bl_current.Height)) {
|
|
|
|
for _, txhash := range bl_current.Tx_hashes { // execute all the transactions
|
|
|
|
if tx_bytes, err := chain.Store.Block_tx_store.ReadTX(txhash); err != nil {
|
|
panic(err)
|
|
} else {
|
|
|
|
var tx transaction.Transaction
|
|
if err = tx.DeserializeHeader(tx_bytes); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// we have loaded a tx successfully, now lets execute it
|
|
fees_collected += chain.process_transaction(tx, balance_tree)
|
|
}
|
|
|
|
}
|
|
|
|
chain.process_miner_transaction(bl_current.Miner_TX, bl_current.Height == 0, balance_tree, fees_collected)
|
|
|
|
} else {
|
|
rlog.Debugf("this block is a side block block height %d blid %s ", chain.Load_Block_Height(full_order[i]), full_order[i])
|
|
|
|
}
|
|
|
|
// we are here, means everything is okay, lets commit the update balance tree
|
|
commit_version, err := graviton.Commit(balance_tree)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
chain.Store.Topo_store.Write(i+base_topo_index, full_order[i], commit_version, chain.Load_Block_Height(full_order[i]))
|
|
|
|
rlog.Debugf("%d %s topo_index %d base topo %d", i, full_order[i], i+base_topo_index, base_topo_index)
|
|
|
|
// this tx must be stored, linked with this block
|
|
|
|
}
|
|
|
|
// set main chain as new topo order
|
|
// we must discard any rusty tips after they go stale
|
|
best_height := int64(chain.Load_Height_for_BL_ID(best))
|
|
|
|
new_tips := []crypto.Hash{}
|
|
for i := range tips {
|
|
rusty_tip_base_distance := chain.calculate_mainchain_distance(tips[i])
|
|
// tips of deviation > 6 will be rejected
|
|
if (best_height - rusty_tip_base_distance) < (config.STABLE_LIMIT - 1) {
|
|
new_tips = append(new_tips, tips[i])
|
|
|
|
} else { // this should be a rarest event, probably should never occur, until the network is under sever attack
|
|
logger.Warnf("Rusty TIP declared stale %s best height %d deviation %d rusty_tip %d", tips[i], best_height, (best_height - rusty_tip_base_distance), rusty_tip_base_distance)
|
|
//chain.transaction_scavenger(dbtx, tips[i]) // scavenge tx if possible
|
|
// TODO we must include any TX from the orphan blocks back to the mempool to avoid losing any TX
|
|
}
|
|
}
|
|
|
|
// do more cleanup of tips for byzantine behaviour
|
|
// this copy is necessary, otherwise data corruption occurs
|
|
tips = append([]crypto.Hash{}, new_tips...)
|
|
best_tip := chain.find_best_tip_cumulative_difficulty(tips)
|
|
|
|
{
|
|
new_tips := map[crypto.Hash]crypto.Hash{}
|
|
new_tips[best_tip] = best_tip
|
|
for i := range tips {
|
|
if best_tip != tips[i] {
|
|
if !chain.validate_tips(best_tip, tips[i]) { // reference is first
|
|
logger.Warnf("Rusty tip %s declaring stale", tips[i])
|
|
//chain.transaction_scavenger(dbtx, tips[i]) // scavenge tx if possible
|
|
} else {
|
|
//new_tips = append(new_tips, tips[i])
|
|
new_tips[tips[i]] = tips[i]
|
|
}
|
|
}
|
|
}
|
|
|
|
rlog.Infof("New tips(after adding %s) %+v", bl.GetHash(), new_tips)
|
|
|
|
chain.Tips = new_tips
|
|
}
|
|
}
|
|
|
|
//chain.store_TIPS(chain.)
|
|
|
|
//chain.Top_ID = block_hash // set new top block id
|
|
|
|
// every 200 block print a line
|
|
if chain.Get_Height()%200 == 0 {
|
|
block_logger.Infof("Chain Height %d", chain.Height)
|
|
}
|
|
|
|
result = true
|
|
|
|
// TODO fix hard fork
|
|
// maintain hard fork votes to keep them SANE
|
|
//chain.Recount_Votes() // does not return anything
|
|
|
|
// enable mempool book keeping
|
|
|
|
func() {
|
|
if r := recover(); r != nil {
|
|
logger.Warnf("Mempool House Keeping triggered panic height = %d", block_height)
|
|
}
|
|
|
|
// discard the transactions from mempool if they are present there
|
|
chain.Mempool.Monitor()
|
|
|
|
for i := 0; i < len(cbl.Txs); i++ {
|
|
txid := cbl.Txs[i].GetHash()
|
|
|
|
switch cbl.Txs[i].TransactionType {
|
|
|
|
case transaction.REGISTRATION:
|
|
if chain.Regpool.Regpool_TX_Exist(txid) {
|
|
rlog.Tracef(1, "Deleting TX from regpool txid=%s", txid)
|
|
chain.Regpool.Regpool_Delete_TX(txid)
|
|
continue
|
|
}
|
|
|
|
case transaction.NORMAL:
|
|
if chain.Mempool.Mempool_TX_Exist(txid) {
|
|
rlog.Tracef(1, "Deleting TX from mempool txid=%s", txid)
|
|
chain.Mempool.Mempool_Delete_TX(txid)
|
|
continue
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// give mempool an oppurtunity to clean up tx, but only if they are not mined
|
|
chain.Mempool.HouseKeeping(uint64(block_height))
|
|
|
|
// ggive regpool a chance to register
|
|
if ss, err := chain.Store.Balance_store.LoadSnapshot(0); err == nil {
|
|
if balance_tree, err := ss.GetTree(BALANCE_TREE); err == nil {
|
|
|
|
chain.Regpool.HouseKeeping(uint64(block_height), func(tx *transaction.Transaction) bool {
|
|
if tx.TransactionType != transaction.REGISTRATION { // tx not registration so delete
|
|
return true
|
|
}
|
|
|
|
if _, err := balance_tree.Get(tx.MinerAddress[:]); err != nil { // address already registered
|
|
return true
|
|
}
|
|
return false // account not already registered, so give another chance
|
|
})
|
|
|
|
}
|
|
}
|
|
|
|
}()
|
|
|
|
return // run any handlers necesary to atomically
|
|
}
|
|
|
|
// this function is called to read blockchain state from DB
|
|
// It is callable at any point in time
|
|
|
|
func (chain *Blockchain) Initialise_Chain_From_DB() {
|
|
chain.Lock()
|
|
defer chain.Unlock()
|
|
|
|
// find the tips from the chain , first by reaching top height
|
|
// then downgrading to top-10 height
|
|
// then reworking the chain to get the tip
|
|
best_height := chain.Load_TOP_HEIGHT()
|
|
chain.Height = best_height
|
|
|
|
chain.Tips = map[crypto.Hash]crypto.Hash{} // reset the map
|
|
// reload top tip from disk
|
|
top := chain.Get_Top_ID()
|
|
|
|
chain.Tips[top] = top // we only can load a single tip from db
|
|
|
|
// get dag unsettled, it's only possible when we have the tips
|
|
// chain.dag_unsettled = chain.Get_DAG_Unsettled() // directly off the disk
|
|
|
|
logger.Infof("Chain Tips %+v Height %d", chain.Tips, chain.Height)
|
|
|
|
}
|
|
|
|
// before shutdown , make sure p2p is confirmed stopped
|
|
func (chain *Blockchain) Shutdown() {
|
|
|
|
chain.Lock() // take the lock as chain is no longer in unsafe mode
|
|
close(chain.Exit_Event) // send signal to everyone we are shutting down
|
|
|
|
chain.Mempool.Shutdown() // shutdown mempool first
|
|
chain.Regpool.Shutdown() // shutdown regpool first
|
|
|
|
logger.Infof("Stopping Blockchain")
|
|
//chain.Store.Shutdown()
|
|
atomic.AddUint32(&globals.Subsystem_Active, ^uint32(0)) // this decrement 1 fom subsystem
|
|
}
|
|
|
|
// get top unstable height
|
|
// this is obtained by getting the highest topo block and getting its height
|
|
func (chain *Blockchain) Get_Height() int64 {
|
|
topo_count := chain.Store.Topo_store.Count()
|
|
if topo_count == 0 {
|
|
return 0
|
|
}
|
|
|
|
//return atomic.LoadUint64(&chain.Height)
|
|
return chain.Load_TOP_HEIGHT()
|
|
}
|
|
|
|
// get height where chain is now stable
|
|
func (chain *Blockchain) Get_Stable_Height() int64 {
|
|
tips := chain.Get_TIPS()
|
|
base, base_height := chain.find_common_base(tips)
|
|
_ = base
|
|
|
|
return int64(base_height)
|
|
}
|
|
|
|
// we should be holding lock at this time, atleast read only
|
|
|
|
func (chain *Blockchain) Get_TIPS() (tips []crypto.Hash) {
|
|
for _, x := range chain.Tips {
|
|
tips = append(tips, x)
|
|
|
|
}
|
|
return tips
|
|
}
|
|
|
|
func (chain *Blockchain) Get_Difficulty() uint64 {
|
|
return chain.Get_Difficulty_At_Tips(chain.Get_TIPS()).Uint64()
|
|
}
|
|
|
|
/*
|
|
func (chain *Blockchain) Get_Cumulative_Difficulty() uint64 {
|
|
|
|
return 0 //chain.Load_Block_Cumulative_Difficulty(chain.Top_ID)
|
|
}
|
|
|
|
func (chain *Blockchain) Get_Median_Block_Size() uint64 { // get current cached median size
|
|
return chain.Median_Block_Size
|
|
}
|
|
*/
|
|
func (chain *Blockchain) Get_Network_HashRate() uint64 {
|
|
return chain.Get_Difficulty() / chain.Get_Current_BlockTime()
|
|
}
|
|
|
|
// this is used to for quick syncs as entire blocks as SHA1,
|
|
// entires block can skipped for verification, if checksum matches what the devs have stored
|
|
func (chain *Blockchain) BlockCheckSum(cbl *block.Complete_Block) []byte {
|
|
h := sha3.New256()
|
|
h.Write(cbl.Bl.Serialize())
|
|
for i := range cbl.Txs {
|
|
h.Write(cbl.Txs[i].Serialize())
|
|
}
|
|
return h.Sum(nil)
|
|
}
|
|
|
|
// various counters/gauges which track a numer of metrics
|
|
// such as number of txs, number of inputs, number of outputs
|
|
// mempool total addition, current mempool size
|
|
// block processing time etcs
|
|
|
|
// Try it once more, this time with a help string.
|
|
var blockchain_tx_counter = prometheus.NewCounter(prometheus.CounterOpts{
|
|
Name: "blockchain_tx_counter",
|
|
Help: "Number of tx mined",
|
|
})
|
|
|
|
var mempool_tx_counter = prometheus.NewCounter(prometheus.CounterOpts{
|
|
Name: "mempool_tx_counter",
|
|
Help: "Total number of tx added in mempool",
|
|
})
|
|
var mempool_tx_count = prometheus.NewGauge(prometheus.GaugeOpts{
|
|
Name: "mempool_tx_count",
|
|
Help: "Number of tx in mempool at this point",
|
|
})
|
|
|
|
// track block size about 2 MB
|
|
var block_size = prometheus.NewHistogram(prometheus.HistogramOpts{
|
|
Name: "block_size_byte",
|
|
Help: "Block size in byte (complete)",
|
|
Buckets: prometheus.LinearBuckets(0, 102400, 10), // start block size 0, each 1 KB step, 2048 such buckets .
|
|
})
|
|
|
|
// track transaction size upto 500 KB
|
|
var transaction_size = prometheus.NewHistogram(prometheus.HistogramOpts{
|
|
Name: "tx_size_byte",
|
|
Help: "TX size in byte",
|
|
Buckets: prometheus.LinearBuckets(0, 10240, 16), // start 0 byte, each 1024 byte, 512 such buckets.
|
|
})
|
|
|
|
// number of tx per block
|
|
var block_tx_count = prometheus.NewHistogram(prometheus.HistogramOpts{
|
|
Name: "block_tx_count",
|
|
Help: "Number of TX in the block",
|
|
Buckets: prometheus.LinearBuckets(0, 20, 25), // start 0 byte, each 1024 byte, 1024 such buckets.
|
|
})
|
|
|
|
//
|
|
var block_processing_time = prometheus.NewHistogram(prometheus.HistogramOpts{
|
|
Name: "block_processing_time_ms",
|
|
Help: "Block processing time milliseconds",
|
|
Buckets: prometheus.LinearBuckets(0, 100, 20), // start 0 ms, each 100 ms, 200 such buckets.
|
|
})
|
|
|
|
// this is the only entrypoint for new txs in the chain
|
|
// add a transaction to MEMPOOL,
|
|
// verifying everything means everything possible
|
|
// this only change mempool, no DB changes
|
|
func (chain *Blockchain) Add_TX_To_Pool(tx *transaction.Transaction) (result bool) {
|
|
if tx.IsRegistration() { // registration tx will not go any forward
|
|
// ggive regpool a chance to register
|
|
if ss, err := chain.Store.Balance_store.LoadSnapshot(0); err == nil {
|
|
if balance_tree, err := ss.GetTree(BALANCE_TREE); err == nil {
|
|
if _, err := balance_tree.Get(tx.MinerAddress[:]); err == nil { // address already registered
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
return chain.Regpool.Regpool_Add_TX(tx, 0)
|
|
}
|
|
|
|
// track counter for the amount of mempool tx
|
|
defer mempool_tx_count.Set(float64(len(chain.Mempool.Mempool_List_TX())))
|
|
|
|
txhash := tx.GetHash()
|
|
|
|
// Coin base TX can not come through this path
|
|
if tx.IsCoinbase() {
|
|
logger.WithFields(log.Fields{"txid": txhash}).Warnf("TX rejected coinbase tx cannot appear in mempool")
|
|
return false
|
|
}
|
|
|
|
// quick check without calculating everything whether tx is in pool, if yes we do nothing
|
|
if chain.Mempool.Mempool_TX_Exist(txhash) {
|
|
rlog.Tracef(2, "TX %s rejected Already in MEMPOOL", txhash)
|
|
return true
|
|
}
|
|
|
|
hf_version := chain.Get_Current_Version_at_Height(chain.Get_Height())
|
|
|
|
// if TX is too big, then it cannot be mined due to fixed block size, reject such TXs here
|
|
// currently, limits are as per consensus
|
|
if uint64(len(tx.Serialize())) > config.STARGATE_HE_MAX_TX_SIZE {
|
|
logger.WithFields(log.Fields{"txid": txhash}).Warnf("TX rejected Size %d byte Max possible %d", len(tx.Serialize()), config.STARGATE_HE_MAX_TX_SIZE)
|
|
return false
|
|
}
|
|
|
|
// check whether enough fees is provided in the transaction
|
|
calculated_fee := chain.Calculate_TX_fee(hf_version, uint64(len(tx.Serialize())))
|
|
provided_fee := tx.Statement.Fees // get fee from tx
|
|
|
|
_ = calculated_fee
|
|
_ = provided_fee
|
|
|
|
//logger.WithFields(log.Fields{"txid": txhash}).Warnf("TX fees check disabled provided fee %d calculated fee %d", provided_fee, calculated_fee)
|
|
|
|
/*
|
|
if calculated_fee > provided_fee { // 2 % margin see blockchain.cpp L 2913
|
|
logger.WithFields(log.Fields{"txid": txhash}).Warnf("TX rejected due to low fees provided fee %d calculated fee %d", provided_fee, calculated_fee)
|
|
|
|
rlog.Warnf("TX %s rejected due to low fees provided fee %d calculated fee %d", txhash, provided_fee, calculated_fee)
|
|
return false
|
|
}
|
|
*/
|
|
|
|
if chain.Verify_Transaction_NonCoinbase(hf_version, tx) && chain.Verify_Transaction_NonCoinbase_DoubleSpend_Check(tx) {
|
|
if chain.Mempool.Mempool_Add_TX(tx, 0) { // new tx come with 0 marker
|
|
rlog.Tracef(2, "Successfully added tx %s to pool", txhash)
|
|
|
|
mempool_tx_counter.Inc()
|
|
return true
|
|
} else {
|
|
rlog.Tracef(2, "TX %s rejected by pool", txhash)
|
|
return false
|
|
}
|
|
}
|
|
|
|
rlog.Warnf("Incoming TX %s could not be verified", txhash)
|
|
return false
|
|
|
|
}
|
|
|
|
// structure used to rank/sort blocks on a number of factors
|
|
type BlockScore struct {
|
|
BLID crypto.Hash
|
|
// Weight uint64
|
|
Height int64 // block height
|
|
Cumulative_Difficulty *big.Int // used to score blocks on cumulative difficulty
|
|
}
|
|
|
|
// Heighest node weight is ordered first, the condition is reverted see eg. at https://golang.org/pkg/sort/#Slice
|
|
// if weights are equal, nodes are sorted by their block ids which will never collide , hopefullly
|
|
// block ids are sorted by lowest byte first diff
|
|
func sort_descending_by_cumulative_difficulty(tips_scores []BlockScore) {
|
|
|
|
sort.Slice(tips_scores, func(i, j int) bool {
|
|
if tips_scores[i].Cumulative_Difficulty.Cmp(tips_scores[j].Cumulative_Difficulty) != 0 { // if diffculty mismatch use them
|
|
|
|
if tips_scores[i].Cumulative_Difficulty.Cmp(tips_scores[j].Cumulative_Difficulty) > 0 { // if i diff > j diff
|
|
return true
|
|
} else {
|
|
return false
|
|
}
|
|
|
|
} else {
|
|
return bytes.Compare(tips_scores[i].BLID[:], tips_scores[j].BLID[:]) == -1
|
|
}
|
|
})
|
|
}
|
|
|
|
func sort_ascending_by_height(tips_scores []BlockScore) {
|
|
|
|
// base is the lowest height
|
|
sort.Slice(tips_scores, func(i, j int) bool { return tips_scores[i].Height < tips_scores[j].Height })
|
|
|
|
}
|
|
|
|
// this will sort the tips based on cumulative difficulty and/or block ids
|
|
// the tips will sorted in descending order
|
|
func (chain *Blockchain) SortTips(tips []crypto.Hash) (sorted []crypto.Hash) {
|
|
if len(tips) == 0 {
|
|
panic("tips cannot be 0")
|
|
}
|
|
if len(tips) == 1 {
|
|
sorted = []crypto.Hash{tips[0]}
|
|
return
|
|
}
|
|
|
|
tips_scores := make([]BlockScore, len(tips), len(tips))
|
|
for i := range tips {
|
|
tips_scores[i].BLID = tips[i]
|
|
tips_scores[i].Cumulative_Difficulty = chain.Load_Block_Cumulative_Difficulty(tips[i])
|
|
}
|
|
|
|
sort_descending_by_cumulative_difficulty(tips_scores)
|
|
|
|
for i := range tips_scores {
|
|
sorted = append(sorted, tips_scores[i].BLID)
|
|
}
|
|
return
|
|
}
|
|
|
|
// side blocks are blocks which lost the race the to become part
|
|
// of main chain, but there transactions are honoured,
|
|
// they are given 67 % reward
|
|
// a block is a side block if it satisfies the following condition
|
|
// if block height is less than or equal to height of past 3*config.STABLE_LIMIT topographical blocks
|
|
// this is part of consensus rule
|
|
// this is the topoheight of this block itself
|
|
func (chain *Blockchain) isblock_SideBlock(blid crypto.Hash) bool {
|
|
block_topoheight := chain.Load_Block_Topological_order(blid)
|
|
if block_topoheight == 0 {
|
|
return false
|
|
}
|
|
// lower reward for byzantine behaviour
|
|
// for as many block as added
|
|
block_height := chain.Load_Height_for_BL_ID(blid)
|
|
|
|
return chain.isblock_SideBlock_internal(blid, block_topoheight, block_height)
|
|
}
|
|
|
|
func (chain *Blockchain) isblock_SideBlock_internal(blid crypto.Hash, block_topoheight int64, block_height int64) (result bool) {
|
|
if block_topoheight == 0 {
|
|
return false
|
|
}
|
|
counter := int64(0)
|
|
for i := block_topoheight - 1; i >= 0 && counter < 3*config.STABLE_LIMIT; i-- {
|
|
counter++
|
|
toporecord, err := chain.Store.Topo_store.Read(i)
|
|
if err != nil {
|
|
panic("Could not load block from previous order")
|
|
}
|
|
if block_height <= toporecord.Height { // lost race (or byzantine behaviour)
|
|
return true // give only 67 % reward
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// this will return the tx combination as valid/invalid
|
|
// this is not used as core consensus but reports only to user that his tx though in the blockchain is invalid
|
|
// a tx is valid, if it exist in a block which is not a side block
|
|
func (chain *Blockchain) IS_TX_Valid(txhash crypto.Hash) (valid_blid crypto.Hash, invalid_blid []crypto.Hash, valid bool) {
|
|
|
|
var tx_bytes []byte
|
|
var err error
|
|
|
|
if tx_bytes, err = chain.Store.Block_tx_store.ReadTX(txhash); err != nil {
|
|
return
|
|
}
|
|
|
|
var tx transaction.Transaction
|
|
if err = tx.DeserializeHeader(tx_bytes); err != nil {
|
|
return
|
|
}
|
|
|
|
blids, _ := chain.Store.Topo_store.binarySearchHeight(int64(tx.Height + 1))
|
|
|
|
var exist_list []crypto.Hash
|
|
|
|
for _, blid := range blids {
|
|
bl, err := chain.Load_BL_FROM_ID(blid)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, bltxhash := range bl.Tx_hashes {
|
|
if bltxhash == txhash {
|
|
exist_list = append(exist_list, blid)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, blid := range exist_list {
|
|
if chain.isblock_SideBlock(blid) {
|
|
invalid_blid = append(invalid_blid, blid)
|
|
} else {
|
|
valid_blid = blid
|
|
valid = true
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
/*
|
|
|
|
|
|
// runs the client protocol which includes the following operations
|
|
// if any TX are being duplicate or double-spend ignore them
|
|
// mark all the valid transactions as valid
|
|
// mark all invalid transactions as invalid
|
|
// calculate total fees based on valid TX
|
|
// we need NOT check ranges/ring signatures here, as they have been done already by earlier steps
|
|
func (chain *Blockchain) client_protocol(dbtx storage.DBTX, bl *block.Block, blid crypto.Hash, height int64, topoheight int64) (total_fees uint64) {
|
|
// run client protocol for all TXs
|
|
for i := range bl.Tx_hashes {
|
|
tx, err := chain.Load_TX_FROM_ID(dbtx, bl.Tx_hashes[i])
|
|
if err != nil {
|
|
panic(fmt.Errorf("Cannot load tx for %x err %s ", bl.Tx_hashes[i], err))
|
|
}
|
|
// mark TX found in this block also for explorer
|
|
chain.store_TX_in_Block(dbtx, blid, bl.Tx_hashes[i])
|
|
|
|
// check all key images as double spend, if double-spend detected mark invalid, else consider valid
|
|
if chain.Verify_Transaction_NonCoinbase_DoubleSpend_Check(dbtx, tx) {
|
|
|
|
chain.consume_keyimages(dbtx, tx, height) // mark key images as consumed
|
|
total_fees += tx.RctSignature.Get_TX_Fee()
|
|
|
|
chain.Store_TX_Height(dbtx, bl.Tx_hashes[i], topoheight) // link the tx with the topo height
|
|
|
|
//mark tx found in this block is valid
|
|
chain.mark_TX(dbtx, blid, bl.Tx_hashes[i], true)
|
|
|
|
} else { // TX is double spend or reincluded by 2 blocks simultaneously
|
|
rlog.Tracef(1,"Double spend TX is being ignored %s %s", blid, bl.Tx_hashes[i])
|
|
chain.mark_TX(dbtx, blid, bl.Tx_hashes[i], false)
|
|
}
|
|
}
|
|
|
|
return total_fees
|
|
}
|
|
|
|
// this undoes everything that is done by client protocol
|
|
// NOTE: this will have any effect, only if client protocol has been run on this block earlier
|
|
func (chain *Blockchain) client_protocol_reverse(dbtx storage.DBTX, bl *block.Block, blid crypto.Hash) {
|
|
// run client protocol for all TXs
|
|
for i := range bl.Tx_hashes {
|
|
tx, err := chain.Load_TX_FROM_ID(dbtx, bl.Tx_hashes[i])
|
|
if err != nil {
|
|
panic(fmt.Errorf("Cannot load tx for %x err %s ", bl.Tx_hashes[i], err))
|
|
}
|
|
// only the valid TX must be revoked
|
|
if chain.IS_TX_Valid(dbtx, blid, bl.Tx_hashes[i]) {
|
|
chain.revoke_keyimages(dbtx, tx) // mark key images as not used
|
|
|
|
chain.Store_TX_Height(dbtx, bl.Tx_hashes[i], -1) // unlink the tx with the topo height
|
|
|
|
//mark tx found in this block is invalid
|
|
chain.mark_TX(dbtx, blid, bl.Tx_hashes[i], false)
|
|
|
|
} else { // TX is double spend or reincluded by 2 blocks simultaneously
|
|
// invalid tx is related
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// scavanger for transactions from rusty/stale tips to reinsert them into pool
|
|
func (chain *Blockchain) transaction_scavenger(dbtx storage.DBTX, blid crypto.Hash) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
logger.Warnf("Recovered while transaction scavenging, Stack trace below ")
|
|
logger.Warnf("Stack trace \n%s", debug.Stack())
|
|
}
|
|
}()
|
|
|
|
logger.Debugf("scavenging transactions from blid %s", blid)
|
|
reachable_blocks := chain.BuildReachableBlocks(dbtx, []crypto.Hash{blid})
|
|
reachable_blocks[blid] = true // add self
|
|
for k, _ := range reachable_blocks {
|
|
if chain.Is_Block_Orphan(k) {
|
|
bl, err := chain.Load_BL_FROM_ID(dbtx, k)
|
|
if err == nil {
|
|
for i := range bl.Tx_hashes {
|
|
tx, err := chain.Load_TX_FROM_ID(dbtx, bl.Tx_hashes[i])
|
|
if err != nil {
|
|
rlog.Warnf("err while scavenging blid %s txid %s err %s", k, bl.Tx_hashes[i], err)
|
|
} else {
|
|
// add tx to pool, it will do whatever is necessarry
|
|
chain.Add_TX_To_Pool(tx)
|
|
}
|
|
}
|
|
} else {
|
|
rlog.Warnf("err while scavenging blid %s err %s", k, err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
*/
|
|
// Finds whether a block is orphan
|
|
// since we donot store any fields, we need to calculate/find the block as orphan
|
|
// using an algorithm
|
|
// if the block is NOT topo ordered , it is orphan/stale
|
|
func (chain *Blockchain) Is_Block_Orphan(hash crypto.Hash) bool {
|
|
return !chain.Is_Block_Topological_order(hash)
|
|
}
|
|
|
|
// this is used to find if a tx is orphan, YES orphan TX
|
|
// these can occur during when they lie only in a side block
|
|
// so the TX becomes orphan ( chances are less may be less that .000001 % but they are there)
|
|
// if a tx is not valid in any of the blocks, it has been mined it is orphan
|
|
func (chain *Blockchain) Is_TX_Orphan(hash crypto.Hash) (result bool) {
|
|
_, _, result = chain.IS_TX_Valid(hash)
|
|
return !result
|
|
}
|
|
|
|
// verifies whether we are lagging
|
|
// return true if we need resync
|
|
// returns false if we are good and resync is not required
|
|
func (chain *Blockchain) IsLagging(peer_cdiff *big.Int) bool {
|
|
|
|
our_diff := new(big.Int).SetInt64(0)
|
|
|
|
high_block, err := chain.Load_Block_Topological_order_at_index(chain.Load_TOPO_HEIGHT())
|
|
if err != nil {
|
|
return false
|
|
} else {
|
|
our_diff = chain.Load_Block_Cumulative_Difficulty(high_block)
|
|
}
|
|
rlog.Tracef(2, "P_cdiff %s cdiff %d our top block %s", peer_cdiff.String(), our_diff.String(), high_block)
|
|
|
|
if our_diff.Cmp(peer_cdiff) < 0 {
|
|
return true // peer's cumulative difficulty is more than ours , active resync
|
|
}
|
|
return false
|
|
}
|
|
|
|
// this function will rewind the chain from the topo height one block at a time
|
|
// this function also runs the client protocol in reverse and also deletes the block from the storage
|
|
func (chain *Blockchain) Rewind_Chain(rewind_count int) (result bool) {
|
|
defer chain.Initialise_Chain_From_DB()
|
|
|
|
chain.Lock()
|
|
defer chain.Unlock()
|
|
|
|
// we must till we reach a safe point
|
|
// safe point is point where a single block exists at specific height
|
|
// this may lead us to rewinding a it more
|
|
//safe := false
|
|
|
|
// TODO we must fix safeness using the stable calculation
|
|
|
|
top_block_topo_index := chain.Load_TOPO_HEIGHT()
|
|
rewinded := int64(0)
|
|
|
|
for { // rewind as many as possible
|
|
if top_block_topo_index-rewinded < 1 || rewinded >= int64(rewind_count) {
|
|
break
|
|
}
|
|
|
|
rewinded++
|
|
|
|
fmt.Printf("rewing 1 block top loop")
|
|
}
|
|
|
|
for { // rewinf till we reach a safe point
|
|
r, err := chain.Store.Topo_store.Read(top_block_topo_index - rewinded)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if chain.IsBlockSyncBlockHeight(r.BLOCK_ID) || r.Height == 0 {
|
|
break
|
|
}
|
|
|
|
fmt.Printf("rewing 1 block bottom loop %v %+v rewinded %d\n", chain.IsBlockSyncBlockHeight(r.BLOCK_ID), r, rewinded)
|
|
rewinded++
|
|
}
|
|
|
|
for i := int64(0); i != rewinded; i++ {
|
|
chain.Store.Topo_store.Clean(top_block_topo_index - i)
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// build reachability graph upto 2*config deeps to answer reachability queries
|
|
func (chain *Blockchain) buildReachability_internal(reachmap map[crypto.Hash]bool, blid crypto.Hash, level int) {
|
|
bl, err := chain.Load_BL_FROM_ID(blid)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
past := bl.Tips
|
|
reachmap[blid] = true // add self to reach map
|
|
|
|
if level >= int(2*config.STABLE_LIMIT) { // stop recursion must be more than checks in add complete block
|
|
return
|
|
}
|
|
for i := range past { // if no past == genesis return
|
|
if _, ok := reachmap[past[i]]; !ok { // process a node, only if has not been processed earlier
|
|
chain.buildReachability_internal(reachmap, past[i], level+1)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// build reachability graph upto 2*limit deeps to answer reachability queries
|
|
func (chain *Blockchain) buildReachability(blid crypto.Hash) map[crypto.Hash]bool {
|
|
reachmap := map[crypto.Hash]bool{}
|
|
chain.buildReachability_internal(reachmap, blid, 0)
|
|
return reachmap
|
|
}
|
|
|
|
// this is part of consensus rule, 2 tips cannot refer to their common parent
|
|
func (chain *Blockchain) VerifyNonReachability(bl *block.Block) bool {
|
|
|
|
reachmaps := make([]map[crypto.Hash]bool, len(bl.Tips), len(bl.Tips))
|
|
for i := range bl.Tips {
|
|
reachmaps[i] = chain.buildReachability(bl.Tips[i])
|
|
}
|
|
|
|
// bruteforce all reachability combinations, max possible 3x3 = 9 combinations
|
|
for i := range bl.Tips {
|
|
for j := range bl.Tips {
|
|
if i == j { // avoid self test
|
|
continue
|
|
}
|
|
|
|
if _, ok := reachmaps[j][bl.Tips[i]]; ok { // if a tip can be referenced as another's past, this is not a tip , probably malicious, discard block
|
|
return false
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// used in the difficulty calculation for consensus and while scavenging
|
|
func (chain *Blockchain) BuildReachableBlocks(tips []crypto.Hash) map[crypto.Hash]bool {
|
|
reachblocks := map[crypto.Hash]bool{} // contains a list of all reachable blocks
|
|
for i := range tips {
|
|
reachmap := chain.buildReachability(tips[i])
|
|
for k, _ := range reachmap {
|
|
reachblocks[k] = true // build unique block list
|
|
}
|
|
}
|
|
return reachblocks
|
|
}
|
|
|
|
// this is part of consensus rule, reachable blocks cannot have keyimages collision with new blocks
|
|
// this is to avoid dishonest miners including dead transactions
|
|
//
|
|
func (chain *Blockchain) BuildReachabilityNonces(bl *block.Block) map[crypto.Hash]bool {
|
|
|
|
nonce_reach_map := map[crypto.Hash]bool{}
|
|
reachblocks := map[crypto.Hash]bool{} // contains a list of all reachable blocks
|
|
for i := range bl.Tips {
|
|
reachmap := chain.buildReachability(bl.Tips[i])
|
|
for k, _ := range reachmap {
|
|
reachblocks[k] = true // build unique block list
|
|
}
|
|
}
|
|
|
|
// load all blocks and process their TX as per client protocol
|
|
for blid, _ := range reachblocks {
|
|
bl, err := chain.Load_BL_FROM_ID(blid)
|
|
if err != nil {
|
|
panic(fmt.Errorf("Cannot load block for %s err %s", blid, err))
|
|
}
|
|
|
|
for i := 0; i < len(bl.Tx_hashes); i++ { // load all tx one by one, skipping as per client_protocol
|
|
|
|
tx_bytes, err := chain.Store.Block_tx_store.ReadTX(bl.Tx_hashes[i])
|
|
if err != nil {
|
|
panic(fmt.Errorf("Cannot load tx for %s err %s", bl.Tx_hashes[i], err))
|
|
}
|
|
|
|
var tx transaction.Transaction
|
|
if err = tx.DeserializeHeader(tx_bytes); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// tx has been loaded, now lets get the nonce
|
|
nonce_reach_map[tx.Proof.Nonce()] = true // add element to map for next check
|
|
}
|
|
}
|
|
return nonce_reach_map
|
|
}
|
|
|
|
// sync blocks have the following specific property
|
|
// 1) the block is singleton at this height
|
|
// basically the condition allow us to confirm weight of future blocks with reference to sync blocks
|
|
// these are the one who settle the chain and guarantee it
|
|
func (chain *Blockchain) IsBlockSyncBlockHeight(blid crypto.Hash) bool {
|
|
return chain.IsBlockSyncBlockHeightSpecific(blid, chain.Get_Height())
|
|
}
|
|
|
|
func (chain *Blockchain) IsBlockSyncBlockHeightSpecific(blid crypto.Hash, chain_height int64) bool {
|
|
|
|
// TODO make sure that block exist
|
|
height := chain.Load_Height_for_BL_ID(blid)
|
|
if height == 0 { // genesis is always a sync block
|
|
return true
|
|
}
|
|
|
|
// top blocks are always considered unstable
|
|
if (height + config.STABLE_LIMIT) > chain_height {
|
|
return false
|
|
}
|
|
|
|
// if block is not ordered, it can never be sync block
|
|
if !chain.Is_Block_Topological_order(blid) {
|
|
return false
|
|
}
|
|
|
|
blocks := chain.Get_Blocks_At_Height(height)
|
|
|
|
if len(blocks) == 0 && height != 0 { // this should NOT occur
|
|
panic("No block exists at this height, not possible")
|
|
}
|
|
|
|
// if len(blocks) == 1 { // ideal blockchain case, it is a sync block
|
|
// return true
|
|
// }
|
|
|
|
// check whether single block exists in the TOPO order index, if no we are NOT a sync block
|
|
|
|
// we are here means we have one oor more block
|
|
blocks_in_main_chain := 0
|
|
for i := range blocks {
|
|
if chain.Is_Block_Topological_order(blocks[i]) {
|
|
blocks_in_main_chain++
|
|
if blocks_in_main_chain >= 2 {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
// we are here if we only have one block in topological order, others are dumped/rejected blocks
|
|
|
|
// collect all blocks of past LIMIT heights
|
|
var preblocks []crypto.Hash
|
|
for i := height - 1; i >= (height-config.STABLE_LIMIT) && i != 0; i-- {
|
|
blocks := chain.Get_Blocks_At_Height(i)
|
|
for j := range blocks { //TODO BUG BUG BUG we need to make sure only main chain blocks are considered
|
|
preblocks = append(preblocks, blocks[j])
|
|
}
|
|
}
|
|
|
|
// we need to find a common base to compare them, otherwise comparision is futile due to duplication
|
|
sync_block_cumulative_difficulty := chain.Load_Block_Cumulative_Difficulty(blid) //+ chain.Load_Block_Difficulty(blid)
|
|
|
|
// if any of the blocks has a cumulative difficulty more than sync block, this situation affects consensus, so mitigate it
|
|
for i := range preblocks {
|
|
cumulative_difficulty := chain.Load_Block_Cumulative_Difficulty(preblocks[i]) // + chain.Load_Block_Difficulty(preblocks[i])
|
|
|
|
//if cumulative_difficulty >= sync_block_cumulative_difficulty {
|
|
if cumulative_difficulty.Cmp(sync_block_cumulative_difficulty) >= 0 {
|
|
rlog.Warnf("Mitigating CONSENSUS issue on block %s height %d child %s cdiff %d sync block cdiff %d", blid, height, preblocks[i], cumulative_difficulty, sync_block_cumulative_difficulty)
|
|
return false
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// key is string of blid and appendded chain height
|
|
var tipbase_cache, _ = hashicorp_lru.New(10240)
|
|
|
|
// base of a tip is last known sync point
|
|
// weight of bases in mentioned in term of height
|
|
// this must not employ any cache
|
|
func (chain *Blockchain) FindTipBase(blid crypto.Hash, chain_height int64) (bs BlockScore) {
|
|
|
|
// see if cache contains it
|
|
if bsi, ok := tipbase_cache.Get(fmt.Sprintf("%s%d", blid, chain_height)); ok {
|
|
bs = bsi.(BlockScore)
|
|
return bs
|
|
}
|
|
|
|
defer func() { // capture return value of bs to cache
|
|
z := bs
|
|
tipbase_cache.Add(fmt.Sprintf("%s%d", blid, chain_height), z)
|
|
}()
|
|
|
|
// if we are genesis return genesis block as base
|
|
|
|
tips := chain.Get_Block_Past(blid)
|
|
if len(tips) == 0 {
|
|
gbl := Generate_Genesis_Block()
|
|
bs = BlockScore{gbl.GetHash(), 0, nil}
|
|
return
|
|
}
|
|
|
|
bases := make([]BlockScore, len(tips), len(tips))
|
|
|
|
for i := range tips {
|
|
if chain.IsBlockSyncBlockHeightSpecific(tips[i], chain_height) {
|
|
// rlog.Tracef(2, "SYNC block %s", tips[i])
|
|
bs = BlockScore{tips[i], chain.Load_Height_for_BL_ID(tips[i]), nil}
|
|
return
|
|
}
|
|
bases[i] = chain.FindTipBase(tips[i], chain_height)
|
|
}
|
|
|
|
sort_ascending_by_height(bases)
|
|
|
|
// logger.Infof("return BASE %s",bases[0])
|
|
bs = bases[0]
|
|
return bs
|
|
}
|
|
|
|
// this will find the sum of work done ( skipping any repetive nodes )
|
|
// all the information is privided in unique_map
|
|
func (chain *Blockchain) FindTipWorkScore_internal(unique_map map[crypto.Hash]*big.Int, blid crypto.Hash, base crypto.Hash, base_height int64) {
|
|
|
|
tips := chain.Get_Block_Past(blid)
|
|
|
|
for i := range tips {
|
|
if _, ok := unique_map[tips[i]]; !ok {
|
|
|
|
ordered := chain.Is_Block_Topological_order(tips[i])
|
|
|
|
if !ordered {
|
|
chain.FindTipWorkScore_internal(unique_map, tips[i], base, base_height) // recursively process any nodes
|
|
//logger.Infof("IBlock is not ordered %s", tips[i])
|
|
} else if ordered && chain.Load_Block_Topological_order(tips[i]) >= chain.Load_Block_Topological_order(base) {
|
|
chain.FindTipWorkScore_internal(unique_map, tips[i], base, base_height) // recursively process any nodes
|
|
|
|
//logger.Infof("IBlock ordered %s %d %d", tips[i],chain.Load_Block_Topological_order(tips[i]), chain.Load_Block_Topological_order(base) )
|
|
}
|
|
}
|
|
}
|
|
|
|
unique_map[blid] = chain.Load_Block_Difficulty(blid)
|
|
|
|
}
|
|
|
|
type cachekey struct {
|
|
blid crypto.Hash
|
|
base crypto.Hash
|
|
base_height int64
|
|
}
|
|
|
|
// find the score of the tip in reference to a base (NOTE: base is always a sync block otherwise results will be wrong )
|
|
func (chain *Blockchain) FindTipWorkScore(blid crypto.Hash, base crypto.Hash, base_height int64) (map[crypto.Hash]*big.Int, *big.Int) {
|
|
|
|
//logger.Infof("BASE %s",base)
|
|
if tmp_map_i, ok := chain.lrucache_workscore.Get(cachekey{blid, base, base_height}); ok {
|
|
work_score := tmp_map_i.(map[crypto.Hash]*big.Int)
|
|
|
|
map_copy := map[crypto.Hash]*big.Int{}
|
|
score := new(big.Int).SetInt64(0)
|
|
for k, v := range work_score {
|
|
map_copy[k] = v
|
|
score.Add(score, v)
|
|
}
|
|
return map_copy, score
|
|
}
|
|
|
|
bl, err := chain.Load_BL_FROM_ID(blid)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("Block NOT found %s", blid))
|
|
}
|
|
unique_map := map[crypto.Hash]*big.Int{}
|
|
|
|
for i := range bl.Tips {
|
|
if _, ok := unique_map[bl.Tips[i]]; !ok {
|
|
//if chain.Load_Height_for_BL_ID(bl.Tips[i]) > base_height {
|
|
// chain.FindTipWorkScore_internal(unique_map,bl.Tips[i],base,base_height) // recursively process any nodes
|
|
//}
|
|
|
|
ordered := chain.Is_Block_Topological_order(bl.Tips[i])
|
|
if !ordered {
|
|
chain.FindTipWorkScore_internal(unique_map, bl.Tips[i], base, base_height) // recursively process any nodes
|
|
// logger.Infof("Block is not ordered %s", bl.Tips[i])
|
|
} else if ordered && chain.Load_Block_Topological_order(bl.Tips[i]) >= chain.Load_Block_Topological_order(base) {
|
|
chain.FindTipWorkScore_internal(unique_map, bl.Tips[i], base, base_height) // recursively process any nodes
|
|
|
|
// logger.Infof("Block ordered %s %d %d", bl.Tips[i],chain.Load_Block_Topological_order(bl.Tips[i]), chain.Load_Block_Topological_order(base) )
|
|
}
|
|
}
|
|
}
|
|
|
|
if base != blid {
|
|
unique_map[base] = chain.Load_Block_Cumulative_Difficulty(base)
|
|
}
|
|
|
|
unique_map[blid] = chain.Load_Block_Difficulty(blid)
|
|
score := new(big.Int).SetInt64(0)
|
|
for _, v := range unique_map {
|
|
score.Add(score, v)
|
|
}
|
|
|
|
//set in cache, save a copy in cache
|
|
{
|
|
map_copy := map[crypto.Hash]*big.Int{}
|
|
for k, v := range unique_map {
|
|
map_copy[k] = v
|
|
}
|
|
chain.lrucache_workscore.Add(cachekey{blid, base, base_height}, map_copy)
|
|
}
|
|
|
|
return unique_map, score
|
|
|
|
}
|
|
|
|
// find the score of the tip in reference to a base (NOTE: base is always a sync block otherwise results will be wrong )
|
|
func (chain *Blockchain) FindTipWorkScore_duringsave(bl *block.Block, diff *big.Int, base crypto.Hash, base_height int64) (map[crypto.Hash]*big.Int, *big.Int) {
|
|
|
|
blid := bl.GetHash()
|
|
//logger.Infof("BASE %s",base)
|
|
if tmp_map_i, ok := chain.lrucache_workscore.Get(cachekey{blid, base, base_height}); ok {
|
|
work_score := tmp_map_i.(map[crypto.Hash]*big.Int)
|
|
|
|
map_copy := map[crypto.Hash]*big.Int{}
|
|
score := new(big.Int).SetInt64(0)
|
|
for k, v := range work_score {
|
|
map_copy[k] = v
|
|
score.Add(score, v)
|
|
}
|
|
return map_copy, score
|
|
}
|
|
|
|
unique_map := map[crypto.Hash]*big.Int{}
|
|
|
|
for i := range bl.Tips {
|
|
if _, ok := unique_map[bl.Tips[i]]; !ok {
|
|
//if chain.Load_Height_for_BL_ID(bl.Tips[i]) > base_height {
|
|
// chain.FindTipWorkScore_internal(unique_map,bl.Tips[i],base,base_height) // recursively process any nodes
|
|
//}
|
|
|
|
ordered := chain.Is_Block_Topological_order(bl.Tips[i])
|
|
if !ordered {
|
|
chain.FindTipWorkScore_internal(unique_map, bl.Tips[i], base, base_height) // recursively process any nodes
|
|
// logger.Infof("Block is not ordered %s", bl.Tips[i])
|
|
} else if ordered && chain.Load_Block_Topological_order(bl.Tips[i]) >= chain.Load_Block_Topological_order(base) {
|
|
chain.FindTipWorkScore_internal(unique_map, bl.Tips[i], base, base_height) // recursively process any nodes
|
|
|
|
// logger.Infof("Block ordered %s %d %d", bl.Tips[i],chain.Load_Block_Topological_order(bl.Tips[i]), chain.Load_Block_Topological_order(base) )
|
|
}
|
|
}
|
|
}
|
|
|
|
if base != blid {
|
|
unique_map[base] = chain.Load_Block_Cumulative_Difficulty(base)
|
|
}
|
|
|
|
unique_map[blid] = new(big.Int).Set(diff) // use whatever diff was supplied
|
|
score := new(big.Int).SetInt64(0)
|
|
for _, v := range unique_map {
|
|
score.Add(score, v)
|
|
}
|
|
|
|
//set in cache, save a copy in cache
|
|
{
|
|
map_copy := map[crypto.Hash]*big.Int{}
|
|
for k, v := range unique_map {
|
|
map_copy[k] = v
|
|
}
|
|
chain.lrucache_workscore.Add(cachekey{blid, base, base_height}, map_copy)
|
|
}
|
|
|
|
return unique_map, score
|
|
|
|
}
|
|
|
|
// this function finds a common base which can be used to compare tips
|
|
// weight is replace by height
|
|
func (chain *Blockchain) find_common_base(tips []crypto.Hash) (base crypto.Hash, base_height int64) {
|
|
|
|
scores := make([]BlockScore, len(tips), len(tips))
|
|
|
|
// var base crypto.Hash
|
|
var best_height int64
|
|
for i := range tips {
|
|
tip_height := chain.Load_Height_for_BL_ID(tips[i])
|
|
if tip_height > best_height {
|
|
best_height = tip_height
|
|
}
|
|
}
|
|
|
|
for i := range tips {
|
|
scores[i] = chain.FindTipBase(tips[i], best_height) // we should chose the lowest weight
|
|
scores[i].Height = chain.Load_Height_for_BL_ID(scores[i].BLID)
|
|
}
|
|
// base is the lowest height
|
|
sort_ascending_by_height(scores)
|
|
|
|
base = scores[0].BLID
|
|
base_height = scores[0].Height
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// this function finds a common base which can be used to compare tips based on cumulative difficulty
|
|
func (chain *Blockchain) find_best_tip(tips []crypto.Hash, base crypto.Hash, base_height int64) (best crypto.Hash) {
|
|
|
|
tips_scores := make([]BlockScore, len(tips), len(tips))
|
|
|
|
for i := range tips {
|
|
tips_scores[i].BLID = tips[i] // we should chose the lowest weight
|
|
_, tips_scores[i].Cumulative_Difficulty = chain.FindTipWorkScore(tips[i], base, base_height)
|
|
}
|
|
|
|
sort_descending_by_cumulative_difficulty(tips_scores)
|
|
|
|
best = tips_scores[0].BLID
|
|
// base_height = scores[0].Weight
|
|
|
|
return best
|
|
|
|
}
|
|
|
|
func (chain *Blockchain) calculate_mainchain_distance_internal_recursive(unique_map map[crypto.Hash]int64, blid crypto.Hash) {
|
|
tips := chain.Get_Block_Past(blid)
|
|
for i := range tips {
|
|
ordered := chain.Is_Block_Topological_order(tips[i])
|
|
if ordered {
|
|
unique_map[tips[i]] = chain.Load_Height_for_BL_ID(tips[i])
|
|
} else {
|
|
chain.calculate_mainchain_distance_internal_recursive(unique_map, tips[i]) // recursively process any nodes
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// NOTE: some of the past may not be in the main chain right now and need to be travelled recursively
|
|
// distance is number of hops to find a node, which is itself
|
|
func (chain *Blockchain) calculate_mainchain_distance(blid crypto.Hash) int64 {
|
|
|
|
unique_map := map[crypto.Hash]int64{}
|
|
//tips := chain.Get_Block_Past(dbtx, blid)
|
|
|
|
//fmt.Printf("tips %+v \n", tips)
|
|
|
|
// if the block is already in order, no need to look back
|
|
|
|
ordered := chain.Is_Block_Topological_order(blid)
|
|
if ordered {
|
|
unique_map[blid] = chain.Load_Height_for_BL_ID(blid)
|
|
} else {
|
|
chain.calculate_mainchain_distance_internal_recursive(unique_map, blid)
|
|
}
|
|
|
|
//for i := range tips {
|
|
//}
|
|
|
|
//fmt.Printf("unique_map %+v \n", unique_map)
|
|
|
|
lowest_height := int64(0x7FFFFFFFFFFFFFFF) // max possible
|
|
// now we need to find the lowest height
|
|
for k, v := range unique_map {
|
|
_ = k
|
|
if lowest_height >= v {
|
|
lowest_height = v
|
|
}
|
|
}
|
|
|
|
return int64(lowest_height)
|
|
}
|
|
|
|
// converts a DAG's partial order into a full order, this function is recursive
|
|
// generate full order should be only callled on the basis of base blocks which satisfy sync block properties as follows
|
|
// generate full order is called on maximum weight tip at every tip change
|
|
// blocks are ordered recursively, till we find a find a block which is already in the chain
|
|
func (chain *Blockchain) Generate_Full_Order(blid crypto.Hash, base crypto.Hash, base_height int64, level int) (order_bucket []crypto.Hash) {
|
|
|
|
// return from cache if possible
|
|
if tmp_order, ok := chain.lrucache_fullorder.Get(cachekey{blid, base, base_height}); ok {
|
|
order := tmp_order.([]crypto.Hash)
|
|
order_bucket = make([]crypto.Hash, len(order), len(order))
|
|
copy(order_bucket, order[0:])
|
|
return
|
|
}
|
|
|
|
bl, err := chain.Load_BL_FROM_ID(blid)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("Block NOT found %s", blid))
|
|
}
|
|
|
|
if len(bl.Tips) == 0 {
|
|
gbl := Generate_Genesis_Block()
|
|
order_bucket = append(order_bucket, gbl.GetHash())
|
|
return
|
|
}
|
|
|
|
// if the block has been previously ordered, stop the recursion and return it as base
|
|
//if chain.Is_Block_Topological_order(blid){
|
|
if blid == base {
|
|
order_bucket = append(order_bucket, blid)
|
|
// logger.Infof("Generate order base reached base %s", base)
|
|
return
|
|
}
|
|
|
|
// we need to order previous tips first
|
|
var tips_scores []BlockScore
|
|
//tips_scores := make([]BlockScore,len(bl.Tips),len(bl.Tips))
|
|
|
|
node_maps := map[crypto.Hash]map[crypto.Hash]*big.Int{}
|
|
_ = node_maps
|
|
for i := range bl.Tips {
|
|
|
|
ordered := chain.Is_Block_Topological_order(bl.Tips[i])
|
|
|
|
if !ordered {
|
|
var score BlockScore
|
|
score.BLID = bl.Tips[i]
|
|
//node_maps[bl.Tips[i]], score.Weight = chain.FindTipWorkScore(bl.Tips[i],base,base_height)
|
|
score.Cumulative_Difficulty = chain.Load_Block_Cumulative_Difficulty(bl.Tips[i])
|
|
|
|
tips_scores = append(tips_scores, score)
|
|
|
|
} else if ordered && chain.Load_Block_Topological_order(bl.Tips[i]) >= chain.Load_Block_Topological_order(base) {
|
|
|
|
// logger.Infof("Generate order topo order wrt base %d %d", chain.Load_Block_Topological_order(dbtx,bl.Tips[i]), chain.Load_Block_Topological_order(dbtx,base))
|
|
var score BlockScore
|
|
score.BLID = bl.Tips[i]
|
|
|
|
//score.Weight = chain.Load_Block_Cumulative_Difficulty(bl.Tips[i])
|
|
score.Cumulative_Difficulty = chain.Load_Block_Cumulative_Difficulty(bl.Tips[i])
|
|
|
|
tips_scores = append(tips_scores, score)
|
|
}
|
|
|
|
}
|
|
|
|
sort_descending_by_cumulative_difficulty(tips_scores)
|
|
|
|
// now we must add the nodes in the topographical order
|
|
|
|
for i := range tips_scores {
|
|
tmp_bucket := chain.Generate_Full_Order(tips_scores[i].BLID, base, base_height, level+1)
|
|
for j := range tmp_bucket {
|
|
//only process if this block is unsettled
|
|
//if !chain.IsBlockSettled(tmp_bucket[j]) {
|
|
// if order is already decided, do not order it again
|
|
if !sliceExists(order_bucket, tmp_bucket[j]) {
|
|
order_bucket = append(order_bucket, tmp_bucket[j])
|
|
}
|
|
//}
|
|
}
|
|
}
|
|
// add self to the end, since all past nodes have been ordered
|
|
order_bucket = append(order_bucket, blid)
|
|
|
|
// logger.Infof("Generate Order %s %+v %+v", blid , order_bucket, tips_scores)
|
|
|
|
//set in cache, save a copy in cache
|
|
{
|
|
order_copy := make([]crypto.Hash, len(order_bucket), len(order_bucket))
|
|
copy(order_copy, order_bucket[0:])
|
|
|
|
chain.lrucache_fullorder.Add(cachekey{blid, base, base_height}, order_copy)
|
|
}
|
|
|
|
if level == 0 {
|
|
//logger.Warnf("generating full order for block %s %d", blid, level)
|
|
//
|
|
// for i := range order_bucket{
|
|
// logger.Infof("%2d %s", i, order_bucket[i])
|
|
// }
|
|
|
|
//logger.Warnf("generating full order finished")
|
|
}
|
|
return
|
|
}
|
|
|
|
// tells whether the hash already exists in slice
|
|
func sliceExists(slice []crypto.Hash, hash crypto.Hash) bool {
|
|
for i := range slice {
|
|
if slice[i] == hash {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
/*
|
|
|
|
|
|
var node_map = map[crypto.Hash]bool{}
|
|
|
|
func collect_nodes(chain *Blockchain, dbtx storage.DBTX, blid crypto.Hash) {
|
|
future := chain.Get_Block_Future(dbtx, blid)
|
|
for i := range future {
|
|
//node_map[future[i]]=true
|
|
|
|
if _, ok := node_map[future[i]]; !ok {
|
|
collect_nodes(chain, dbtx, future[i]) // recursive add node
|
|
}
|
|
}
|
|
|
|
node_map[blid] = true
|
|
|
|
}
|
|
*/
|
|
|
|
/*
|
|
func writenode(chain *Blockchain, dbtx storage.DBTX, w *bufio.Writer, blid crypto.Hash) { // process a node, recursively
|
|
|
|
collect_nodes(chain, dbtx, blid)
|
|
|
|
sync_blocks := map[crypto.Hash]uint64{}
|
|
|
|
for k, _ := range node_map {
|
|
if chain.IsBlockSyncBlockHeight(dbtx, k) {
|
|
// sync_blocks = append(sync_blocks,
|
|
sync_blocks[k] = uint64(chain.Load_Height_for_BL_ID(dbtx, k))
|
|
}
|
|
}
|
|
|
|
w.WriteString(fmt.Sprintf("node [ fontsize=12 style=filled ]\n{\n"))
|
|
for k := range node_map {
|
|
|
|
//anticone := chain.Get_AntiCone_Unsettled(k)
|
|
|
|
color := "white"
|
|
|
|
if chain.IsBlockSyncBlockHeight(dbtx, k) {
|
|
color = "green"
|
|
}
|
|
|
|
|
|
|
|
|
|
//w.WriteString(fmt.Sprintf("L%s [ fillcolor=%s label = \"%s %d height %d score %d stored %d order %d\" ];\n", k.String(), color, k.String(), 0, chain.Load_Height_for_BL_ID(dbtx, k), cumulative_difficulty, chain.Load_Block_Cumulative_Difficulty(dbtx, k), chain.Load_Block_Topological_order(dbtx, k)))
|
|
w.WriteString(fmt.Sprintf("L%s [ fillcolor=%s label = \"%s %d height %d score %d stored %d order %d\" ];\n", k.String(), color, k.String(), 0, chain.Load_Height_for_BL_ID(dbtx, k), 0, chain.Load_Block_Cumulative_Difficulty(dbtx, k), chain.Load_Block_Topological_order(dbtx, k)))
|
|
}
|
|
w.WriteString(fmt.Sprintf("}\n"))
|
|
|
|
// now dump the interconnections
|
|
for k := range node_map {
|
|
future := chain.Get_Block_Future(dbtx, k)
|
|
for i := range future {
|
|
w.WriteString(fmt.Sprintf("L%s -> L%s ;\n", k.String(), future[i].String()))
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
func WriteBlockChainTree(chain *Blockchain, filename string) (err error) {
|
|
|
|
dbtx, err := chain.store.BeginTX(false)
|
|
if err != nil {
|
|
logger.Warnf("Could NOT add block to chain. Error opening writable TX, err %s", err)
|
|
return
|
|
}
|
|
|
|
defer dbtx.Rollback()
|
|
|
|
f, err := os.Create(filename)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer f.Close()
|
|
|
|
w := bufio.NewWriter(f)
|
|
defer w.Flush()
|
|
w.WriteString("digraph dero_blockchain_graph { \n")
|
|
|
|
blid, err := chain.Load_Block_Topological_order_at_index(nil, 158800)
|
|
if err != nil {
|
|
logger.Warnf("Cannot get block at topoheight %d err: %s", 158800, err)
|
|
return
|
|
}
|
|
|
|
writenode(chain, dbtx, w, blid)
|
|
//g := Generate_Genesis_Block()
|
|
//writenode(chain, dbtx, w, g.GetHash())
|
|
|
|
w.WriteString("}\n")
|
|
|
|
return
|
|
}
|
|
|
|
*/
|