// 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 } 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]) } } //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 }