1540 lines
52 KiB
Go
Raw Normal View History

2021-11-22 16:05:02 +00:00
// 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 "bytes"
import "runtime/debug"
import "strings"
import "runtime"
import "context"
import "golang.org/x/crypto/sha3"
import "golang.org/x/sync/semaphore"
import "github.com/go-logr/logr"
import "sync/atomic"
import "github.com/hashicorp/golang-lru"
import "github.com/deroproject/derohe/rpc"
import "github.com/deroproject/derohe/config"
import "github.com/deroproject/derohe/cryptography/crypto"
import "github.com/deroproject/derohe/errormsg"
import "github.com/deroproject/derohe/metrics"
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/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
Top_ID crypto.Hash // id of the top block
Pruned int64 // until where the chain has been pruned
MiniBlocks *block.MiniBlocksCollection // used for consensus
Tips map[crypto.Hash]crypto.Hash // current tips
mining_blocks_cache *lru.Cache // used to cache blocks which have been supplied to mining
cache_IsMiniblockPowValid *lru.Cache // used to cache mini blocks pow test result
cache_IsNonceValidTips *lru.Cache // used to cache nonce tests on specific tips
cache_IsAddressHashValid *lru.Cache // used to cache some outputs
cache_Get_Difficulty_At_Tips *lru.Cache // used to cache some outputs
cache_BlockPast *lru.Cache // used to cache a blocks past
cache_BlockHeight *lru.Cache // used to cache a blocks past
2021-11-24 09:20:51 +00:00
cache_VersionMerkle *lru.Cache // used to cache a versions merkle root
2021-11-22 16:05:02 +00:00
integrator_address rpc.Address // integrator rewards will be given to this address
cache_enabled bool // enables all cache, based on ENV DISABLE_CACHE
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
simulator bool // is simulator mode
P2P_Block_Relayer func(*block.Complete_Block, uint64) // tell p2p to broadcast any block this daemon hash found
2021-11-27 04:25:17 +00:00
P2P_MiniBlock_Relayer func(mbl block.MiniBlock, peerid uint64)
2021-11-22 16:05:02 +00:00
RPC_NotifyNewBlock *sync.Cond // used to notify rpc that a new block has been found
RPC_NotifyHeightChanged *sync.Cond // used to notify rpc that chain height has changed due to addition of block
RPC_NotifyNewMiniBlock *sync.Cond // used to notify rpc that a new mini block has been found
Sync bool // whether the sync is active, used while bootstrapping
sync.RWMutex
}
var logger logr.Logger = logr.Discard() // default discard all logs
// 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
logger = globals.Logger.WithName("CORE")
logger.V(1).Info("Initialising")
if err = chain.Store.Initialize(params); err != nil {
return nil, err
}
chain.Tips = map[crypto.Hash]crypto.Hash{}
chain.MiniBlocks = block.CreateMiniBlockCollection()
var addr *rpc.Address
if params["--integrator-address"] == nil {
if addr, err = rpc.NewAddress(strings.TrimSpace(globals.Config.Dev_Address)); err != nil {
return nil, err
}
} else {
if addr, err = rpc.NewAddress(strings.TrimSpace(params["--integrator-address"].(string))); err != nil {
return nil, err
}
}
chain.integrator_address = *addr
logger.Info("will use", "integrator_address", chain.integrator_address.String())
if chain.cache_IsMiniblockPowValid, err = lru.New(8192); err != nil { // temporary cache for miniblock difficulty
return nil, err
}
if chain.cache_Get_Difficulty_At_Tips, err = lru.New(8192); err != nil { // temporary cache for difficulty
return nil, err
}
if chain.cache_IsNonceValidTips, err = lru.New(100 * 1024); err != nil { // temporary cache for nonce checks
return nil, err
}
if chain.cache_IsAddressHashValid, err = lru.New(100 * 1024); err != nil { // temporary cache for valid address
return nil, err
}
if chain.mining_blocks_cache, err = lru.New(256); err != nil { // temporary cache for miniing blocks
return nil, err
}
2021-11-24 09:20:51 +00:00
if chain.cache_BlockPast, err = lru.New(1024); err != nil { // temporary cache for a blocks past
2021-11-22 16:05:02 +00:00
return nil, err
}
2021-11-24 09:20:51 +00:00
if chain.cache_BlockHeight, err = lru.New(10 * 1024); err != nil { // temporary cache for a blocks height
return nil, err
}
if chain.cache_VersionMerkle, err = lru.New(1024); err != nil { // temporary cache for a snapshot version
2021-11-22 16:05:02 +00:00
return nil, err
}
chain.cache_enabled = os.Getenv("DISABLE_CACHE") == "" // disable cache if the environ var is set
if !chain.cache_enabled {
logger.Info("All caching except mining jobs will be disabled")
}
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
if chain.Mempool, err = mempool.Init_Mempool(params); err != nil {
return nil, err
}
if chain.Regpool, err = regpool.Init_Regpool(params); err != nil {
return nil, err
}
chain.RPC_NotifyNewBlock = sync.NewCond(&sync.Mutex{}) // used by dero daemon to notify all websockets that new block has arrived
chain.RPC_NotifyHeightChanged = sync.NewCond(&sync.Mutex{}) // used by dero daemon to notify all websockets that chain height has changed
chain.RPC_NotifyNewMiniBlock = sync.NewCond(&sync.Mutex{}) // used by dero daemon to notify all websockets that new miniblock has arrived
if !chain.Store.IsBalancesIntialized() {
logger.Info("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.Error(err, "Failed to add genesis block, we can no longer continue.")
return nil, err
}
}
init_hard_forks(params) // hard forks must be initialized asap
chain.Initialise_Chain_From_DB() // load the chain from the disk
if chain.Pruned >= 1 {
logger.Info("Chain Pruned till", "topoheight", chain.Pruned)
}
metrics.Version = config.Version.String()
go metrics.Dump_metrics_data_directly(logger, globals.Arguments["--node-tag"]) // enable metrics if someone needs them
chain.Sync = true
if chain.Get_Height() <= 1 {
if globals.Arguments["--fastsync"] != nil && globals.Arguments["--fastsync"].(bool) {
chain.Sync = !globals.Arguments["--fastsync"].(bool)
}
}
atomic.AddUint32(&globals.Subsystem_Active, 1) // increment subsystem
globals.Cron.AddFunc("@every 360s", clean_up_valid_cache) // cleanup valid tx cache
globals.Cron.AddFunc("@every 60s", func() { // mempool house keeping
stable_height := int64(0)
if r := recover(); r != nil {
logger.Error(nil, "Mempool House Keeping triggered panic", "r", r, "height", stable_height)
}
stable_height = chain.Get_Stable_Height()
// give mempool an oppurtunity to clean up tx, but only if they are not mined
chain.Mempool.HouseKeeping(uint64(stable_height))
top_block_topo_index := chain.Load_TOPO_HEIGHT()
2021-11-24 09:20:51 +00:00
if top_block_topo_index < 4 {
2021-11-22 16:05:02 +00:00
return
}
2021-11-24 09:20:51 +00:00
top_block_topo_index -= 4
2021-11-22 16:05:02 +00:00
blid, err := chain.Load_Block_Topological_order_at_index(top_block_topo_index)
if err != nil {
panic(err)
}
record_version, err := chain.ReadBlockSnapshotVersion(blid)
if err != nil {
panic(err)
}
// give regpool a chance to register
if ss, err := chain.Store.Balance_store.LoadSnapshot(record_version); err == nil {
if balance_tree, err := ss.GetTree(config.BALANCE_TREE); err == nil {
chain.Regpool.HouseKeeping(uint64(stable_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 &chain, nil
}
// return integrator address
func (chain *Blockchain) IntegratorAddress() rpc.Address {
return chain.integrator_address
}
// 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()
chain.Pruned = chain.LocatePruneTopo()
// 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
logger.V(1).Info("Reloaded Chain from disk", "Tips", chain.Tips, "Height", 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.Info("Stopping Blockchain")
//chain.Store.Shutdown()
atomic.AddUint32(&globals.Subsystem_Active, ^uint32(0)) // this decrement 1 fom subsystem
logger.Info("Stopped Blockchain")
}
// 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()
defer globals.Recover(1)
bl := cbl.Bl // small pointer to block
block_hash = bl.GetHash()
block_logger := logger.WithName(fmt.Sprintf("blid_%s", block_hash)).V(1)
for k := range chain.Tips { // very fast path
if block_hash == k {
return errormsg.ErrAlreadyExists, false // block already in chain skipping it
}
}
// check if block already exist skip it
if chain.Is_Block_Topological_order(block_hash) {
return errormsg.ErrAlreadyExists, false // block already in chain skipping it
}
result = false
height_changed := false
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.V(1).Error(nil, "Recovered while adding new block", "blid", block_hash, "r", r, "stack", fmt.Sprintf("%s", string(debug.Stack())))
result = false
err = errormsg.ErrPanic
}
if result == true { // block was successfully added, commit it atomically
logger.V(2).Info("Block successfully accepted by chain", "blid", block_hash.String(), "err", err)
// gracefully try to instrument
func() {
defer func() {
if r := recover(); r != nil {
logger.V(1).Error(nil, "Recovered while instrumenting", "r", r, "stack", debug.Stack())
}
}()
metrics.Set.GetOrCreateCounter("blockchain_tx_total").Add(len(cbl.Bl.Tx_hashes))
metrics.Set.GetOrCreateHistogram("block_txcount_histogram_short").Update(float64(len(cbl.Bl.Tx_hashes)))
metrics.Set.GetOrCreateHistogram("block_processing_duration_histogram_seconds").UpdateDuration(processing_start)
// tracks counters for tx internals, do we need to serialize everytime, just for stats
{
complete_block_size := 0
for i := 0; i < len(cbl.Txs); i++ {
tx_size := len(cbl.Txs[i].Serialize())
complete_block_size += tx_size
metrics.Set.GetOrCreateHistogram("transaction_size_histogram_bytes").Update(float64(tx_size))
metrics.Set.GetOrCreateCounter(fmt.Sprintf(`transaction_total{type="%s"}`, cbl.Txs[i].TransactionType.String())).Inc()
if len(cbl.Txs[i].Payloads) >= 1 {
metrics.Set.GetOrCreateHistogram("transaction_ring_histogram_short").Update(float64(cbl.Txs[i].Payloads[0].Statement.RingSize))
metrics.Set.GetOrCreateHistogram("transaction_payloads_histogram_short").Update(float64(len(cbl.Txs[i].Payloads)))
}
}
metrics.Set.GetOrCreateHistogram("block_size_histogram_bytes").Update(float64(complete_block_size))
}
}()
// notify everyone who needs to know that a new block is in the chain
chain.RPC_NotifyNewBlock.L.Lock()
chain.RPC_NotifyNewBlock.Broadcast()
chain.RPC_NotifyNewBlock.L.Unlock()
if height_changed {
chain.RPC_NotifyHeightChanged.L.Lock()
chain.RPC_NotifyHeightChanged.Broadcast()
chain.RPC_NotifyHeightChanged.L.Unlock()
}
} else {
logger.V(1).Error(err, "Block rejected by chain", "BLID", block_hash)
}
}()
// first of all lets do some quick checks
// before doing extensive checks
result = false
// check if block already exist skip it
if chain.Is_Block_Topological_order(block_hash) {
return errormsg.ErrAlreadyExists, false // block already in chain skipping it
}
for k := range chain.Tips {
if block_hash == k {
return errormsg.ErrAlreadyExists, false // block already in chain skipping it
}
}
// only 2 tips allowed in block
if len(bl.Tips) >= 3 {
block_logger.V(1).Error(fmt.Errorf("More than 2 tips present in block rejecting"), "")
return errormsg.ErrPastMissing, false
}
// check whether the tips exist in our chain, if not reject
for i := range bl.Tips {
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
block_logger.V(1).Error(fmt.Errorf("Tip is NOT present in chain, skipping it till we get a parent"), "", "missing_tip", bl.Tips[i].String())
return errormsg.ErrPastMissing, false
}
}
block_height := chain.Calculate_Height_At_Tips(bl.Tips)
for i := range bl.Tips { // previous block can be refer to only recent blocks, making some attacks almost impossible
if block_height != chain.Load_Block_Height(bl.Tips[i])+1 {
block_logger.V(1).Error(fmt.Errorf("Block rejected since it is in too past"), "", "block_height", block_height, "tip_height", chain.Load_Block_Height(bl.Tips[i]))
return errormsg.ErrInvalidBlock, false
}
}
if block_height == 0 && int64(bl.Height) == block_height && len(bl.Tips) != 0 {
block_logger.Error(fmt.Errorf("Genesis block cannot have tips."), "", "tip_count", len(bl.Tips))
return errormsg.ErrInvalidBlock, false
}
if len(bl.Tips) >= 1 && bl.Height == 0 {
block_logger.Error(fmt.Errorf("Genesis block can only be at height 0"), "", "tip_count", len(bl.Tips))
return errormsg.ErrInvalidBlock, false
}
if block_height != 0 && block_height < chain.Get_Stable_Height() {
block_logger.Error(fmt.Errorf("Block rejected since it is stale."), "", "stable height", chain.Get_Stable_Height(), "block height", block_height)
return errormsg.ErrInvalidBlock, false
}
// make sure time is NOT into future,
// if clock diff is more than 50 millisecs, reject the block
if bl.Timestamp > (uint64(globals.Time().UTC().UnixMilli() + 50)) { // give 50 millisec passing
block_logger.Error(fmt.Errorf("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 {
//fmt.Printf("timestamp prev %d cur timestamp %d\n", chain.Load_Block_Timestamp(bl.Tips[i]), bl.Timestamp)
block_logger.Error(fmt.Errorf("Block timestamp is less than its parent."), "rejecting block")
return errormsg.ErrInvalidTimestamp, false
}
}
// check whether the major version ( hard fork) is valid
if !chain.Check_Block_Version(bl) {
block_logger.Error(fmt.Errorf("Rejecting !! Block has invalid fork version"), "actual", bl.Major_Version, "expected", chain.Get_Current_Version_at_Height(chain.Calculate_Height_At_Tips(bl.Tips)))
return errormsg.ErrInvalidBlock, false
}
// verify whether the tips are reachable from one another
if bl.Height >= 2 && !chain.CheckDagStructure(bl.Tips) {
block_logger.Error(fmt.Errorf("Rejecting !! Block has invalid reachability"), "Invalid rechability", "tips", bl.Tips)
return errormsg.ErrInvalidBlock, false
}
// if the block is referencing any past tip too distant into its history
for i := range bl.Tips {
if int64(bl.Height)-1 != chain.Load_Block_Height(bl.Tips[i]) {
block_logger.Error(fmt.Errorf("Rusty TIP mined by ROGUE miner discarding block"), "", "best height", bl.Height, "deviation", int64(bl.Height)-chain.Load_Block_Height(bl.Tips[i]))
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.Error(fmt.Errorf("Block is bigger than max permitted"), "Rejecting", "Actual", block_size, "MAX", config.STARGATE_HE_MAX_BLOCK_SIZE)
return errormsg.ErrInvalidSize, false
}
}
}
// verify everything related to miniblocks in one go
2021-11-27 04:25:17 +00:00
if !chain.simulator {
if err = Verify_MiniBlocks(*cbl.Bl); err != nil { // verifies the miniblocks all refer to current block
2021-11-22 16:05:02 +00:00
return err, false
}
if bl.Height != 0 { // a genesis block doesn't have miniblock
// verify hash of miniblock for corruption
if err = chain.Verify_MiniBlocks_HashCheck(cbl); err != nil {
return err, false
}
}
for _, mbl := range bl.MiniBlocks {
var miner_hash crypto.Hash
copy(miner_hash[:], mbl.KeyHash[:])
2021-11-27 04:25:17 +00:00
if mbl.Final == false && !chain.IsAddressHashValid(true, miner_hash) {
2021-11-22 16:05:02 +00:00
err = fmt.Errorf("miner address not registered")
return err, false
}
}
// verify Pow of miniblocks
for i, mbl := range bl.MiniBlocks {
if !chain.VerifyMiniblockPoW(bl, mbl) {
block_logger.Error(fmt.Errorf("MiniBlock has invalid PoW"), "rejecting", "i", i)
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.Error(fmt.Errorf("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.Error(fmt.Errorf("Miner tx failed it is not coinbase"), "rejecting")
return errormsg.ErrInvalidBlock, false
}
// always check whether the coin base tx is okay
if bl.Height != 0 {
if err = chain.Verify_Transaction_Coinbase(cbl, &bl.Miner_TX); err != nil { // if miner address is not registered give error
//block_logger.Warnf("Error verifying coinbase tx, err :'%s'", err)
return err, false
}
}
}
// 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.Error(fmt.Errorf("Missing TX"), "Incomplete block", "expected_tx", len(bl.Tx_hashes), "actual_tx", 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.Error(fmt.Errorf("duplicate TX"), "Incomplete block", "duplicate count", len(bl.Tx_hashes)-len(tx_checklist))
return errormsg.ErrInvalidBlock, false
}
for i, tx := range cbl.Txs {
if tx.Height >= bl.Height {
block_logger.Error(fmt.Errorf("Invalid TX Height"), "TX height cannot be more than block", "txid", cbl.Txs[i].GetHash().String())
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.Error(fmt.Errorf("Missing TX"), "TX missing", "txid", tx_hash.String())
return errormsg.ErrInvalidBlock, false
}
}
}
// another check, whether the block contains any duplicate registration within the block
// block wide duplicate input detector
{
reg_map := map[string]bool{}
for i := 0; i < len(cbl.Txs); i++ {
if cbl.Txs[i].TransactionType == transaction.REGISTRATION {
if _, ok := reg_map[string(cbl.Txs[i].MinerAddress[:])]; ok {
block_logger.Error(fmt.Errorf("Double Registration TX"), "duplicate registration", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
reg_map[string(cbl.Txs[i].MinerAddress[:])] = true
}
}
}
// another check, whether the block contains any colliding txs
if len(bl.Tips) == 2 {
for i := range cbl.Txs {
if cbl.Txs[i].IsProofRequired() {
if cbl.Txs[i].BLID == bl.Tips[0] || cbl.Txs[i].BLID == bl.Tips[1] {
block_logger.Error(fmt.Errorf("Colliding TXs"), "may contain colliding transactions", "txid", 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 || cbl.Txs[i].TransactionType == transaction.BURN_TX || cbl.Txs[i].TransactionType == transaction.SC_TX {
for j := range cbl.Txs[i].Payloads {
if _, ok := nonce_map[cbl.Txs[i].Payloads[j].Proof.Nonce()]; ok {
block_logger.Error(fmt.Errorf("Double Spend TX within block"), "duplicate nonce", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
nonce_map[cbl.Txs[i].Payloads[j].Proof.Nonce()] = true
}
}
}
}
// all blocks except genesis will have history
// so make sure txs are connected
if bl.Height >= 1 && len(cbl.Txs) > 0 {
history := map[crypto.Hash]bool{}
var history_array []crypto.Hash
for i := range bl.Tips {
h := int64(bl.Height) - 25
if h < 0 {
h = 0
}
history_array = append(history_array, chain.get_ordered_past(bl.Tips[i], h)...)
}
for _, h := range history_array {
history[h] = true
}
block_height = chain.Calculate_Height_At_Tips(bl.Tips)
for i, tx := range cbl.Txs {
if cbl.Txs[i].TransactionType == transaction.NORMAL || cbl.Txs[i].TransactionType == transaction.BURN_TX || cbl.Txs[i].TransactionType == transaction.SC_TX {
if history[cbl.Txs[i].BLID] != true {
block_logger.Error(fmt.Errorf("Double Spend TX within block"), "unreferable history", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
if tx.Height != uint64(chain.Load_Height_for_BL_ID(cbl.Txs[i].BLID)) {
block_logger.Error(fmt.Errorf("Double Spend TX within block"), "blid/height mismatch", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
if block_height-int64(tx.Height) < TX_VALIDITY_HEIGHT {
} else {
block_logger.Error(fmt.Errorf("Double Spend TX within block"), "long distance tx not supported", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
if tx.TransactionType == transaction.SC_TX {
if tx.SCDATA.Has(rpc.SCACTION, rpc.DataUint64) {
if rpc.SC_INSTALL == rpc.SC_ACTION(tx.SCDATA.Value(rpc.SCACTION, rpc.DataUint64).(uint64)) {
txid := tx.GetHash()
2021-11-24 09:20:51 +00:00
if txid[0] < 0x80 || txid[31] < 0x80 { // last byte should be more than 0x80
2021-11-22 16:05:02 +00:00
block_logger.Error(fmt.Errorf("Invalid SCID"), "SCID installing tx must end with >0x80 byte", "txid", cbl.Txs[i].GetHash())
return errormsg.ErrTXDoubleSpend, false
}
}
}
}
}
}
}
// we need to verify each tx with tips
{
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))
sem := semaphore.NewWeighted(int64(runtime.NumCPU()))
for i := 0; i < len(cbl.Txs); i++ {
sem.Acquire(context.Background(), 1)
go func(j int) {
defer sem.Release(1)
defer wg.Done()
2021-11-27 04:25:17 +00:00
if atomic.LoadInt32(&fail_count) >= 1 { // fail fast
return
}
2021-11-22 16:05:02 +00:00
if err := chain.Verify_Transaction_NonCoinbase_CheckNonce_Tips(hf_version, cbl.Txs[j], bl.Tips); err != nil { // transaction verification failed
atomic.AddInt32(&fail_count, 1) // increase fail count by 1
block_logger.Error(err, "tx nonce verification failed", "txid", cbl.Txs[j].GetHash())
}
}(i)
}
wg.Wait() // wait for verifications to finish
if fail_count > 0 { // check the result
block_logger.Error(fmt.Errorf("TX nonce verification failed"), "rejecting block")
return errormsg.ErrInvalidTX, 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
sem := semaphore.NewWeighted(int64(runtime.NumCPU()))
for i := 0; i < len(cbl.Txs); i++ {
sem.Acquire(context.Background(), 1)
go func(j int) {
defer sem.Release(1)
defer wg.Done()
2021-11-27 04:25:17 +00:00
if atomic.LoadInt32(&fail_count) >= 1 { // fail fast
return
}
2021-11-22 16:05:02 +00:00
if err := chain.Verify_Transaction_NonCoinbase(cbl.Txs[j]); err != nil { // transaction verification failed
atomic.AddInt32(&fail_count, 1) // increase fail count by 1
block_logger.Error(err, "tx verification failed", "txid", cbl.Txs[j].GetHash())
}
}(i)
}
wg.Wait() // wait for verifications to finish
if fail_count > 0 { // check the result
block_logger.Error(fmt.Errorf("TX verification failed"), "rejecting block")
return errormsg.ErrInvalidTX, false
}
}
// we need to do more checks but only after tx has been expanded
{
var check_data cbl_verify // used to verify sanity of new block
for i := 0; i < len(cbl.Txs); i++ {
if !(cbl.Txs[i].IsCoinbase() || cbl.Txs[i].IsRegistration()) { // all other tx must go through this check
if err = check_data.check(cbl.Txs[i], false); err == nil {
check_data.check(cbl.Txs[i], true) // keep in record for future tx
} else {
block_logger.Error(err, "Invalid TX within block", "txid", cbl.Txs[i].GetHash())
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, 0)
// 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)
height_changed = true
block_logger.Info("Chain extended", "new height", chain.Height)
} else {
block_logger.Info("Chain extended but height is same", "new height", chain.Height)
}
if height_changed {
var full_order []crypto.Hash
var base_topo_index int64 // new topo id will start from here
if cbl.Bl.Height == 0 {
full_order = append(full_order, cbl.Bl.GetHash())
} else {
current_tip := chain.Get_Top_ID()
new_tip := cbl.Bl.GetHash()
full_order, base_topo_index = chain.Generate_Full_Order_New(current_tip, new_tip)
}
// we will directly use graviton to mov in to history
2021-11-28 11:00:42 +00:00
logger.V(1).Info("Full order data", "full_order", full_order, "base_topo_index", base_topo_index)
2021-11-22 16:05:02 +00:00
if base_topo_index < 0 {
logger.Error(nil, "negative base topo, not possible, probably disk corruption or core issue")
os.Exit(0)
}
topos_written := false
for i := int64(0); i < int64(len(full_order)); i++ {
logger.V(3).Info("will execute order ", "i", i, "blid", full_order[i].String())
current_topo_block := i + base_topo_index
//previous_topo_block := current_topo_block - 1
if !topos_written && current_topo_block == chain.Load_Block_Topological_order(full_order[i]) { // skip if same order
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 {
logger.Error(err, "Cannot load block for client protocol,probably DB corruption", "blid", bl_current_hash.String())
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 {
}
var balance_tree, sc_meta *graviton.Tree
var ss *graviton.Snapshot
if bl_current.Height == 0 { // if it's genesis block
if ss, err = chain.Store.Balance_store.LoadSnapshot(0); err != nil {
panic(err)
}
} else { // we already have a block before us, use it
record_version, err := chain.ReadBlockSnapshotVersion(full_order[i-1])
if err != nil {
panic(err)
}
2021-11-28 11:00:42 +00:00
logger.V(1).Info("reading block snapshot", "blid", full_order[i-1], "i", i, "record_version", record_version)
2021-11-22 16:05:02 +00:00
ss, err = chain.Store.Balance_store.LoadSnapshot(record_version)
if err != nil {
panic(err)
}
2021-11-29 11:04:03 +00:00
}
if balance_tree, err = ss.GetTree(config.BALANCE_TREE); err != nil {
panic(err)
}
if sc_meta, err = ss.GetTree(config.SC_META); err != nil {
panic(err)
2021-11-22 16:05:02 +00:00
}
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
var data_trees []*graviton.Tree
if chain.isblock_SideBlock_internal(full_order[i], current_topo_block, int64(bl_current.Height)) {
logger.V(3).Info("this block is a side block", "height", chain.Load_Block_Height(full_order[i]), "blid", full_order[i])
} else {
logger.V(3).Info("this block is a full block", "height", chain.Load_Block_Height(full_order[i]), "blid", full_order[i])
sc_change_cache := map[crypto.Hash]*graviton.Tree{} // cache entire changes for entire block
// install hardcoded contracts
if err = chain.install_hardcoded_contracts(sc_change_cache, ss, balance_tree, sc_meta, bl_current.Height); err != nil {
panic(err)
}
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.Deserialize(tx_bytes); err != nil {
panic(err)
}
for t := range tx.Payloads {
if !tx.Payloads[t].SCID.IsZero() {
tree, _ := ss.GetTree(string(tx.Payloads[t].SCID[:]))
sc_change_cache[tx.Payloads[t].SCID] = tree
}
}
// we have loaded a tx successfully, now lets execute it
tx_fees := chain.process_transaction(sc_change_cache, tx, balance_tree, bl_current.Height)
//fmt.Printf("transaction %s type %s data %+v\n", txhash, tx.TransactionType, tx.SCDATA)
if tx.TransactionType == transaction.SC_TX {
tx_fees, err = chain.process_transaction_sc(sc_change_cache, ss, bl_current.Height, uint64(current_topo_block), bl_current.Timestamp/1000, bl_current_hash, tx, balance_tree, sc_meta)
//fmt.Printf("Processsing sc err %s\n", err)
if err == nil { // TODO process gasg here
}
}
fees_collected += tx_fees
}
}
// at this point, we must commit all the SCs, so entire tree hash is interlinked
for scid, v := range sc_change_cache {
meta_bytes, err := sc_meta.Get(SC_Meta_Key(scid))
if err != nil {
panic(err)
}
var meta SC_META_DATA // the meta contains metadata about SC
if err := meta.UnmarshalBinary(meta_bytes); err != nil {
panic(err)
}
if meta.DataHash, err = v.Hash(); err != nil { // encode data tree hash
panic(err)
}
sc_meta.Put(SC_Meta_Key(scid), meta.MarshalBinary())
data_trees = append(data_trees, v)
/*fmt.Printf("will commit tree name %x \n", v.GetName())
c := v.Cursor()
for k, v, err := c.First(); err == nil; k, v, err = c.Next() {
fmt.Printf("key=%x, value=%x\n", k, v)
}*/
}
chain.process_miner_transaction(bl_current, bl_current.Height == 0, balance_tree, fees_collected, bl_current.Height)
}
// we are here, means everything is okay, lets commit the update balance tree
data_trees = append(data_trees, balance_tree, sc_meta)
//fmt.Printf("committing data trees %+v\n", data_trees)
commit_version, err := graviton.Commit(data_trees...)
if err != nil {
panic(err)
}
chain.StoreBlock(bl_current, commit_version)
topos_written = true
chain.Store.Topo_store.Write(current_topo_block, full_order[i], commit_version, chain.Load_Block_Height(full_order[i]))
2021-11-28 11:00:42 +00:00
if logger.V(1).Enabled() {
2021-11-22 16:05:02 +00:00
merkle_root, err := chain.Load_Merkle_Hash(commit_version)
if err != nil {
panic(err)
}
2021-11-28 11:00:42 +00:00
logger.V(1).Info("storing topo", "i", i, "blid", full_order[i].String(), "topoheight", current_topo_block, "commit_version", commit_version, "committed_merkle", merkle_root)
2021-11-22 16:05:02 +00:00
}
}
}
{
// calculate new set of tips
// this is done by removing all known tips which are in the past
// and add this block as tip
old_tips := chain.Get_TIPS()
var tips []crypto.Hash
new_tips := map[crypto.Hash]crypto.Hash{}
for i := range old_tips {
for j := range bl.Tips {
if bl.Tips[j] == old_tips[i] {
goto skip_tip
}
}
tips = append(tips, old_tips[i])
skip_tip:
}
tips = append(tips, bl.GetHash()) // add current block as new tip
chain_height := chain.Get_Height()
for i := range tips {
tip_height := int64(chain.Load_Height_for_BL_ID(tips[i]))
2021-11-28 11:00:42 +00:00
if (chain_height - tip_height) == 0 {
2021-11-22 16:05:02 +00:00
new_tips[tips[i]] = tips[i]
} else { // this should be a rare event, unless network has very high latency
logger.V(2).Info("Rusty TIP declared stale", "tip", tips[i], "best height", chain_height, "tip_height", tip_height)
//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
}
}
//block_logger.Info("New tips(after adding block) ", "tips", new_tips)
chain.Tips = new_tips
}
// every 2000 block print a line
if chain.Get_Height()%2000 == 0 {
block_logger.Info(fmt.Sprintf("Chain Height %d", chain.Height))
}
purge_count := chain.MiniBlocks.PurgeHeight(chain.Get_Stable_Height()) // purge all miniblocks upto this height
logger.V(2).Info("Purged miniblock", "count", purge_count)
result = true
// TODO fix hard fork
// maintain hard fork votes to keep them SANE
//chain.Recount_Votes() // does not return anything
return // run any handlers necesary to atomically
}
// 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 {
return chain.Get_Height() - config.STABLE_LIMIT
}
// 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
}
// check whether the block is a tip
func (chain *Blockchain) Is_Block_Tip(blid crypto.Hash) (result bool) {
for k := range chain.Tips {
if blid == k {
return true
}
}
return false
}
func (chain *Blockchain) Get_Difficulty() uint64 {
return chain.Get_Difficulty_At_Tips(chain.Get_TIPS()).Uint64()
}
func (chain *Blockchain) Get_Network_HashRate() uint64 {
return chain.Get_Difficulty()
}
// 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)
}
// 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) error {
var err error
if tx.IsPremine() {
return fmt.Errorf("premine tx not mineable")
}
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(config.BALANCE_TREE); err == nil {
if _, err := balance_tree.Get(tx.MinerAddress[:]); err == nil { // address already registered
return fmt.Errorf("address already registered")
} else { // add to regpool
if chain.Regpool.Regpool_Add_TX(tx, 0) {
return nil
} else {
return fmt.Errorf("registration for address is already pending")
}
}
} else {
return err
}
} else {
return err
}
}
switch tx.TransactionType {
case transaction.BURN_TX, transaction.NORMAL, transaction.SC_TX:
default:
return fmt.Errorf("such transaction type cannot appear in mempool")
}
txhash := tx.GetHash()
// Coin base TX can not come through this path
if tx.IsCoinbase() {
logger.Error(fmt.Errorf("coinbase tx cannot appear in mempool"), "tx_rejected", "txid", txhash)
return fmt.Errorf("TX rejected coinbase tx cannot appear in mempool")
}
chain_height := uint64(chain.Get_Height())
/*if chain_height > tx.Height {
rlog.Tracef(2, "TX %s rejected since chain has already progressed", txhash)
return fmt.Errorf("TX %s rejected since chain has already progressed", txhash)
}*/
// 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 fmt.Errorf("TX %s rejected Already in MEMPOOL", txhash)
}
// check whether tx is already mined
if _, err = chain.Store.Block_tx_store.ReadTX(txhash); err == nil {
//rlog.Tracef(2, "TX %s rejected Already mined in some block", txhash)
return fmt.Errorf("TX %s rejected Already mined in some block", txhash)
}
hf_version := chain.Get_Current_Version_at_Height(int64(chain_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.Error(fmt.Errorf("Huge TX"), "TX rejected", "Actual Size", len(tx.Serialize()), "max possible ", config.STARGATE_HE_MAX_TX_SIZE)
return fmt.Errorf("TX rejected Size %d byte Max possible %d", len(tx.Serialize()), config.STARGATE_HE_MAX_TX_SIZE)
}
// check whether enough fees is provided in the transaction
calculated_fee := chain.Calculate_TX_fee(hf_version, uint64(len(tx.Serialize())))
provided_fee := tx.Fees() // get fee from tx
//logger.WithFields(log.Fields{"txid": txhash}).Warnf("TX fees check disabled provided fee %d calculated fee %d", provided_fee, calculated_fee)
if !chain.simulator && calculated_fee > provided_fee {
err = fmt.Errorf("TX %s rejected due to low fees provided fee %d calculated fee %d", txhash, provided_fee, calculated_fee)
return err
}
2021-11-24 09:20:51 +00:00
if tx.TransactionType == transaction.SC_TX {
if tx.SCDATA.Has(rpc.SCACTION, rpc.DataUint64) {
if rpc.SC_INSTALL == rpc.SC_ACTION(tx.SCDATA.Value(rpc.SCACTION, rpc.DataUint64).(uint64)) {
txid := tx.GetHash()
if txid[0] < 0x80 || txid[31] < 0x80 { // last byte should be more than 0x80
return fmt.Errorf("Invalid SCID ID,it must not start with 0x80")
}
}
}
}
2021-11-22 16:05:02 +00:00
if err := chain.Verify_Transaction_NonCoinbase_CheckNonce_Tips(hf_version, tx, chain.Get_TIPS()); err != nil { // transaction verification failed
2021-11-27 04:25:17 +00:00
logger.V(2).Error(err, "Incoming TX nonce verification failed", "txid", txhash, "stacktrace", globals.StackTrace(false))
2021-11-22 16:05:02 +00:00
return fmt.Errorf("Incoming TX %s nonce verification failed, err %s", txhash, err)
}
if err := chain.Verify_Transaction_NonCoinbase(tx); err != nil {
2021-11-27 04:25:17 +00:00
logger.V(2).Error(err, "Incoming TX could not be verified", "txid", txhash)
2021-11-22 16:05:02 +00:00
return fmt.Errorf("Incoming TX %s could not be verified, err %s", txhash, err)
}
if chain.Mempool.Mempool_Add_TX(tx, 0) { // new tx come with 0 marker
//rlog.Tracef(2, "Successfully added tx %s to pool", txhash)
return nil
} else {
//rlog.Tracef(2, "TX %s rejected by pool by mempool", txhash)
return fmt.Errorf("TX %s rejected by pool by mempool", txhash)
}
}
// side blocks are blocks which lost the race the to become part
// of main chain,
// a block is a side block if it satisfies the following condition
// if no other block exists on this height before this
// 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)
}
// todo optimize/ run more checks
func (chain *Blockchain) isblock_SideBlock_internal(blid crypto.Hash, block_topoheight int64, block_height int64) (result bool) {
if block_topoheight == 0 { // genesis cannot be side block
return false
}
toporecord, err := chain.Store.Topo_store.Read(block_topoheight - 1)
if err != nil {
panic("Could not load block from previous order")
}
if block_height == toporecord.Height { // lost race (or byzantine behaviour)
return true
}
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.Deserialize(tx_bytes); err != nil {
return
}
blids_list := chain.Find_Blocks_Height_Range(int64(tx.Height+1), int64(tx.Height+1)+2*TX_VALIDITY_HEIGHT)
var exist_list []crypto.Hash
for _, blid := range blids_list {
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 , this is removed so as this case can be tested well
}
}
}
for _, blid := range exist_list {
if chain.Isblock_SideBlock(blid) {
invalid_blid = append(invalid_blid, blid)
} else {
valid_blid = blid
valid = true
}
}
return
}
// 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
}
// 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()
if rewind_count == 0 {
return
}
top_block_topo_index := chain.Load_TOPO_HEIGHT()
rewinded := int64(0)
2021-11-27 04:25:17 +00:00
for {
2021-11-22 16:05:02 +00:00
r, err := chain.Store.Topo_store.Read(top_block_topo_index - rewinded)
if err != nil {
panic(err)
}
2021-11-27 04:25:17 +00:00
if top_block_topo_index-rewinded < 1 || rewinded >= int64(rewind_count) {
2021-11-22 16:05:02 +00:00
break
}
2021-11-27 04:25:17 +00:00
if r.Height == 1 {
break
}
2021-11-22 16:05:02 +00:00
rewinded++
}
2021-11-27 04:25:17 +00:00
for i := int64(0); i < rewinded; i++ {
2021-11-22 16:05:02 +00:00
chain.Store.Topo_store.Clean(top_block_topo_index - i)
}
return true
}
// this is part of consensus rule, 2 tips cannot refer to different parents
func (chain *Blockchain) CheckDagStructure(tips []crypto.Hash) bool {
if chain.Load_Height_for_BL_ID(tips[0]) <= 2 { // before this we cannot complete checks
return true
}
for i := range tips { // first make sure all the tips are at same height
if chain.Load_Height_for_BL_ID(tips[0]) != chain.Load_Height_for_BL_ID(tips[i]) {
return false
}
}
2021-11-27 04:25:17 +00:00
if len(tips) == 2 && tips[0] == tips[1] {
return false
}
2021-11-22 16:05:02 +00:00
switch len(tips) {
case 1:
past := chain.Get_Block_Past(tips[0])
switch len(past) {
case 1: // nothing to do here
case 2:
if chain.Load_Height_for_BL_ID(past[0]) != chain.Load_Height_for_BL_ID(past[1]) {
return false
}
past0 := chain.Get_Block_Past(past[0])
if len(past0) != 1 { //only 1 tip in past
return false
}
past1 := chain.Get_Block_Past(past[1])
if len(past1) != 1 { //only 1 tip in past
fmt.Printf("checking tips %+v past1 failed %d for %s\n", tips, len(past0), tips[0])
return false
}
if past0[0] != past1[0] { // avoid any tips which fail reachability test
return false
}
}
case 2: // lets make sure both tips originate from same parent
pasttip0 := chain.Get_Block_Past(tips[0])
if len(pasttip0) != 1 { //only 1 tip in past
return false
}
pasttip1 := chain.Get_Block_Past(tips[1])
if len(pasttip0) != len(pasttip1) {
return false
}
if pasttip0[0] != pasttip1[0] { // avoid any tips which fail reachability test
return false
}
default:
return false
}
return true
}
// 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 false
}
return true
}
// converts a DAG's partial order into a full order, this function is recursive
// dag can be processed only one height at a time
// blocks are ordered recursively, till we find a find a block which is already in the chain
// this could be done via binary search also, but this is also easy
func (chain *Blockchain) Generate_Full_Order_New(current_tip crypto.Hash, new_tip crypto.Hash) (order []crypto.Hash, topo int64) {
start := time.Now()
defer logger.V(2).Info("generating full order", "took", time.Now().Sub(start))
matchtill := chain.Load_Height_for_BL_ID(new_tip)
step_size := int64(10)
for {
matchtill -= step_size
if matchtill < 0 {
matchtill = 0
}
current_history := chain.get_ordered_past(current_tip, matchtill)
new_history := chain.get_ordered_past(new_tip, matchtill)
if matchtill == 0 {
if current_history[0] != new_history[0] {
panic("genesis not matching")
}
topo = 0
order = append(order, new_history...)
return
}
if current_history[0] != new_history[0] { // base are not matching, step back further
continue
}
if current_history[0] != new_history[0] ||
current_history[1] != new_history[1] ||
current_history[2] != new_history[2] ||
current_history[3] != new_history[3] {
continue // base are not matching, step back further
}
order = append(order, new_history[:]...)
topo = chain.Load_Block_Topological_order(order[0])
return
}
return
}
// we will collect atleast 50 blocks or till genesis
func (chain *Blockchain) get_ordered_past(tip crypto.Hash, tillheight int64) (order []crypto.Hash) {
order = append(order, tip)
current := tip
for chain.Load_Height_for_BL_ID(current) > tillheight {
past := chain.Get_Block_Past(current)
switch len(past) {
case 0: // we reached genesis return
return
case 1:
order = append(order, past[0])
current = past[0]
case 2:
if bytes.Compare(past[0][:], past[1][:]) < 0 {
order = append(order, past[0], past[1])
} else {
order = append(order, past[1], past[0])
}
current = past[0]
default:
panic("data corruption")
}
}
for i, j := 0, len(order)-1; i < j; i, j = i+1, j-1 {
order[i], order[j] = order[j], order[i]
}
return
}