2020-12-19 10:01:29 +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
import "fmt"
import "time"
import "sync"
import "strings"
import "math/rand"
import "runtime/debug"
import "golang.org/x/xerrors"
import "golang.org/x/time/rate"
import "github.com/romana/rlog"
// this file creates the blobs which can be used to mine new blocks
import "github.com/deroproject/derohe/block"
import "github.com/deroproject/derohe/config"
import "github.com/deroproject/derohe/crypto"
import "github.com/deroproject/derohe/globals"
import "github.com/deroproject/derohe/address"
//import "github.com/deroproject/derohe/emission"
import "github.com/deroproject/derohe/transaction"
import "github.com/deroproject/graviton"
//NOTE: this function is quite naughty due to chicken and egg problem
// we cannot calculate reward until we known blocksize ( exactly upto byte size )
// also note that we cannot calculate blocksize until we know reward
// how we do it
// reward field is a varint, ( can be 8 bytes )
// so max deviation can be 8 bytes, due to reward
// so we do a bruterforce till the reward is obtained but lets try to KISS
// the top hash over which to do mining now ( it should already be in the chain)
// this is work in progress
// TODO we need to rework fees algorithm, to improve its quality and lower fees
func ( chain * Blockchain ) Create_new_miner_block ( miner_address address . Address , tx * transaction . Transaction ) ( cbl * block . Complete_Block , bl block . Block ) {
//chain.Lock()
//defer chain.Unlock()
var err error
cbl = & block . Complete_Block { }
if tx != nil { // make sure tx is registration and it valid
if tx . IsRegistration ( ) {
if tx . IsRegistrationValid ( ) {
} else {
err = fmt . Errorf ( "Registration TX is invalid" )
}
} else {
err = fmt . Errorf ( "TX is not registration" )
}
if err != nil {
panic ( err )
}
}
topoheight := chain . Load_TOPO_HEIGHT ( )
toporecord , err := chain . Store . Topo_store . Read ( topoheight )
if err != nil {
panic ( err )
}
ss , err := chain . Store . Balance_store . LoadSnapshot ( toporecord . State_Version )
if err != nil {
panic ( err )
}
balance_tree , err := ss . GetTree ( BALANCE_TREE )
if err != nil {
panic ( err )
}
// use best 3 tips
//tips := chain.SortTips( chain.Load_TIPS_ATOMIC())
tips := chain . SortTips ( chain . Get_TIPS ( ) )
for i := range tips {
if len ( bl . Tips ) >= 3 {
break
}
2020-12-21 15:38:03 +00:00
if ! chain . verifyNonReachabilitytips ( append ( [ ] crypto . Hash { tips [ i ] } , bl . Tips ... ) ) { // avoid any tips which fail reachability test
continue
}
if len ( bl . Tips ) == 0 || ( len ( bl . Tips ) >= 1 && chain . Load_Height_for_BL_ID ( bl . Tips [ 0 ] ) >= chain . Load_Height_for_BL_ID ( tips [ i ] ) && chain . Load_Height_for_BL_ID ( bl . Tips [ 0 ] ) - chain . Load_Height_for_BL_ID ( tips [ i ] ) <= config . STABLE_LIMIT / 2 ) {
bl . Tips = append ( bl . Tips , tips [ i ] )
}
2020-12-19 10:01:29 +00:00
}
//fmt.Printf("miner block placing tips %+v\n", bl.Tips)
height := chain . Calculate_Height_At_Tips ( bl . Tips ) // we are 1 higher than previous highest tip
var tx_hash_list_included [ ] crypto . Hash // these tx will be included ( due to block size limit )
sizeoftxs := uint64 ( 0 ) // size of all non coinbase tx included within this block
//fees_collected := uint64(0)
_ = sizeoftxs
// add upto 100 registration tx each registration tx is 99 bytes, so 100 tx will take 9900 bytes or 10KB
{
tx_hash_list_sorted := chain . Regpool . Regpool_List_TX ( ) // hash of all tx expected to be included within this block , sorted by fees
for i := range tx_hash_list_sorted {
tx := chain . Regpool . Regpool_Get_TX ( tx_hash_list_sorted [ i ] )
if tx != nil {
_ , err = balance_tree . Get ( tx . MinerAddress [ : ] )
if err != nil {
if xerrors . Is ( err , graviton . ErrNotFound ) { // address needs registration
cbl . Txs = append ( cbl . Txs , tx )
tx_hash_list_included = append ( tx_hash_list_included , tx_hash_list_sorted [ i ] )
} else {
panic ( err )
}
}
}
}
}
//rlog.Infof("Total tx in pool %d", len(tx_hash_list_sorted))
//reachable_key_images := chain.BuildReachabilityKeyImages(dbtx, &bl) // this requires only bl.Tips
// select 10% tx based on fees
// select 90% tx randomly
// random selection helps us to easily reach 80 TPS
// first of lets find the tx fees collected by consuming txs from mempool
tx_hash_list_sorted := chain . Mempool . Mempool_List_TX_SortedInfo ( ) // hash of all tx expected to be included within this block , sorted by fees
i := 0
for ; i < len ( tx_hash_list_sorted ) ; i ++ {
if ( sizeoftxs + tx_hash_list_sorted [ i ] . Size ) > ( 10 * config . STARGATE_HE_MAX_BLOCK_SIZE ) / 100 { // limit block to max possible
break
}
tx := chain . Mempool . Mempool_Get_TX ( tx_hash_list_sorted [ i ] . Hash )
if tx != nil && int64 ( tx . Height ) + 1 == height {
/ *
// skip and delete any mempool tx
if chain . Verify_Transaction_NonCoinbase_DoubleSpend_Check ( tx ) == false {
chain . Mempool . Mempool_Delete_TX ( tx_hash_list_sorted [ i ] . Hash )
continue
}
failed := false
for j := 0 ; j < len ( tx . Vin ) ; j ++ {
if _ , ok := reachable_key_images [ tx . Vin [ j ] . ( transaction . Txin_to_key ) . K_image ] ; ok {
rlog . Warnf ( "TX already in history, but tx %s is still in mempool HOW ?? skipping it" , tx_hash_list_sorted [ i ] . Hash )
failed = true
break
}
}
if failed {
continue
}
* /
rlog . Tracef ( 1 , "Adding Top Sorted tx %s to Complete_Block current size %.2f KB max possible %.2f KB\n" , tx_hash_list_sorted [ i ] . Hash , float32 ( sizeoftxs + tx_hash_list_sorted [ i ] . Size ) / 1024.0 , float32 ( config . STARGATE_HE_MAX_BLOCK_SIZE ) / 1024.0 )
sizeoftxs += tx_hash_list_sorted [ i ] . Size
cbl . Txs = append ( cbl . Txs , tx )
tx_hash_list_included = append ( tx_hash_list_included , tx_hash_list_sorted [ i ] . Hash )
}
}
// any left over transactions, should be randomly selected
tx_hash_list_sorted = tx_hash_list_sorted [ i : ]
// do random shuffling, can we get away with len/2 random shuffling
rand . Shuffle ( len ( tx_hash_list_sorted ) , func ( i , j int ) {
tx_hash_list_sorted [ i ] , tx_hash_list_sorted [ j ] = tx_hash_list_sorted [ j ] , tx_hash_list_sorted [ i ]
} )
// if we were crossing limit, transactions would be randomly selected
// otherwise they will sorted by fees
// now select as much as possible
for i := range tx_hash_list_sorted {
if ( sizeoftxs + tx_hash_list_sorted [ i ] . Size ) > ( config . STARGATE_HE_MAX_BLOCK_SIZE ) { // limit block to max possible
break
}
tx := chain . Mempool . Mempool_Get_TX ( tx_hash_list_sorted [ i ] . Hash )
if tx != nil && int64 ( tx . Height ) + 1 == height {
rlog . Tracef ( 1 , "Adding Random tx %s to Complete_Block current size %.2f KB max possible %.2f KB\n" , tx_hash_list_sorted [ i ] . Hash , float32 ( sizeoftxs + tx_hash_list_sorted [ i ] . Size ) / 1024.0 , float32 ( config . STARGATE_HE_MAX_BLOCK_SIZE ) / 1024.0 )
sizeoftxs += tx_hash_list_sorted [ i ] . Size
cbl . Txs = append ( cbl . Txs , tx )
tx_hash_list_included = append ( tx_hash_list_included , tx_hash_list_sorted [ i ] . Hash )
}
}
// collect tx list + their fees
// now we have all major parts of block, assemble the block
bl . Major_Version = uint64 ( chain . Get_Current_Version_at_Height ( height ) )
bl . Minor_Version = uint64 ( chain . Get_Ideal_Version_at_Height ( height ) ) // This is used for hard fork voting,
bl . Height = uint64 ( height )
bl . Timestamp = uint64 ( uint64 ( time . Now ( ) . UTC ( ) . Unix ( ) ) )
//bl.Miner_TX, err = Create_Miner_TX2(int64(bl.Major_Version), height, miner_address)
//if err != nil {
// logger.Warnf("Error while creating miner block, err %s", err)
//}
bl . Miner_TX . Version = 1
bl . Miner_TX . TransactionType = transaction . COINBASE // what about unregistered users
copy ( bl . Miner_TX . MinerAddress [ : ] , miner_address . Compressed ( ) )
// check whether the
_ , err = balance_tree . Get ( bl . Miner_TX . MinerAddress [ : ] )
if err != nil {
if xerrors . Is ( err , graviton . ErrNotFound ) { // address needs registration
bl . Miner_TX . TransactionType = transaction . REGISTRATION
if tx == nil {
err = fmt . Errorf ( "Signature is exactly 64 bytes in size" )
return
}
copy ( bl . Miner_TX . C [ : ] , tx . C [ : 32 ] )
copy ( bl . Miner_TX . S [ : ] , tx . S [ : 32 ] )
} else {
panic ( err )
}
}
//bl.Prev_Hash = top_hash
bl . Nonce = rand . New ( globals . NewCryptoRandSource ( ) ) . Uint32 ( ) // nonce can be changed by the template header
for i := range tx_hash_list_included {
bl . Tx_hashes = append ( bl . Tx_hashes , tx_hash_list_included [ i ] )
}
cbl . Bl = & bl
//logger.Infof("miner block %+v address %X", bl, miner_address.Compressed())
return
}
// returns a new block template ready for mining
// block template has the following format
// miner block header in hex +
// miner tx in hex +
// 2 bytes ( inhex 4 bytes for number of tx )
// tx hashes that follow
var cache_block block . Block
var cache_block_mutex sync . Mutex
func ( chain * Blockchain ) Create_new_block_template_mining ( top_hash crypto . Hash , miner_address address . Address , reserve_space int ) ( bl block . Block , blockhashing_blob string , block_template_blob string , reserved_pos int ) {
rlog . Debugf ( "Mining block will give reward to %s" , miner_address )
cache_block_mutex . Lock ( )
defer cache_block_mutex . Unlock ( )
if ( cache_block . Timestamp + 1 ) < ( uint64 ( uint64 ( time . Now ( ) . UTC ( ) . Unix ( ) ) ) ) || ( cache_block . Timestamp > 0 && int64 ( cache_block . Height ) != chain . Get_Height ( ) + 1 ) {
_ , bl = chain . Create_new_miner_block ( miner_address , nil )
cache_block = bl // setup cache for 1 sec
} else {
bl = cache_block
copy ( bl . Miner_TX . MinerAddress [ : ] , miner_address . Compressed ( ) )
}
blockhashing_blob = fmt . Sprintf ( "%x" , bl . GetBlockWork ( ) )
// block template is all the parts of the block in dismantled form
// first is the block header
// then comes the miner tx
// then comes all the tx headers
block_template_blob = fmt . Sprintf ( "%x" , bl . Serialize ( ) )
// lets locate extra nonce
pos := strings . Index ( blockhashing_blob , "0000000000000000000000000000000000000000000000000000000000000000" )
pos = pos / 2 // we searched in hexadecimal form but we need to give position in byte form
reserved_pos = pos
return
}
// rate limiter is deployed, in case RPC is exposed over internet
// someone should not be just giving fake inputs and delay chain syncing
var accept_limiter = rate . NewLimiter ( 1.0 , 4 ) // 1 block per sec, burst of 4 blocks is okay
var accept_lock = sync . Mutex { }
var duplicate_height_check = map [ uint64 ] bool { }
// accept work given by us
// we should verify that the transaction list supplied back by the miner exists in the mempool
// otherwise the miner is trying to attack the network
func ( chain * Blockchain ) Accept_new_block ( block_template [ ] byte , blockhashing_blob [ ] byte ) ( blid crypto . Hash , result bool , err error ) {
// check whether we are in lowcpu mode, if yes reject the block
if globals . Arguments [ "--lowcpuram" ] . ( bool ) {
globals . Logger . Warnf ( "Mining is deactivated since daemon is running in low cpu mode, please check program options." )
return blid , false , fmt . Errorf ( "Please decativate lowcpuram mode" )
}
if globals . Arguments [ "--sync-node" ] . ( bool ) {
globals . Logger . Warnf ( "Mining is deactivated since daemon is running with --sync-mode, please check program options." )
return blid , false , fmt . Errorf ( "Please deactivate --sync-node option before mining" )
}
accept_lock . Lock ( )
defer accept_lock . Unlock ( )
cbl := & block . Complete_Block { }
bl := block . Block { }
//logger.Infof("Incoming block for accepting %x", block_template)
// safety so if anything wrong happens, verification fails
defer func ( ) {
if r := recover ( ) ; r != nil {
logger . Warnf ( "Recovered while accepting new block, Stack trace below " )
logger . Warnf ( "Stack trace \n%s" , debug . Stack ( ) )
err = fmt . Errorf ( "Error while parsing block" )
}
} ( )
err = bl . Deserialize ( block_template )
if err != nil {
logger . Warnf ( "Error parsing submitted work block template err %s" , err )
return
}
length_of_block_header := len ( bl . Serialize ( ) )
template_data := block_template [ length_of_block_header : ]
if len ( blockhashing_blob ) >= 2 {
err = bl . CopyNonceFromBlockWork ( blockhashing_blob )
if err != nil {
logger . Warnf ( "Submitted block has been rejected, since blockhashing_blob is invalid" )
return
}
}
if len ( template_data ) != 0 {
logger . Warnf ( "Extra bytes (%d) left over while accepting block from mining pool %x" , len ( template_data ) , template_data )
}
// if we reach here, everything looks ok
// try to craft a complete block by grabbing entire tx from the mempool
//logger.Debugf("Block parsed successfully")
blid = bl . GetHash ( )
// if a duplicate block is being sent, reject the block
if _ , ok := duplicate_height_check [ bl . Height ] ; ok {
logger . Warnf ( "Block %s rejected by chain due to duplicate hwork." , bl . GetHash ( ) )
err = fmt . Errorf ( "Error duplicate work" )
return
}
// lets build up the complete block
// collect tx list + their fees
var tx * transaction . Transaction
for i := range bl . Tx_hashes {
tx = chain . Mempool . Mempool_Get_TX ( bl . Tx_hashes [ i ] )
if tx != nil {
cbl . Txs = append ( cbl . Txs , tx )
continue
} else {
tx = chain . Regpool . Regpool_Get_TX ( bl . Tx_hashes [ i ] )
if tx != nil {
cbl . Txs = append ( cbl . Txs , tx )
continue
}
}
var tx_bytes [ ] byte
if tx_bytes , err = chain . Store . Block_tx_store . ReadTX ( bl . Tx_hashes [ i ] ) ; err != nil {
return
}
tx = & transaction . Transaction { }
if err = tx . DeserializeHeader ( tx_bytes ) ; err != nil {
return
}
if err != nil {
logger . Warnf ( "Tx %s not found in pool or DB, rejecting submitted block" , bl . Tx_hashes [ i ] )
return
}
cbl . Txs = append ( cbl . Txs , tx )
}
cbl . Bl = & bl // the block is now complete, lets try to add it to chain
if ! accept_limiter . Allow ( ) { // if rate limiter allows, then add block to chain
logger . Warnf ( "Block %s rejected by chain." , bl . GetHash ( ) )
return
}
err , result = chain . Add_Complete_Block ( cbl )
if result {
duplicate_height_check [ bl . Height ] = true
logger . Infof ( "Block %s successfully accepted at height %d, Notifying Network" , bl . GetHash ( ) , bl . Height )
cache_block_mutex . Lock ( )
defer cache_block_mutex . Unlock ( )
cache_block . Timestamp = 0 // expire cache block
if ! chain . simulator { // if not in simulator mode, relay block to the chain
chain . P2P_Block_Relayer ( cbl , 0 ) // lets relay the block to network
}
} else {
logger . Warnf ( "Block Rejected %s error %s" , bl . GetHash ( ) , err )
}
return
}