DERO-HE STARGATE Testnet Release36
This commit is contained in:
parent
f0d3e7a6e8
commit
d2e9638394
@ -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
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
@ -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,68 +796,29 @@ 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 {
|
|
||||||
|
|
||||||
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
|
|
||||||
logger.V(1).Info("Full order data", "full_order", full_order, "base_topo_index", base_topo_index)
|
|
||||||
|
|
||||||
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
|
// TODO we must run smart contracts and TXs in this order
|
||||||
// basically client protocol must run here
|
// basically client protocol must run here
|
||||||
// even if the HF has triggered we may still accept, old blocks for some time
|
// 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
|
// so hf is detected block-wise and processed as such
|
||||||
|
|
||||||
bl_current_hash := full_order[i]
|
bl_current_hash := cbl.Bl.GetHash()
|
||||||
bl_current, err1 := chain.Load_BL_FROM_ID(bl_current_hash)
|
bl_current := cbl.Bl
|
||||||
if err1 != nil {
|
current_topo_block := bl_current.Height
|
||||||
logger.Error(err, "Cannot load block for client protocol,probably DB corruption", "blid", bl_current_hash.String())
|
logger.V(3).Info("will insert block ", "blid", bl_current_hash.String())
|
||||||
return errormsg.ErrInvalidBlock, false
|
|
||||||
}
|
|
||||||
|
|
||||||
//fmt.Printf("\ni %d bl %+v\n",i, bl_current)
|
//fmt.Printf("\ni %d bl %+v\n",i, bl_current)
|
||||||
|
|
||||||
height_current := chain.Calculate_Height_At_Tips(bl_current.Tips)
|
height_current := chain.Calculate_Height_At_Tips(bl_current.Tips)
|
||||||
hard_fork_version_current := chain.Get_Current_Version_at_Height(height_current)
|
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
|
// generate miner TX rewards as per client protocol
|
||||||
if hard_fork_version_current == 1 {
|
if hard_fork_version_current == 1 {
|
||||||
|
|
||||||
@ -871,12 +832,12 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
|
|||||||
}
|
}
|
||||||
} else { // we already have a block before us, use it
|
} else { // we already have a block before us, use it
|
||||||
|
|
||||||
record_version, err := chain.ReadBlockSnapshotVersion(full_order[i-1])
|
record_version, err := chain.ReadBlockSnapshotVersion(bl.Tips[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.V(1).Info("reading block snapshot", "blid", full_order[i-1], "i", i, "record_version", record_version)
|
logger.V(1).Info("reading block snapshot", "blid", bl.Tips[0], "record_version", record_version)
|
||||||
|
|
||||||
ss, err = chain.Store.Balance_store.LoadSnapshot(record_version)
|
ss, err = chain.Store.Balance_store.LoadSnapshot(record_version)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -899,10 +860,7 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
|
|||||||
//chain.Store.Topo_store.Write(i+base_topo_index, full_order[i],0, int64(bl_current.Height)) // write entry so as sideblock could work
|
//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
|
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
|
sc_change_cache := map[crypto.Hash]*graviton.Tree{} // cache entire changes for entire block
|
||||||
|
|
||||||
@ -980,18 +938,54 @@ func (chain *Blockchain) Add_Complete_Block(cbl *block.Complete_Block) (err erro
|
|||||||
}
|
}
|
||||||
|
|
||||||
chain.StoreBlock(bl_current, commit_version)
|
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]))
|
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() {
|
if logger.V(1).Enabled() {
|
||||||
merkle_root, err := chain.Load_Merkle_Hash(commit_version)
|
merkle_root, err := chain.Load_Merkle_Hash(commit_version)
|
||||||
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)
|
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) {
|
|
||||||
invalid_blid = append(invalid_blid, blid)
|
|
||||||
} else {
|
|
||||||
valid_blid = blid
|
valid_blid = blid
|
||||||
valid = true
|
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
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
@ -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)
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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"),
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
@ -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>
|
||||||
|
|
||||||
|
|
||||||
|
@ -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() {
|
||||||
|
@ -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,
|
||||||
|
@ -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")
|
||||||
|
@ -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 {
|
||||||
|
@ -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) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user