DERO-HE STARGATE Testnet Release36

This commit is contained in:
Captain 2021-12-09 15:59:01 +00:00
parent f0d3e7a6e8
commit d2e9638394
No known key found for this signature in database
GPG Key ID: 18CDB3ED5E85D2D4
13 changed files with 366 additions and 281 deletions

View File

@ -191,7 +191,7 @@ func (bl *Block) Deserialize(buf []byte) (err error) {
defer func() { defer func() {
if r := recover(); r != nil { if r := recover(); r != nil {
err = fmt.Errorf("Invalid Block cannot deserialize '%x' stack %s", hex.EncodeToString(buf), string(debug.Stack())) err = fmt.Errorf("Invalid Block cannot deserialize '%s' stack %s", hex.EncodeToString(buf), string(debug.Stack()))
return return
} }
}() }()

View File

@ -236,11 +236,11 @@ func Blockchain_Start(params map[string]interface{}) (*Blockchain, error) {
top_block_topo_index := chain.Load_TOPO_HEIGHT() top_block_topo_index := chain.Load_TOPO_HEIGHT()
if top_block_topo_index < 4 { if top_block_topo_index < 2 {
return return
} }
top_block_topo_index -= 4 top_block_topo_index -= 2
blid, err := chain.Load_Block_Topological_order_at_index(top_block_topo_index) blid, err := chain.Load_Block_Topological_order_at_index(top_block_topo_index)
if err != nil { if err != nil {
@ -423,9 +423,9 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
} }
} }
// only 2 tips allowed in block // only 1 tips allowed in block
if len(bl.Tips) >= 3 { if len(bl.Tips) > 1 {
block_logger.V(1).Error(fmt.Errorf("More than 2 tips present in block rejecting"), "") block_logger.V(1).Error(fmt.Errorf("More than 1 tips present in block rejecting"), "")
return errormsg.ErrPastMissing, false return errormsg.ErrPastMissing, false
} }
@ -455,10 +455,10 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
return errormsg.ErrInvalidBlock, false return errormsg.ErrInvalidBlock, false
} }
if block_height != 0 && block_height < chain.Get_Stable_Height() { //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) // 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 // return errormsg.ErrInvalidBlock, false
} //}
// make sure time is NOT into future, // make sure time is NOT into future,
// if clock diff is more than 50 millisecs, reject the block // if clock diff is more than 50 millisecs, reject the block
@ -796,201 +796,195 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
// if the block is on a lower height tip, the block will not increase chain height // 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) height := chain.Load_Height_for_BL_ID(block_hash)
if height > chain.Get_Height() || height == 0 { // exception for genesis block if height > chain.Get_Height() || height == 0 { // exception for genesis block
atomic.StoreInt64(&chain.Height, height) //atomic.StoreInt64(&chain.Height, height)
//chain.Store_TOP_HEIGHT(dbtx, height)
height_changed = true height_changed = true
block_logger.Info("Chain extended", "new height", chain.Height) block_logger.Info("Chain extended", "new height", chain.Height)
} else { } else {
block_logger.Info("Chain extended but height is same", "new height", chain.Height) block_logger.Info("Chain extended but height is same", "new height", chain.Height)
} }
if height_changed { {
// 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
var full_order []crypto.Hash bl_current_hash := cbl.Bl.GetHash()
var base_topo_index int64 // new topo id will start from here bl_current := cbl.Bl
current_topo_block := bl_current.Height
logger.V(3).Info("will insert block ", "blid", bl_current_hash.String())
//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)
// generate miner TX rewards as per client protocol
if hard_fork_version_current == 1 {
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 var balance_tree, sc_meta *graviton.Tree
logger.V(1).Info("Full order data", "full_order", full_order, "base_topo_index", base_topo_index) var ss *graviton.Snapshot
if bl_current.Height == 0 { // if it's genesis block
if base_topo_index < 0 { if ss, err = chain.Store.Balance_store.LoadSnapshot(0); err != nil {
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)
}
logger.V(1).Info("reading block snapshot", "blid", full_order[i-1], "i", i, "record_version", record_version)
ss, err = chain.Store.Balance_store.LoadSnapshot(record_version)
if err != nil {
panic(err)
}
}
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) panic(err)
} }
} else { // we already have a block before us, use it
fees_collected := uint64(0) record_version, err := chain.ReadBlockSnapshotVersion(bl.Tips[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 { if err != nil {
panic(err) panic(err)
} }
chain.StoreBlock(bl_current, commit_version) logger.V(1).Info("reading block snapshot", "blid", bl.Tips[0], "record_version", record_version)
topos_written = true
chain.Store.Topo_store.Write(current_topo_block, full_order[i], commit_version, chain.Load_Block_Height(full_order[i])) ss, err = chain.Store.Balance_store.LoadSnapshot(record_version)
if logger.V(1).Enabled() { if err != nil {
merkle_root, err := chain.Load_Merkle_Hash(commit_version) panic(err)
}
}
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)
}
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
{
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 { if err != nil {
panic(err) panic(err)
} }
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)
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)
if height_changed {
// we need to write history until the entire chain is fixed
fix_bl := bl_current
fix_pos := bl_current.Height
fix_commit_version := commit_version
for ; ; fix_pos-- {
chain.Store.Topo_store.Write(int64(fix_bl.Height), fix_bl.GetHash(), fix_commit_version, int64(fix_bl.Height))
logger.V(1).Info("fixed loop", "topo", fix_pos)
if fix_pos == 0 { // break if we reached genesis
break
}
r, err := chain.Store.Topo_store.Read(int64(fix_pos - 1))
if err != nil {
panic(err)
}
if fix_bl.Tips[0] == r.BLOCK_ID { // break if we reached a common point
break
}
// prepare for another round
fix_commit_version, err = chain.ReadBlockSnapshotVersion(fix_bl.Tips[0])
if err != nil {
panic(err)
}
fix_bl, err = chain.Load_BL_FROM_ID(fix_bl.Tips[0])
if err != nil {
panic(err)
}
} }
} }
if logger.V(1).Enabled() {
merkle_root, err := chain.Load_Merkle_Hash(commit_version)
if err != nil {
panic(err)
}
logger.V(1).Info("storing topo", "topo", int64(bl_current.Height), "blid", bl_current_hash, "topoheight", current_topo_block, "commit_version", commit_version, "committed_merkle", merkle_root)
}
} }
{ {
@ -1227,14 +1221,11 @@ func (chain *Blockchain) Add_TX_To_Pool(tx *transaction.Transaction) error {
// this is the topoheight of this block itself // this is the topoheight of this block itself
func (chain *Blockchain) Isblock_SideBlock(blid crypto.Hash) bool { func (chain *Blockchain) Isblock_SideBlock(blid crypto.Hash) bool {
block_topoheight := chain.Load_Block_Topological_order(blid) block_topoheight := chain.Load_Block_Topological_order(blid)
if block_topoheight == 0 { if block_topoheight >= 0 {
return false 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) return true
} }
// todo optimize/ run more checks // todo optimize/ run more checks
@ -1288,12 +1279,8 @@ func (chain *Blockchain) IS_TX_Valid(txhash crypto.Hash) (valid_blid crypto.Hash
} }
for _, blid := range exist_list { for _, blid := range exist_list {
if chain.Isblock_SideBlock(blid) { valid_blid = blid
invalid_blid = append(invalid_blid, blid) valid = true
} else {
valid_blid = blid
valid = true
}
} }
@ -1352,6 +1339,8 @@ func (chain *Blockchain) Rewind_Chain(rewind_count int) (result bool) {
chain.Store.Topo_store.Clean(top_block_topo_index - i) chain.Store.Topo_store.Clean(top_block_topo_index - i)
} }
chain.MiniBlocks.PurgeHeight(0xffffffffffffff) // purge all miniblocks upto this height
return true return true
} }
@ -1456,55 +1445,6 @@ func (chain *Blockchain) IsBlockSyncBlockHeightSpecific(blid crypto.Hash, chain_
return true 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 // we will collect atleast 50 blocks or till genesis
func (chain *Blockchain) get_ordered_past(tip crypto.Hash, tillheight int64) (order []crypto.Hash) { func (chain *Blockchain) get_ordered_past(tip crypto.Hash, tillheight int64) (order []crypto.Hash) {
order = append(order, tip) order = append(order, tip)
@ -1536,3 +1476,50 @@ func (chain *Blockchain) get_ordered_past(tip crypto.Hash, tillheight int64) (or
} }
return return
} }
// this will flip chain top, depending on which block has more work
// more worked block is normally identified in < 2 secs
func (chain *Blockchain) flip_top() {
chain.Lock()
defer chain.Unlock()
height := chain.Get_Height()
var tips []crypto.Hash
if len(chain.Get_TIPS()) <= 1 {
return
}
// lets fill in the tips from miniblocks, list is already sorted
if keys := chain.MiniBlocks.GetAllKeys(height + 1); len(keys) >= 1 {
top_id := chain.Get_Top_ID()
for _, key := range keys {
mbls := chain.MiniBlocks.GetAllMiniBlocks(key)
if len(mbls) < 1 {
continue
}
mbl := mbls[0]
tips = tips[:0]
tip := convert_uint32_to_crypto_hash(mbl.Past[0])
if ehash, ok := chain.ExpandMiniBlockTip(tip); ok {
if ehash != top_id { // we need to flip top
fix_commit_version, err := chain.ReadBlockSnapshotVersion(ehash)
if err != nil {
panic(err)
}
fix_bl, err := chain.Load_BL_FROM_ID(ehash)
if err != nil {
panic(err)
}
chain.Store.Topo_store.Write(int64(fix_bl.Height), ehash, fix_commit_version, int64(fix_bl.Height))
// fmt.Printf("flipped top from %s to %s\n", top_id, ehash)
return
}
}
}
}
}

View File

@ -163,7 +163,7 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
} }
for i := range tips { for i := range tips {
if len(bl.Tips) < 2 { //only 2 tips max if len(bl.Tips) < 1 { //only 1 tip max
var check_tips []crypto.Hash var check_tips []crypto.Hash
check_tips = append(check_tips, bl.Tips...) check_tips = append(check_tips, bl.Tips...)
check_tips = append(check_tips, tips[i]) check_tips = append(check_tips, tips[i])
@ -176,7 +176,7 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
} }
height := chain.Calculate_Height_At_Tips(bl.Tips) // we are 1 higher than previous highest tip height := chain.Calculate_Height_At_Tips(bl.Tips) // we are 1 higher than previous highest tip
history := map[crypto.Hash]bool{} history := map[crypto.Hash]crypto.Hash{}
var history_array []crypto.Hash var history_array []crypto.Hash
for i := range bl.Tips { for i := range bl.Tips {
@ -187,7 +187,17 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
history_array = append(history_array, chain.get_ordered_past(bl.Tips[i], h)...) history_array = append(history_array, chain.get_ordered_past(bl.Tips[i], h)...)
} }
for _, h := range history_array { for _, h := range history_array {
history[h] = true
version, err := chain.ReadBlockSnapshotVersion(h)
if err != nil {
panic(err)
}
hash, err := chain.Load_Merkle_Hash(version)
if err != nil {
panic(err)
}
history[h] = hash
} }
var tx_hash_list_included []crypto.Hash // these tx will be included ( due to block size limit ) var tx_hash_list_included []crypto.Hash // these tx will be included ( due to block size limit )
@ -243,7 +253,7 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
if tx := chain.Mempool.Mempool_Get_TX(tx_hash_list_sorted[i].Hash); tx != nil { if tx := chain.Mempool.Mempool_Get_TX(tx_hash_list_sorted[i].Hash); tx != nil {
if int64(tx.Height) < height { if int64(tx.Height) < height {
if history[tx.BLID] != true { if _, ok := history[tx.BLID]; !ok {
logger.V(8).Info("not selecting tx since the reference with which it was made is not in history", "txid", tx_hash_list_sorted[i].Hash) logger.V(8).Info("not selecting tx since the reference with which it was made is not in history", "txid", tx_hash_list_sorted[i].Hash)
continue continue
} }
@ -255,16 +265,7 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
} }
} }
version, err := chain.ReadBlockSnapshotVersion(tx.BLID) if history[tx.BLID] != tx.Payloads[0].Statement.Roothash {
if err != nil {
continue
}
hash, err := chain.Load_Merkle_Hash(version)
if err != nil {
continue
}
if hash != tx.Payloads[0].Statement.Roothash {
//return fmt.Errorf("Tx statement roothash mismatch expected %x actual %x", tx.Payloads[0].Statement.Roothash, hash[:]) //return fmt.Errorf("Tx statement roothash mismatch expected %x actual %x", tx.Payloads[0].Statement.Roothash, hash[:])
continue continue
} }
@ -276,12 +277,12 @@ func (chain *Blockchain) Create_new_miner_block(miner_address rpc.Address) (cbl
sizeoftxs += tx_hash_list_sorted[i].Size sizeoftxs += tx_hash_list_sorted[i].Size
cbl.Txs = append(cbl.Txs, tx) cbl.Txs = append(cbl.Txs, tx)
tx_hash_list_included = append(tx_hash_list_included, tx_hash_list_sorted[i].Hash) tx_hash_list_included = append(tx_hash_list_included, tx_hash_list_sorted[i].Hash)
logger.V(8).Info("tx selected for mining ", "txlist", tx_hash_list_sorted[i].Hash) logger.V(1).Info("tx selected for mining ", "txlist", tx_hash_list_sorted[i].Hash)
} else { } else {
logger.V(8).Info("not selecting tx due to pre_check failure", "txid", tx_hash_list_sorted[i].Hash) logger.V(8).Info("not selecting tx due to pre_check failure", "txid", tx_hash_list_sorted[i].Hash)
} }
} else { } else {
logger.V(8).Info("not selecting tx due to nonce failure", "txid", tx_hash_list_sorted[i].Hash) logger.V(1).Info("not selecting tx due to nonce failure", "txid", tx_hash_list_sorted[i].Hash)
} }
} else { } else {
logger.V(8).Info("not selecting tx due to height difference", "txid", tx_hash_list_sorted[i].Hash) logger.V(8).Info("not selecting tx due to height difference", "txid", tx_hash_list_sorted[i].Hash)

View File

@ -122,6 +122,8 @@ func (chain *Blockchain) InsertMiniBlock(mbl block.MiniBlock) (err error, result
chain.RPC_NotifyNewMiniBlock.L.Lock() chain.RPC_NotifyNewMiniBlock.L.Lock()
chain.RPC_NotifyNewMiniBlock.Broadcast() chain.RPC_NotifyNewMiniBlock.Broadcast()
chain.RPC_NotifyNewMiniBlock.L.Unlock() chain.RPC_NotifyNewMiniBlock.L.Unlock()
chain.flip_top()
} }
return err, result return err, result
} }

View File

@ -33,6 +33,8 @@ import "github.com/deroproject/graviton"
import "github.com/deroproject/derohe/config" import "github.com/deroproject/derohe/config"
import "github.com/deroproject/derohe/block" import "github.com/deroproject/derohe/block"
import "github.com/deroproject/derohe/rpc"
import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/cryptography/crypto" import "github.com/deroproject/derohe/cryptography/crypto"
import "github.com/deroproject/derohe/transaction" import "github.com/deroproject/derohe/transaction"
import "github.com/deroproject/derohe/cryptography/bn256" import "github.com/deroproject/derohe/cryptography/bn256"
@ -178,7 +180,12 @@ func (chain *Blockchain) Verify_Transaction_NonCoinbase_CheckNonce_Tips(hf_versi
//fmt.Printf("tx nonce %d tip nonce %d\n", tx_nb.NonceHeight, tip_nb.NonceHeight) //fmt.Printf("tx nonce %d tip nonce %d\n", tx_nb.NonceHeight, tip_nb.NonceHeight)
if tip_nb.NonceHeight > tx_nb.NonceHeight { if tip_nb.NonceHeight > tx_nb.NonceHeight {
return fmt.Errorf("Invalid Nonce, not usable, expected %d actual %d", tip_nb.NonceHeight, tx_nb.NonceHeight) addr, err1 := rpc.NewAddressFromCompressedKeys(key_compressed)
if err1 != nil {
panic(err1)
}
addr.Mainnet = globals.IsMainnet()
return fmt.Errorf("Invalid Nonce, not usable, expected %d actual %d address %s", tip_nb.NonceHeight, tx_nb.NonceHeight, addr.String())
} }
} }
} }

View File

@ -351,7 +351,7 @@ func readline_loop(l *readline.Instance, chain *blockchain.Blockchain, logger lo
}() }()
//restart_loop: restart_loop:
for { for {
line, err := l.Readline() line, err := l.Readline()
if err == io.EOF { if err == io.EOF {
@ -534,7 +534,54 @@ func readline_loop(l *readline.Instance, chain *blockchain.Blockchain, logger lo
case command == "print_tree": // prints entire block chain tree case command == "print_tree": // prints entire block chain tree
//WriteBlockChainTree(chain, "/tmp/graph.dot") //WriteBlockChainTree(chain, "/tmp/graph.dot")
case command == "install_block":
case command == "block_export":
var hash crypto.Hash
if len(line_parts) == 2 && len(line_parts[1]) == 64 {
bl_raw, err := hex.DecodeString(strings.ToLower(line_parts[1]))
if err != nil {
fmt.Printf("err while decoding blid err %s\n", err)
continue
}
copy(hash[:32], []byte(bl_raw))
} else {
fmt.Printf("block_export needs a single block id as argument\n")
continue
}
var cbl *block.Complete_Block
bl, err := chain.Load_BL_FROM_ID(hash)
if err != nil {
fmt.Printf("Err %s\n", err)
continue
}
cbl = &block.Complete_Block{Bl: bl}
for _, txid := range bl.Tx_hashes {
var tx transaction.Transaction
if tx_bytes, err := chain.Store.Block_tx_store.ReadTX(txid); err != nil {
fmt.Printf("err while reading txid err %s\n", err)
continue restart_loop
} else if err = tx.Deserialize(tx_bytes); err != nil {
fmt.Printf("err deserializing tx err %s\n", err)
continue restart_loop
}
cbl.Txs = append(cbl.Txs, &tx)
}
cbl_bytes := p2p.Convert_CBL_TO_P2PCBL(cbl, true)
if err := os.WriteFile(fmt.Sprintf("/tmp/%s.block", hash), cbl_bytes, 0755); err != nil {
fmt.Printf("err writing block %s\n", err)
continue
}
fmt.Printf("successfully exported block to %s\n", fmt.Sprintf("/tmp/%s.block", hash))
case command == "block_import":
var hash crypto.Hash var hash crypto.Hash
if len(line_parts) == 2 && len(line_parts[1]) == 64 { if len(line_parts) == 2 && len(line_parts[1]) == 64 {
@ -549,27 +596,16 @@ func readline_loop(l *readline.Instance, chain *blockchain.Blockchain, logger lo
continue continue
} }
var bl block.Block
var cbl *block.Complete_Block var cbl *block.Complete_Block
if block_data, err := os.ReadFile(fmt.Sprintf("/tmp/blocks/%s", hash)); err == nil { if block_data, err := os.ReadFile(fmt.Sprintf("/tmp/%s.block", hash)); err == nil {
if err = bl.Deserialize(block_data); err != nil { // we should deserialize the block here cbl = p2p.Convert_P2PCBL_TO_CBL(block_data)
logger.Error(err, "fError deserialiing block, block id %x len(data) %d data %x", hash[:], len(block_data), block_data, err)
continue
}
cbl = &block.Complete_Block{Bl: &bl}
} else { } else {
fmt.Printf("err reading block %s\n", err) fmt.Printf("err reading block %s\n", err)
continue continue
} }
// bl, err := chain.Load_BL_FROM_ID(hash)
// if err != nil {
// fmt.Printf("Err %s\n", err)
// }
err, _ = chain.Add_Complete_Block(cbl) err, _ = chain.Add_Complete_Block(cbl)
fmt.Printf("err adding block %s\n", err) fmt.Printf("err adding block %s\n", err)
@ -1007,6 +1043,8 @@ var completer = readline.NewPrefixCompleter(
readline.PcItem("peer_list"), readline.PcItem("peer_list"),
readline.PcItem("print_bc"), readline.PcItem("print_bc"),
readline.PcItem("print_block"), readline.PcItem("print_block"),
readline.PcItem("block_export"),
readline.PcItem("block_import"),
// readline.PcItem("print_tx"), // readline.PcItem("print_tx"),
readline.PcItem("status"), readline.PcItem("status"),
readline.PcItem("sync_info"), readline.PcItem("sync_info"),

View File

@ -220,6 +220,7 @@ type txinfo struct {
Hex string // raw tx Hex string // raw tx
Height string // height at which tx was mined Height string // height at which tx was mined
HeightBuilt uint64 // height at which tx was built HeightBuilt uint64 // height at which tx was built
BLID string // BLID in whose reference this tx was built
RootHash string // roothash which forms the basis for balance tree RootHash string // roothash which forms the basis for balance tree
TransactionType string // transaction type TransactionType string // transaction type
Depth int64 Depth int64
@ -424,6 +425,7 @@ func load_tx_info_from_tx(info *txinfo, tx *transaction.Transaction) (err error)
info.RootHash = fmt.Sprintf("%x", tx.Payloads[0].Statement.Roothash[:]) info.RootHash = fmt.Sprintf("%x", tx.Payloads[0].Statement.Roothash[:])
} }
info.HeightBuilt = tx.Height info.HeightBuilt = tx.Height
info.BLID = fmt.Sprintf("%x", tx.BLID)
//info.In = len(tx.Vin) //info.In = len(tx.Vin)
//info.Out = len(tx.Vout) //info.Out = len(tx.Vout)

View File

@ -76,7 +76,7 @@
{{end}} {{end}}
{{if or (eq .info.TransactionType "NORMAL") (eq .info.TransactionType "BURN") (eq .info.TransactionType "SC") }} {{if or (eq .info.TransactionType "NORMAL") (eq .info.TransactionType "BURN") (eq .info.TransactionType "SC") }}
<H5 style="margin:5px">built BLID : <a href="/block/{{.info.BLID}}">{{.info.BLID}}</a> </H5>
<H5 style="margin:5px">Tx RootHash: {{.info.RootHash}} built height : {{.info.HeightBuilt}} </H5> <H5 style="margin:5px">Tx RootHash: {{.info.RootHash}} built height : {{.info.HeightBuilt}} </H5>

View File

@ -210,6 +210,7 @@ func main() {
} }
go mine_block_auto(chain, genesis_wallet.GetAddress()) // automatically keep mining blocks go mine_block_auto(chain, genesis_wallet.GetAddress()) // automatically keep mining blocks
globals.Cron.Start() // start cron jobs
// This tiny goroutine continuously updates status as required // This tiny goroutine continuously updates status as required
go func() { go func() {

View File

@ -105,7 +105,7 @@ var Mainnet = CHAIN_CONFIG{Name: "mainnet",
} }
var Testnet = CHAIN_CONFIG{Name: "testnet", // testnet will always have last 3 bytes 0 var Testnet = CHAIN_CONFIG{Name: "testnet", // testnet will always have last 3 bytes 0
Network_ID: uuid.FromBytesOrNil([]byte{0x59, 0xd7, 0xf7, 0xe9, 0xdd, 0x48, 0xd5, 0xfd, 0x13, 0x0a, 0xf6, 0xe0, 0x75, 0x00, 0x00, 0x00}), Network_ID: uuid.FromBytesOrNil([]byte{0x59, 0xd7, 0xf7, 0xe9, 0xdd, 0x48, 0xd5, 0xfd, 0x13, 0x0a, 0xf6, 0xe0, 0x76, 0x00, 0x00, 0x00}),
GETWORK_Default_Port: 10100, GETWORK_Default_Port: 10100,
P2P_Default_Port: 40401, P2P_Default_Port: 40401,
RPC_Default_Port: 40402, RPC_Default_Port: 40402,

View File

@ -20,4 +20,4 @@ import "github.com/blang/semver/v4"
// right now it has to be manually changed // right now it has to be manually changed
// do we need to include git commitsha?? // do we need to include git commitsha??
var Version = semver.MustParse("3.4.96-1.DEROHE.STARGATE+25112021") var Version = semver.MustParse("3.4.97-1.DEROHE.STARGATE+26112021")

View File

@ -174,8 +174,9 @@ try_again:
} else if chain.Get_Height()-response.Common.Height != 0 && chain.Get_Height()-response.Start_height <= config.STABLE_LIMIT { } else if chain.Get_Height()-response.Common.Height != 0 && chain.Get_Height()-response.Start_height <= config.STABLE_LIMIT {
pop_count = chain.Load_TOPO_HEIGHT() - response.Start_topoheight pop_count = chain.Load_TOPO_HEIGHT() - response.Start_topoheight
} else if chain.Get_Height() < connection.Height && chain.Get_Height()-response.Start_height > config.STABLE_LIMIT { // we must somehow notify that deviation is way too much and manual interaction is necessary, so as any bug for chain deviationmay be detected } else if chain.Get_Height() < connection.Height && chain.Get_Height()-response.Start_height > config.STABLE_LIMIT { // we must somehow notify that deviation is way too much and manual interaction is necessary, so as any bug for chain deviationmay be detected
connection.logger.V(1).Error(nil, "we have or others have deviated too much.you may have to use --sync-node option", "our topoheight", chain.Load_TOPO_HEIGHT(), "peer topoheight start", response.Start_topoheight) //connection.logger.V(1).Error(nil, "we have or others have deviated too much.you may have to use --sync-node option", "our topoheight", chain.Load_TOPO_HEIGHT(), "peer topoheight start", response.Start_topoheight)
return //return
pop_count = chain.Load_TOPO_HEIGHT() - response.Start_topoheight
} }
if pop_count >= 1 { // peer is claiming his chain is good and we should rewind if pop_count >= 1 { // peer is claiming his chain is good and we should rewind
@ -211,7 +212,7 @@ try_again:
var orequest ObjectList var orequest ObjectList
var oresponse Objects var oresponse Objects
//fmt.Printf("inserting blocks %d\n", (int64(i) + response.Start_topoheight)) //fmt.Printf("inserting blocks %d %x\n", (int64(i) + response.Start_topoheight), response.Block_list[i][:])
orequest.Block_list = append(orequest.Block_list, response.Block_list[i]) orequest.Block_list = append(orequest.Block_list, response.Block_list[i])
fill_common(&orequest.Common) fill_common(&orequest.Common)
if err := connection.Client.Call("Peer.GetObject", orequest, &oresponse); err != nil { if err := connection.Client.Call("Peer.GetObject", orequest, &oresponse); err != nil {
@ -226,7 +227,7 @@ try_again:
// if we reached here we were able to verify basic chain structure // if we reached here we were able to verify basic chain structure
chain.Rewind_Chain(int(pop_count)) // pop as many blocks as necessary, assumming peer has given us good chain // chain.Rewind_Chain(int(pop_count)) // pop as many blocks as necessary, assumming peer has given us good chain
failcount := 0 failcount := 0
for i := range response.Block_list { for i := range response.Block_list {

View File

@ -11,6 +11,7 @@ import "github.com/fxamacker/cbor/v2"
import "github.com/klauspost/reedsolomon" import "github.com/klauspost/reedsolomon"
import "github.com/deroproject/derohe/block" import "github.com/deroproject/derohe/block"
import "github.com/deroproject/derohe/transaction"
import "github.com/deroproject/derohe/globals" import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/errormsg" import "github.com/deroproject/derohe/errormsg"
import "github.com/deroproject/derohe/config" import "github.com/deroproject/derohe/config"
@ -248,6 +249,51 @@ func is_already_chunked_by_us(blid crypto.Hash, data_shard_count, parity_shard_c
return return
} }
// convert complete block to p2p block format
func Convert_CBL_TO_P2PCBL(cbl *block.Complete_Block, processblock bool) []byte {
var cbor_cbl Complete_Block
if processblock {
cbor_cbl.Block = cbl.Bl.Serialize()
}
for _, tx := range cbl.Txs {
cbor_cbl.Txs = append(cbor_cbl.Txs, tx.Serialize())
}
if len(cbor_cbl.Txs) != len(cbl.Bl.Tx_hashes) {
panic("invalid complete block")
}
cbl_serialized, err := cbor.Marshal(cbor_cbl)
if err != nil {
panic(err)
}
return cbl_serialized
}
// convert p2p complete block to complete block format
func Convert_P2PCBL_TO_CBL(input []byte) *block.Complete_Block {
var cbor_cbl Complete_Block
cbl := &block.Complete_Block{Bl: &block.Block{}}
if err := cbor.Unmarshal(input, &cbor_cbl); err != nil {
panic(err)
}
if err := cbl.Bl.Deserialize(cbor_cbl.Block); err != nil {
panic(err)
}
for _, tx_bytes := range cbor_cbl.Txs {
var tx transaction.Transaction
if err := tx.Deserialize(tx_bytes); err != nil {
panic(err)
}
cbl.Txs = append(cbl.Txs, &tx)
}
return cbl
}
// note we do not send complete block, // note we do not send complete block,
// since other nodes already have most of the mempool, let the mempool be used as much as possible // since other nodes already have most of the mempool, let the mempool be used as much as possible
func convert_block_to_chunks(cbl *block.Complete_Block, data_shard_count, parity_shard_count int) ([32]byte, int) { func convert_block_to_chunks(cbl *block.Complete_Block, data_shard_count, parity_shard_count int) ([32]byte, int) {