diff --git a/cmd/client/wallet/main.go b/cmd/client/wallet/main.go index 0fc13d041f..a639f64c8d 100644 --- a/cmd/client/wallet/main.go +++ b/cmd/client/wallet/main.go @@ -94,7 +94,7 @@ var ( transferSenderPtr = transferCommand.String("from", "0", "Specify the sender account address or index") transferReceiverPtr = transferCommand.String("to", "", "Specify the receiver account") transferAmountPtr = transferCommand.Float64("amount", 0, "Specify the amount to transfer") - transferGasPricePtr = transferCommand.Uint64("gasPrice", 0, "Specify the gas price amount. Unit is Nano.") + transferGasPricePtr = transferCommand.Uint64("gasPrice", 1, "Specify the gas price amount. Unit is Nano.") transferShardIDPtr = transferCommand.Int("shardID", -1, "Specify the shard ID for the transfer") transferToShardIDPtr = transferCommand.Int("toShardID", -1, "Specify the destination shard ID for the transfer") transferInputDataPtr = transferCommand.String("inputData", "", "Base64-encoded input data to embed in the transaction") diff --git a/core/blockchain.go b/core/blockchain.go index 97310214c3..061c80c6cb 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1136,6 +1136,7 @@ func (bc *BlockChain) InsertChain(chain types.Blocks) (int, error) { n, events, logs, err := bc.insertChain(chain) bc.PostChainEvents(events, logs) if err == nil { + // TODO: incorporate these into insertChain (already done in master) for idx, block := range chain { header := block.Header() header.Logger(utils.Logger()).Info(). @@ -1176,6 +1177,8 @@ func (bc *BlockChain) InsertChain(chain types.Blocks) (int, error) { } } + // This should be done after everything about adding a block is done. + bc.PostChainEvents(events, logs) return n, err } diff --git a/core/tx_pool.go b/core/tx_pool.go index c1da5ca2e6..acecac9cfc 100644 --- a/core/tx_pool.go +++ b/core/tx_pool.go @@ -292,7 +292,7 @@ func (pool *TxPool) loop() { defer journal.Stop() // Track the previous head headers for transaction reorgs - //head := pool.chain.CurrentBlock() + head := pool.chain.CurrentBlock() // Keep waiting for and reacting to the various events for { @@ -301,12 +301,11 @@ func (pool *TxPool) loop() { case ev := <-pool.chainHeadCh: if ev.Block != nil { pool.mu.Lock() - //if pool.chainconfig.IsHomestead(ev.Block.Number()) { - // pool.homestead = true - //} - //pool.reset(head.Header(), ev.Block.Header()) - //head = ev.Block - + if pool.chainconfig.IsS3(ev.Block.Epoch()) { + pool.homestead = true + } + pool.reset(head.Header(), ev.Block.Header()) + head = ev.Block pool.mu.Unlock() } // Be unsubscribed due to system stopped diff --git a/internal/configs/sharding/fixedschedule.go b/internal/configs/sharding/fixedschedule.go index 59fc1f92af..1fd9cd71ef 100644 --- a/internal/configs/sharding/fixedschedule.go +++ b/internal/configs/sharding/fixedschedule.go @@ -2,9 +2,6 @@ package shardingconfig import ( "math/big" - "time" - - "github.com/harmony-one/harmony/common/denominations" ) const ( @@ -55,43 +52,6 @@ func (s fixedSchedule) RandomnessStartingEpoch() uint64 { return mainnetRandomnessStartingEpoch } -func (s fixedSchedule) MaxTxAmountLimit() *big.Int { - amountBigInt := big.NewInt(mainnetMaxTxAmountLimit) - amountBigInt = amountBigInt.Mul(amountBigInt, big.NewInt(denominations.One)) - return amountBigInt -} - -func (s fixedSchedule) MaxNumRecentTxsPerAccountLimit() uint64 { - return mainnetMaxNumRecentTxsPerAccountLimit -} - -func (s fixedSchedule) MaxTxPoolSizeLimit() int { - return mainnetMaxTxPoolSizeLimit -} - -func (s fixedSchedule) MaxNumTxsPerBlockLimit() int { - return mainnetMaxNumTxsPerBlockLimit -} - -func (s fixedSchedule) RecentTxDuration() time.Duration { - return mainnetRecentTxDuration -} - -func (s fixedSchedule) EnableTxnThrottling() bool { - return mainnetEnableTxnThrottling -} - -func (s fixedSchedule) TxsThrottleConfig() *TxsThrottleConfig { - return &TxsThrottleConfig{ - MaxTxAmountLimit: s.MaxTxAmountLimit(), - MaxNumRecentTxsPerAccountLimit: s.MaxNumRecentTxsPerAccountLimit(), - MaxTxPoolSizeLimit: s.MaxTxPoolSizeLimit(), - MaxNumTxsPerBlockLimit: s.MaxNumTxsPerBlockLimit(), - RecentTxDuration: s.RecentTxDuration(), - EnableTxnThrottling: s.EnableTxnThrottling(), - } -} - func (s fixedSchedule) GetNetworkID() NetworkID { return DevNet } diff --git a/internal/configs/sharding/localnet.go b/internal/configs/sharding/localnet.go index 2a0f0b302e..47130a6868 100644 --- a/internal/configs/sharding/localnet.go +++ b/internal/configs/sharding/localnet.go @@ -3,9 +3,7 @@ package shardingconfig import ( "fmt" "math/big" - "time" - "github.com/harmony-one/harmony/common/denominations" "github.com/harmony-one/harmony/internal/genesis" ) @@ -26,13 +24,6 @@ const ( localnetConsensusRatio = float64(0.1) localnetRandomnessStartingEpoch = 0 - - localnetMaxTxAmountLimit = 1e3 // unit is in One - localnetMaxNumRecentTxsPerAccountLimit = 1e2 - localnetMaxTxPoolSizeLimit = 8000 - localnetMaxNumTxsPerBlockLimit = 1000 - localnetRecentTxDuration = time.Hour - localnetEnableTxnThrottling = false ) func (localnetSchedule) InstanceForEpoch(epoch *big.Int) Instance { @@ -97,43 +88,6 @@ func (ls localnetSchedule) RandomnessStartingEpoch() uint64 { return localnetRandomnessStartingEpoch } -func (ls localnetSchedule) MaxTxAmountLimit() *big.Int { - amountBigInt := big.NewInt(localnetMaxTxAmountLimit) - amountBigInt = amountBigInt.Mul(amountBigInt, big.NewInt(denominations.One)) - return amountBigInt -} - -func (ls localnetSchedule) MaxNumRecentTxsPerAccountLimit() uint64 { - return localnetMaxNumRecentTxsPerAccountLimit -} - -func (ls localnetSchedule) MaxTxPoolSizeLimit() int { - return localnetMaxTxPoolSizeLimit -} - -func (ls localnetSchedule) MaxNumTxsPerBlockLimit() int { - return localnetMaxNumTxsPerBlockLimit -} - -func (ls localnetSchedule) RecentTxDuration() time.Duration { - return localnetRecentTxDuration -} - -func (ls localnetSchedule) EnableTxnThrottling() bool { - return localnetEnableTxnThrottling -} - -func (ls localnetSchedule) TxsThrottleConfig() *TxsThrottleConfig { - return &TxsThrottleConfig{ - MaxTxAmountLimit: ls.MaxTxAmountLimit(), - MaxNumRecentTxsPerAccountLimit: ls.MaxNumRecentTxsPerAccountLimit(), - MaxTxPoolSizeLimit: ls.MaxTxPoolSizeLimit(), - MaxNumTxsPerBlockLimit: ls.MaxNumTxsPerBlockLimit(), - RecentTxDuration: ls.RecentTxDuration(), - EnableTxnThrottling: ls.EnableTxnThrottling(), - } -} - func (ls localnetSchedule) GetNetworkID() NetworkID { return LocalNet } diff --git a/internal/configs/sharding/mainnet.go b/internal/configs/sharding/mainnet.go index efaeb963e3..2b5c22a5e9 100644 --- a/internal/configs/sharding/mainnet.go +++ b/internal/configs/sharding/mainnet.go @@ -2,9 +2,7 @@ package shardingconfig import ( "math/big" - "time" - "github.com/harmony-one/harmony/common/denominations" "github.com/harmony-one/harmony/internal/genesis" ) @@ -29,13 +27,6 @@ const ( mainnetV1_4Epoch = 46 mainnetV1_5Epoch = 54 - mainnetMaxTxAmountLimit = 1e3 // unit is interface{} One - mainnetMaxNumRecentTxsPerAccountLimit = 1e2 - mainnetMaxTxPoolSizeLimit = 8000 - mainnetMaxNumTxsPerBlockLimit = 1000 - mainnetRecentTxDuration = time.Hour - mainnetEnableTxnThrottling = false - // MainNetHTTPPattern is the http pattern for mainnet. MainNetHTTPPattern = "https://api.s%d.t.hmny.io" // MainNetWSPattern is the websocket pattern for mainnet. @@ -135,43 +126,6 @@ func (ms mainnetSchedule) RandomnessStartingEpoch() uint64 { return mainnetRandomnessStartingEpoch } -func (ms mainnetSchedule) MaxTxAmountLimit() *big.Int { - amountBigInt := big.NewInt(mainnetMaxTxAmountLimit) - amountBigInt = amountBigInt.Mul(amountBigInt, big.NewInt(denominations.One)) - return amountBigInt -} - -func (ms mainnetSchedule) MaxNumRecentTxsPerAccountLimit() uint64 { - return mainnetMaxNumRecentTxsPerAccountLimit -} - -func (ms mainnetSchedule) MaxTxPoolSizeLimit() int { - return mainnetMaxTxPoolSizeLimit -} - -func (ms mainnetSchedule) MaxNumTxsPerBlockLimit() int { - return mainnetMaxNumTxsPerBlockLimit -} - -func (ms mainnetSchedule) RecentTxDuration() time.Duration { - return mainnetRecentTxDuration -} - -func (ms mainnetSchedule) EnableTxnThrottling() bool { - return mainnetEnableTxnThrottling -} - -func (ms mainnetSchedule) TxsThrottleConfig() *TxsThrottleConfig { - return &TxsThrottleConfig{ - MaxTxAmountLimit: ms.MaxTxAmountLimit(), - MaxNumRecentTxsPerAccountLimit: ms.MaxNumRecentTxsPerAccountLimit(), - MaxTxPoolSizeLimit: ms.MaxTxPoolSizeLimit(), - MaxNumTxsPerBlockLimit: ms.MaxNumTxsPerBlockLimit(), - RecentTxDuration: ms.RecentTxDuration(), - EnableTxnThrottling: ms.EnableTxnThrottling(), - } -} - func (ms mainnetSchedule) GetNetworkID() NetworkID { return MainNet } diff --git a/internal/configs/sharding/pangaea.go b/internal/configs/sharding/pangaea.go index 6930e3aa8e..e8a0743188 100644 --- a/internal/configs/sharding/pangaea.go +++ b/internal/configs/sharding/pangaea.go @@ -2,11 +2,8 @@ package shardingconfig import ( "math/big" - "time" "github.com/ethereum/go-ethereum/common" - "github.com/harmony-one/harmony/common/denominations" - "github.com/harmony-one/harmony/internal/genesis" ) @@ -15,8 +12,6 @@ const ( PangaeaHTTPPattern = "https://api.s%d.pga.hmny.io" // PangaeaWSPattern is the websocket pattern for pangaea. PangaeaWSPattern = "wss://ws.s%d.pga.hmny.io" - // transaction throttling disabled on pangaea network - pangaeaEnableTxnThrottling = false ) // PangaeaSchedule is the Pangaea sharding configuration schedule. @@ -64,43 +59,6 @@ func (ps pangaeaSchedule) RandomnessStartingEpoch() uint64 { return mainnetRandomnessStartingEpoch } -func (ps pangaeaSchedule) MaxTxAmountLimit() *big.Int { - amountBigInt := big.NewInt(mainnetMaxTxAmountLimit) - amountBigInt = amountBigInt.Mul(amountBigInt, big.NewInt(denominations.One)) - return amountBigInt -} - -func (ps pangaeaSchedule) MaxNumRecentTxsPerAccountLimit() uint64 { - return mainnetMaxNumRecentTxsPerAccountLimit -} - -func (ps pangaeaSchedule) MaxTxPoolSizeLimit() int { - return mainnetMaxTxPoolSizeLimit -} - -func (ps pangaeaSchedule) MaxNumTxsPerBlockLimit() int { - return mainnetMaxNumTxsPerBlockLimit -} - -func (ps pangaeaSchedule) RecentTxDuration() time.Duration { - return mainnetRecentTxDuration -} - -func (ps pangaeaSchedule) EnableTxnThrottling() bool { - return pangaeaEnableTxnThrottling -} - -func (ps pangaeaSchedule) TxsThrottleConfig() *TxsThrottleConfig { - return &TxsThrottleConfig{ - MaxTxAmountLimit: ps.MaxTxAmountLimit(), - MaxNumRecentTxsPerAccountLimit: ps.MaxNumRecentTxsPerAccountLimit(), - MaxTxPoolSizeLimit: ps.MaxTxPoolSizeLimit(), - MaxNumTxsPerBlockLimit: ps.MaxNumTxsPerBlockLimit(), - RecentTxDuration: ps.RecentTxDuration(), - EnableTxnThrottling: ps.EnableTxnThrottling(), - } -} - func (pangaeaSchedule) GetNetworkID() NetworkID { return Pangaea } diff --git a/internal/configs/sharding/shardingconfig.go b/internal/configs/sharding/shardingconfig.go index e53d8ba2b4..d46c9eaebd 100644 --- a/internal/configs/sharding/shardingconfig.go +++ b/internal/configs/sharding/shardingconfig.go @@ -5,7 +5,6 @@ package shardingconfig import ( "fmt" "math/big" - "time" "github.com/harmony-one/harmony/internal/genesis" ) @@ -37,27 +36,6 @@ type Schedule interface { //RandomnessStartingEpoch returns starting epoch of randonness generation RandomnessStartingEpoch() uint64 - // Max amount limit for a valid transaction - MaxTxAmountLimit() *big.Int - - // Max number of transactions of a particular account per block level - MaxNumRecentTxsPerAccountLimit() uint64 - - // Max total number of transactions allowed as pending transactions in transaction pool - MaxTxPoolSizeLimit() int - - // Max total number of transactions allowed to be processed per block - MaxNumTxsPerBlockLimit() int - - // How long "recent" means for transaction in time Duration unit - RecentTxDuration() time.Duration - - // EnableTxnThrottling is the switch for transaction throttling - EnableTxnThrottling() bool - - // configuration for throttling pending transactions - TxsThrottleConfig() *TxsThrottleConfig - // GetNetworkID() return networkID type. GetNetworkID() NetworkID @@ -90,51 +68,6 @@ type Instance interface { ReshardingEpoch() []*big.Int } -// TxThrottleFlag is the throttling flag for each transaction -// Refer below enum declaration for more context. -type TxThrottleFlag int - -// TxThrottleFlag is determined per transaction -// during the new block proposal and pending transactions throttling -const ( - TxSelect TxThrottleFlag = iota - TxUnselect - TxInvalid -) - -func (result TxThrottleFlag) String() string { - switch result { - case TxSelect: - return "TxSelect" - case TxUnselect: - return "TxUnselect" - case TxInvalid: - return "TxInvalid" - } - return "TxThrottleUnknown" -} - -// TxsThrottleConfig contains configuration for throttling pending transactions per node block -type TxsThrottleConfig struct { - // Max amount limit for a valid transaction - MaxTxAmountLimit *big.Int - - // Max number of transactions of a particular account for the past hour - RecentTxDuration time.Duration - - // Max number of transactions of a particular account for the past hour - MaxNumRecentTxsPerAccountLimit uint64 - - // Max total number of transactions allowed as pending transactions in transaction pool - MaxTxPoolSizeLimit int - - // Max total number of transactions allowed to be processed per block - MaxNumTxsPerBlockLimit int - - // EnableTxnThrottling is the switch for transaction throttling - EnableTxnThrottling bool -} - // genShardingStructure return sharding structure, given shard number and its patterns. func genShardingStructure(shardNum, shardID int, httpPattern, wsPattern string) []map[string]interface{} { res := []map[string]interface{}{} diff --git a/internal/configs/sharding/testnet.go b/internal/configs/sharding/testnet.go index 903b75dd5a..8a549df565 100644 --- a/internal/configs/sharding/testnet.go +++ b/internal/configs/sharding/testnet.go @@ -2,9 +2,7 @@ package shardingconfig import ( "math/big" - "time" - "github.com/harmony-one/harmony/common/denominations" "github.com/harmony-one/harmony/internal/genesis" ) @@ -23,13 +21,6 @@ const ( testnetVdfDifficulty = 10000 // This takes about 20s to finish the vdf - testnetMaxTxAmountLimit = 1e3 // unit is in One - testnetMaxNumRecentTxsPerAccountLimit = 1e2 - testnetMaxTxPoolSizeLimit = 8000 - testnetMaxNumTxsPerBlockLimit = 1000 - testnetRecentTxDuration = time.Hour - testnetEnableTxnThrottling = false - // TestNetHTTPPattern is the http pattern for testnet. TestNetHTTPPattern = "https://api.s%d.b.hmny.io" // TestNetWSPattern is the websocket pattern for testnet. @@ -99,43 +90,6 @@ func (ts testnetSchedule) RandomnessStartingEpoch() uint64 { return mainnetRandomnessStartingEpoch } -func (ts testnetSchedule) MaxTxAmountLimit() *big.Int { - amountBigInt := big.NewInt(testnetMaxTxAmountLimit) - amountBigInt = amountBigInt.Mul(amountBigInt, big.NewInt(denominations.One)) - return amountBigInt -} - -func (ts testnetSchedule) MaxNumRecentTxsPerAccountLimit() uint64 { - return testnetMaxNumRecentTxsPerAccountLimit -} - -func (ts testnetSchedule) MaxTxPoolSizeLimit() int { - return testnetMaxTxPoolSizeLimit -} - -func (ts testnetSchedule) MaxNumTxsPerBlockLimit() int { - return testnetMaxNumTxsPerBlockLimit -} - -func (ts testnetSchedule) RecentTxDuration() time.Duration { - return testnetRecentTxDuration -} - -func (ts testnetSchedule) EnableTxnThrottling() bool { - return testnetEnableTxnThrottling -} - -func (ts testnetSchedule) TxsThrottleConfig() *TxsThrottleConfig { - return &TxsThrottleConfig{ - MaxTxAmountLimit: ts.MaxTxAmountLimit(), - MaxNumRecentTxsPerAccountLimit: ts.MaxNumRecentTxsPerAccountLimit(), - MaxTxPoolSizeLimit: ts.MaxTxPoolSizeLimit(), - MaxNumTxsPerBlockLimit: ts.MaxNumTxsPerBlockLimit(), - RecentTxDuration: ts.RecentTxDuration(), - EnableTxnThrottling: ts.EnableTxnThrottling(), - } -} - func (ts testnetSchedule) GetNetworkID() NetworkID { return TestNet } diff --git a/node/node.go b/node/node.go index 91509a3297..6972ceb288 100644 --- a/node/node.go +++ b/node/node.go @@ -52,8 +52,6 @@ const ( ) const ( - // TxPoolLimit is the limit of transaction pool. - TxPoolLimit = 20000 // NumTryBroadCast is the number of times trying to broadcast NumTryBroadCast = 3 // ClientRxQueueSize is the number of client messages to queue before tail-dropping. @@ -286,23 +284,19 @@ func (node *Node) tryBroadcast(tx *types.Transaction) { // Add new transactions to the pending transaction list. func (node *Node) addPendingTransactions(newTxs types.Transactions) { - txPoolLimit := core.ShardingSchedule.MaxTxPoolSizeLimit() node.pendingTxMutex.Lock() - for _, tx := range newTxs { - if _, ok := node.pendingTransactions[tx.Hash()]; !ok { - node.pendingTransactions[tx.Hash()] = tx - } - if len(node.pendingTransactions) > txPoolLimit { - break - } - } + + node.TxPool.AddRemotes(newTxs) + node.pendingTxMutex.Unlock() - utils.Logger().Info().Int("length of newTxs", len(newTxs)).Int("totalPending", len(node.pendingTransactions)).Msg("Got more transactions") + + pendingCount, queueCount := node.TxPool.Stats() + utils.Logger().Info().Int("length of newTxs", len(newTxs)).Int("totalPending", pendingCount).Int("totalQueued", queueCount).Msg("Got more transactions") } // Add new staking transactions to the pending staking transaction list. func (node *Node) addPendingStakingTransactions(newStakingTxs staking.StakingTransactions) { - txPoolLimit := core.ShardingSchedule.MaxTxPoolSizeLimit() + txPoolLimit := 1000 // TODO: incorporate staking txn into TxPool node.pendingStakingTxMutex.Lock() for _, tx := range newStakingTxs { if _, ok := node.pendingStakingTransactions[tx.Hash()]; !ok { @@ -313,7 +307,7 @@ func (node *Node) addPendingStakingTransactions(newStakingTxs staking.StakingTra } } node.pendingStakingTxMutex.Unlock() - utils.Logger().Info().Int("length of newStakingTxs", len(newStakingTxs)).Int("totalPending", len(node.pendingTransactions)).Msg("Got more staking transactions") + utils.Logger().Info().Int("length of newStakingTxs", len(newStakingTxs)).Int("totalPending", len(node.pendingStakingTransactions)).Msg("Got more staking transactions") } // AddPendingStakingTransaction staking transactions @@ -331,7 +325,6 @@ func (node *Node) AddPendingTransaction(newTx *types.Transaction) { utils.Logger().Info().Str("Hash", newTx.Hash().Hex()).Msg("Broadcasting Tx") node.tryBroadcast(newTx) } - utils.Logger().Debug().Int("totalPending", len(node.pendingTransactions)).Msg("Got ONE more transaction") } // AddPendingReceipts adds one receipt message to pending list. @@ -356,76 +349,6 @@ func (node *Node) AddPendingReceipts(receipts *types.CXReceiptsProof) { utils.Logger().Info().Int("totalPendingReceipts", len(node.pendingCXReceipts)).Msg("Got ONE more receipt message") } -// Take out a subset of valid transactions from the pending transaction list -// Note the pending transaction list will then contain the rest of the txs -func (node *Node) getTransactionsForNewBlock( - coinbase common.Address, -) (types.Transactions, staking.StakingTransactions) { - - txsThrottleConfig := core.ShardingSchedule.TxsThrottleConfig() - - // the next block number to be added in consensus protocol, which is always one more than current chain header block - newBlockNum := node.Blockchain().CurrentBlock().NumberU64() + 1 - - // remove old (> txsThrottleConfigRecentTxDuration) blockNum keys from recentTxsStats and initiailize for the new block - for blockNum := range node.recentTxsStats { - recentTxsBlockNumGap := uint64(txsThrottleConfig.RecentTxDuration / node.BlockPeriod) - if recentTxsBlockNumGap < newBlockNum-blockNum { - delete(node.recentTxsStats, blockNum) - } - } - node.recentTxsStats[newBlockNum] = make(types.BlockTxsCounts) - - // Must update to the correct current state before processing potential txns - if err := node.Worker.UpdateCurrent(coinbase); err != nil { - utils.Logger().Error(). - Err(err). - Msg("Failed updating worker's state before txn selection") - return types.Transactions{}, staking.StakingTransactions{} - } - - node.pendingTxMutex.Lock() - defer node.pendingTxMutex.Unlock() - node.pendingStakingTxMutex.Lock() - defer node.pendingStakingTxMutex.Unlock() - - pendingTransactions := types.Transactions{} - pendingStakingTransactions := staking.StakingTransactions{} - - for _, tx := range node.pendingTransactions { - pendingTransactions = append(pendingTransactions, tx) - } - for _, tx := range node.pendingStakingTransactions { - pendingStakingTransactions = append(pendingStakingTransactions, tx) - } - - selected, unselected, invalid := node.Worker.SelectTransactionsForNewBlock(newBlockNum, pendingTransactions, node.recentTxsStats, txsThrottleConfig, coinbase) - selectedStaking, unselectedStaking, invalidStaking := - node.Worker.SelectStakingTransactionsForNewBlock(newBlockNum, pendingStakingTransactions, coinbase) - - node.pendingTransactions = make(map[common.Hash]*types.Transaction) - for _, unselectedTx := range unselected { - node.pendingTransactions[unselectedTx.Hash()] = unselectedTx - } - utils.Logger().Info(). - Int("remainPending", len(node.pendingTransactions)). - Int("selected", len(selected)). - Int("invalidDiscarded", len(invalid)). - Msg("Selecting Transactions") - - node.pendingStakingTransactions = make(map[common.Hash]*staking.StakingTransaction) - for _, unselectedStakingTx := range unselectedStaking { - node.pendingStakingTransactions[unselectedStakingTx.Hash()] = unselectedStakingTx - } - utils.Logger().Info(). - Int("remainPending", len(node.pendingStakingTransactions)). - Int("selected", len(unselectedStaking)). - Int("invalidDiscarded", len(invalidStaking)). - Msg("Selecting Transactions") - - return selected, selectedStaking -} - func (node *Node) startRxPipeline( receiver p2p.GroupReceiver, queue *msgq.Queue, numWorkers int, ) { diff --git a/node/node_handler_test.go b/node/node_handler_test.go index 6f1f2cc773..44a6d07379 100644 --- a/node/node_handler_test.go +++ b/node/node_handler_test.go @@ -34,11 +34,9 @@ func TestAddNewBlock(t *testing.T) { nodeconfig.SetNetworkType(nodeconfig.Devnet) node := New(host, consensus, testDBFactory, false) - selectedTxs, selectedStakingTxs := node.getTransactionsForNewBlock(common.Address{}) - node.Worker.CommitTransactions( - selectedTxs, selectedStakingTxs, common.Address{}, - func([]types.RPCTransactionError) {}, - ) + txs := make(map[common.Address]types.Transactions) + node.Worker.CommitTransactions(txs, common.Address{}, + func([]types.RPCTransactionError) {}) block, _ := node.Worker.FinalizeNewBlock([]byte{}, []byte{}, 0, common.Address{}, nil, nil) err = node.AddNewBlock(block) @@ -69,12 +67,9 @@ func TestVerifyNewBlock(t *testing.T) { } node := New(host, consensus, testDBFactory, false) - selectedTxs, selectedStakingTxs := node.getTransactionsForNewBlock(common.Address{}) - node.Worker.CommitTransactions( - selectedTxs, selectedStakingTxs, common.Address{}, - func([]types.RPCTransactionError) {}, - ) - + txs := make(map[common.Address]types.Transactions) + node.Worker.CommitTransactions(txs, common.Address{}, + func([]types.RPCTransactionError) {}) block, _ := node.Worker.FinalizeNewBlock([]byte{}, []byte{}, 0, common.Address{}, nil, nil) if err := node.VerifyNewBlock(block); err != nil { diff --git a/node/node_newblock.go b/node/node_newblock.go index 8083c861a0..4c4bdf7c16 100644 --- a/node/node_newblock.go +++ b/node/node_newblock.go @@ -81,9 +81,14 @@ func (node *Node) proposeNewBlock() (*types.Block, error) { coinbase := node.Consensus.SelfAddress // Prepare transactions including staking transactions - selectedTxs, selectedStakingTxs := node.getTransactionsForNewBlock(coinbase) - if err := node.Worker.CommitTransactions( - selectedTxs, selectedStakingTxs, coinbase, + pending, err := node.TxPool.Pending() + if err != nil { + utils.Logger().Err(err).Msg("Failed to fetch pending transactions") + return nil, err + } + + node.Worker.UpdateCurrent(coinbase) + if err := node.Worker.CommitTransactions(pending, coinbase, func(payload []types.RPCTransactionError) { node.errorSink.Lock() for i := range payload { @@ -91,8 +96,7 @@ func (node *Node) proposeNewBlock() (*types.Block, error) { node.errorSink.failedTxns = node.errorSink.failedTxns.Next() } node.errorSink.Unlock() - }, - ); err != nil { + }); err != nil { ctxerror.Log15(utils.GetLogger().Error, ctxerror.New("cannot commit transactions"). WithCause(err)) diff --git a/node/worker/worker.go b/node/worker/worker.go index 8c9b41f2e3..b72484bbcc 100644 --- a/node/worker/worker.go +++ b/node/worker/worker.go @@ -5,6 +5,8 @@ import ( "math/big" "time" + common2 "github.com/harmony-one/harmony/internal/common" + "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/rlp" "github.com/harmony-one/harmony/block" @@ -15,7 +17,6 @@ import ( "github.com/harmony-one/harmony/core/types" "github.com/harmony-one/harmony/core/values" "github.com/harmony-one/harmony/core/vm" - shardingconfig "github.com/harmony-one/harmony/internal/configs/sharding" "github.com/harmony-one/harmony/internal/ctxerror" "github.com/harmony-one/harmony/internal/params" "github.com/harmony-one/harmony/internal/utils" @@ -25,6 +26,8 @@ import ( // environment is the worker's current environment and holds all of the current state information. type environment struct { + signer types.Signer + state *state.DB // apply state changes here gasPool *core.GasPool // available gas used to pack transactions @@ -50,99 +53,98 @@ type Worker struct { gasCeil uint64 } -// Returns a tuple where the first value is the txs sender account address, -// the second is the throttling result enum for the transaction of interest. -// Throttling happens based on the amount, frequency, etc. -func (w *Worker) throttleTxs(selected types.Transactions, recentTxsStats types.RecentTxsStats, txsThrottleConfig *shardingconfig.TxsThrottleConfig, tx *types.Transaction) (common.Address, shardingconfig.TxThrottleFlag) { - var sender common.Address - msg, err := tx.AsMessage(types.MakeSigner(w.config, w.chain.CurrentBlock().Epoch())) - if err != nil { - utils.Logger().Error().Err(err).Str("txId", tx.Hash().Hex()).Msg("Error when parsing tx into message") - } else { - sender = msg.From() - } - - // do not throttle transactions if disabled - if !txsThrottleConfig.EnableTxnThrottling { - return sender, shardingconfig.TxSelect - } - - // already selected max num txs - if len(selected) > txsThrottleConfig.MaxNumTxsPerBlockLimit { - utils.Logger().Info().Str("txId", tx.Hash().Hex()).Int("MaxNumTxsPerBlockLimit", txsThrottleConfig.MaxNumTxsPerBlockLimit).Msg("Throttling tx with max num txs per block limit") - return sender, shardingconfig.TxUnselect - } - - // throttle a single sender sending too many transactions in one block - if tx.Value().Cmp(txsThrottleConfig.MaxTxAmountLimit) > 0 { - utils.Logger().Info().Str("txId", tx.Hash().Hex()).Uint64("MaxTxAmountLimit", txsThrottleConfig.MaxTxAmountLimit.Uint64()).Uint64("txAmount", tx.Value().Uint64()).Msg("Throttling tx with max amount limit") - return sender, shardingconfig.TxInvalid - } - - // throttle too large transaction - var numTxsPastHour uint64 - for _, blockTxsCounts := range recentTxsStats { - numTxsPastHour += blockTxsCounts[sender] - } - if numTxsPastHour >= txsThrottleConfig.MaxNumRecentTxsPerAccountLimit { - utils.Logger().Info().Str("txId", tx.Hash().Hex()).Uint64("MaxNumRecentTxsPerAccountLimit", txsThrottleConfig.MaxNumRecentTxsPerAccountLimit).Msg("Throttling tx with max txs per account in a single block limit") - return sender, shardingconfig.TxInvalid - } - - return sender, shardingconfig.TxSelect -} - -// SelectTransactionsForNewBlock selects transactions for new block. -func (w *Worker) SelectTransactionsForNewBlock(newBlockNum uint64, txs types.Transactions, recentTxsStats types.RecentTxsStats, txsThrottleConfig *shardingconfig.TxsThrottleConfig, coinbase common.Address) (types.Transactions, types.Transactions, types.Transactions) { +// CommitTransactions commits transactions for new block. +func (w *Worker) CommitTransactions( + pendingNormal map[common.Address]types.Transactions, + coinbase common.Address, + txnErrorSink func([]types.RPCTransactionError), +) error { if w.current.gasPool == nil { w.current.gasPool = new(core.GasPool).AddGas(w.current.header.GasLimit()) } - selected := types.Transactions{} - unselected := types.Transactions{} - invalid := types.Transactions{} - for _, tx := range txs { - if tx.ShardID() != w.chain.ShardID() { - invalid = append(invalid, tx) + txs := types.NewTransactionsByPriceAndNonce(w.current.signer, pendingNormal) + coalescedLogs := []*types.Log{} + erroredTxns := []types.RPCTransactionError{} + // NORMAL + for { + // If we don't have enough gas for any further transactions then we're done + if w.current.gasPool.Gas() < params.TxGas { + utils.Logger().Info().Uint64("have", w.current.gasPool.Gas()).Uint64("want", params.TxGas).Msg("Not enough gas for further transactions") + break + } + // Retrieve the next transaction and abort if all done + tx := txs.Peek() + if tx == nil { + break + } + // Error may be ignored here. The error has already been checked + // during transaction acceptance is the transaction pool. + // We use the eip155 signer regardless of the current hf. + from, _ := types.Sender(w.current.signer, tx) + // Check whether the tx is replay protected. If we're not in the EIP155 hf + // phase, start ignoring the sender until we do. + if tx.Protected() && !w.config.IsEIP155(w.current.header.Number()) { + utils.Logger().Info().Str("hash", tx.Hash().Hex()).Str("eip155Epoch", w.config.EIP155Epoch.String()).Msg("Ignoring reply protected transaction") + txs.Pop() continue } + // Start executing the transaction + w.current.state.Prepare(tx.Hash(), common.Hash{}, len(w.current.txs)) - sender, flag := w.throttleTxs(selected, recentTxsStats, txsThrottleConfig, tx) - switch flag { - case shardingconfig.TxUnselect: - unselected = append(unselected, tx) + if tx.ShardID() != w.chain.ShardID() { + txs.Shift() + continue + } - case shardingconfig.TxInvalid: - invalid = append(invalid, tx) + logs, err := w.commitTransaction(tx, coinbase) - case shardingconfig.TxSelect: - snap := w.current.state.Snapshot() - _, err := w.commitTransaction(tx, coinbase) - if err != nil { - w.current.state.RevertToSnapshot(snap) - invalid = append(invalid, tx) - utils.Logger().Error().Err(err).Str("txId", tx.Hash().Hex()).Msg("Commit transaction error") - } else { - selected = append(selected, tx) - // handle the case when msg was not able to extracted from tx - if len(sender.String()) > 0 { - recentTxsStats[newBlockNum][sender]++ - } - } + if err != nil { + erroredTxns = append(erroredTxns, types.RPCTransactionError{ + tx.Hash().Hex(), time.Now().Unix(), err.Error(), + }) } - // log invalid or unselected txs - if flag == shardingconfig.TxUnselect || flag == shardingconfig.TxInvalid { - utils.Logger().Info().Str("txId", tx.Hash().Hex()).Str("txThrottleFlag", flag.String()).Msg("Transaction Throttle flag") + sender, _ := common2.AddressToBech32(from) + switch err { + case core.ErrGasLimitReached: + // Pop the current out-of-gas transaction without shifting in the next from the account + utils.Logger().Info().Str("sender", sender).Msg("Gas limit exceeded for current block") + txs.Pop() + + case core.ErrNonceTooLow: + // New head notification data race between the transaction pool and miner, shift + utils.Logger().Info().Str("sender", sender).Uint64("nonce", tx.Nonce()).Msg("Skipping transaction with low nonce") + txs.Shift() + + case core.ErrNonceTooHigh: + // Reorg notification data race between the transaction pool and miner, skip account = + utils.Logger().Info().Str("sender", sender).Uint64("nonce", tx.Nonce()).Msg("Skipping account with high nonce") + txs.Pop() + + case nil: + // Everything ok, collect the logs and shift in the next transaction from the same account + coalescedLogs = append(coalescedLogs, logs...) + txs.Shift() + + default: + // Strange error, discard the transaction and get the next in line (note, the + // nonce-too-high clause will prevent us from executing in vain). + utils.Logger().Info().Str("hash", tx.Hash().Hex()).AnErr("err", err).Msg("Transaction failed, account skipped") + txs.Shift() } - - utils.Logger().Info().Str("txId", tx.Hash().Hex()).Uint64("txGasLimit", tx.Gas()).Msg("Transaction gas limit info") } - utils.Logger().Info().Uint64("newBlockNum", newBlockNum).Uint64("blockGasLimit", w.current.header.GasLimit()).Uint64("blockGasUsed", w.current.header.GasUsed()).Msg("Block gas limit and usage info") + // Here call the error functions + txnErrorSink(erroredTxns) - return selected, unselected, invalid + utils.Logger().Info(). + Int("newTxns", len(w.current.txs)). + Uint64("blockGasLimit", w.current.header.GasLimit()). + Uint64("blockGasUsed", w.current.header.GasUsed()). + Msg("Block gas limit and usage info") + return nil } // SelectStakingTransactionsForNewBlock selects staking transactions for new block. @@ -155,10 +157,11 @@ func (w *Worker) SelectStakingTransactionsForNewBlock( return nil, nil, nil } + // TODO: gas pool should be initialized once for both normal and staking transactions // staking transaction share the same gasPool with normal transactions - if w.current.gasPool == nil { - w.current.gasPool = new(core.GasPool).AddGas(w.current.header.GasLimit()) - } + //if w.current.gasPool == nil { + // w.current.gasPool = new(core.GasPool).AddGas(w.current.header.GasLimit()) + //} selected := staking.StakingTransactions{} unselected := staking.StakingTransactions{} @@ -176,12 +179,9 @@ func (w *Worker) SelectStakingTransactionsForNewBlock( } } - utils.Logger().Info().Uint64("newBlockNum", newBlockNum).Uint64("blockGasLimit", - w.current.header.GasLimit()).Uint64("blockGasUsed", - w.current.header.GasUsed()).Msg("[SelectStakingTransaction] Block gas limit and usage info") + utils.Logger().Info().Int("newTxns", len(w.current.txs)).Uint64("blockGasLimit", w.current.header.GasLimit()).Uint64("blockGasUsed", w.current.header.GasUsed()).Msg("Block gas limit and usage info") return selected, unselected, invalid - } func (w *Worker) commitStakingTransaction(tx *staking.StakingTransaction, coinbase common.Address) ([]*types.Log, error) { @@ -225,43 +225,6 @@ func (w *Worker) commitTransaction(tx *types.Transaction, coinbase common.Addres return receipt.Logs, nil } -// CommitTransactions commits transactions including staking transactions. -func (w *Worker) CommitTransactions( - txs types.Transactions, - stakingTxns staking.StakingTransactions, - coinbase common.Address, - txnErrorSink func([]types.RPCTransactionError), -) error { - // Must update to the correct current state before processing potential txns - if err := w.UpdateCurrent(coinbase); err != nil { - utils.Logger().Error(). - Err(err). - Msg("Failed updating worker's state before committing txns") - return err - } - - if w.current.gasPool == nil { - w.current.gasPool = new(core.GasPool).AddGas(w.current.header.GasLimit()) - } - - for _, tx := range txs { - snap := w.current.state.Snapshot() - _, err := w.commitTransaction(tx, coinbase) - if err != nil { - txnErrorSink([]types.RPCTransactionError{{ - tx.Hash().Hex(), time.Now().Unix(), err.Error(), - }}) - w.current.state.RevertToSnapshot(snap) - return err - } - } - for _, stakingTx := range stakingTxns { - _ = stakingTx - // TODO: add logic to commit staking txns - } - return nil -} - // CommitReceipts commits a list of already verified incoming cross shard receipts func (w *Worker) CommitReceipts(receiptsList []*types.CXReceiptsProof) error { if w.current.gasPool == nil { @@ -312,6 +275,7 @@ func (w *Worker) makeCurrent(parent *types.Block, header *block.Header) error { return err } env := &environment{ + signer: types.NewEIP155Signer(w.config.ChainID), state: state, header: header, } @@ -420,8 +384,9 @@ func New(config *params.ChainConfig, chain *core.BlockChain, engine consensus_en chain: chain, engine: engine, } - worker.gasFloor = 500000000000000000 - worker.gasCeil = 1000000000000000000 + // 10 time capacity than Ethereum + worker.gasFloor = 80000000 + worker.gasCeil = 120000000 parent := worker.chain.CurrentBlock() num := parent.Number() diff --git a/node/worker/worker_test.go b/node/worker/worker_test.go index c0b463cb9b..0ec6078fb7 100644 --- a/node/worker/worker_test.go +++ b/node/worker/worker_test.go @@ -5,6 +5,8 @@ import ( "math/rand" "testing" + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/crypto" "github.com/ethereum/go-ethereum/ethdb" blockfactory "github.com/harmony-one/harmony/block/factory" @@ -14,7 +16,6 @@ import ( "github.com/harmony-one/harmony/core/vm" chain2 "github.com/harmony-one/harmony/internal/chain" "github.com/harmony-one/harmony/internal/params" - staking "github.com/harmony-one/harmony/staking/types" ) var ( @@ -75,12 +76,14 @@ func TestCommitTransactions(t *testing.T) { tx, _ := types.SignTx(types.NewTransaction(baseNonce, testBankAddress, uint32(0), big.NewInt(int64(denominations.One*randAmount)), params.TxGas, nil, nil), types.HomesteadSigner{}, testBankKey) // Commit the tx to the worker + txs := make(map[common.Address]types.Transactions) + txs[testBankAddress] = types.Transactions{tx} err := worker.CommitTransactions( - types.Transactions{tx}, - staking.StakingTransactions{}, + txs, testBankAddress, func([]types.RPCTransactionError) {}, ) + if err != nil { t.Error(err) } diff --git a/test/chain/main.go b/test/chain/main.go index 3b6a51e096..9506c66f02 100644 --- a/test/chain/main.go +++ b/test/chain/main.go @@ -17,7 +17,6 @@ import ( "github.com/harmony-one/harmony/crypto/hash" "github.com/harmony-one/harmony/internal/params" pkgworker "github.com/harmony-one/harmony/node/worker" - staking "github.com/harmony-one/harmony/staking/types" ) const ( @@ -126,12 +125,12 @@ func fundFaucetContract(chain *core.BlockChain) { amount := 720000 tx, _ := types.SignTx(types.NewTransaction(nonce+uint64(4), StakingAddress, 0, big.NewInt(int64(amount)), params.TxGas, nil, nil), types.HomesteadSigner{}, FaucetPriKey) txs = append(txs, tx) - err := contractworker.CommitTransactions( - txs, - staking.StakingTransactions{}, - testUserAddress, - func([]types.RPCTransactionError) {}, - ) + + txmap := make(map[common.Address]types.Transactions) + txmap[FaucetAddress] = txs + + err := contractworker.CommitTransactions(txmap, testUserAddress, + func([]types.RPCTransactionError) {}) if err != nil { fmt.Println(err) } @@ -169,12 +168,11 @@ func callFaucetContractToFundAnAddress(chain *core.BlockChain) { callEnc = append(callEnc, paddedAddress...) callfaucettx, _ := types.SignTx(types.NewTransaction(nonce+uint64(5), faucetContractAddress, 0, big.NewInt(0), params.TxGasContractCreation*10, nil, callEnc), types.HomesteadSigner{}, FaucetPriKey) - err = contractworker.CommitTransactions( - types.Transactions{callfaucettx}, - staking.StakingTransactions{}, - testUserAddress, - func([]types.RPCTransactionError) {}, - ) + txmap := make(map[common.Address]types.Transactions) + txmap[FaucetAddress] = types.Transactions{callfaucettx} + + err = contractworker.CommitTransactions(txmap, testUserAddress, + func([]types.RPCTransactionError) {}) if err != nil { fmt.Println(err) } @@ -203,153 +201,6 @@ func playFaucetContract(chain *core.BlockChain) { callFaucetContractToFundAnAddress(chain) } -func playSetupStakingContract(chain *core.BlockChain) { - fmt.Println() - fmt.Println("--------- ************************** ---------") - fmt.Println() - fmt.Println("--------- Now Setting up Staking Contract ---------") - fmt.Println() - //worker := pkgworker.New(params.TestChainConfig, chain, consensus.NewFaker(), crypto.PubkeyToAddress(StakingPriKey.PublicKey), 0) - - state = contractworker.GetCurrentState() - fmt.Println("Before Staking Balances") - fmt.Println("user address balance") - fmt.Println(state.GetBalance(allRandomUserAddress[0])) - fmt.Println("The balances for 2 more users:") - fmt.Println(state.GetBalance(allRandomUserAddress[1])) - fmt.Println(state.GetBalance(allRandomUserAddress[2])) - - nonce = contractworker.GetCurrentState().GetNonce(crypto.PubkeyToAddress(StakingPriKey.PublicKey)) - dataEnc = common.FromHex(StakingContractBinary) - stx, _ := types.SignTx(types.NewContractCreation(nonce, 0, big.NewInt(0), params.TxGasContractCreation*10, nil, dataEnc), types.HomesteadSigner{}, StakingPriKey) - stakingtxns = append(stakingtxns, stx) - - stakeContractAddress = crypto.CreateAddress(StakingAddress, nonce+uint64(0)) - - state = contractworker.GetCurrentState() - fmt.Println("stake contract balance :") - fmt.Println(state.GetBalance(stakeContractAddress)) - fmt.Println("stake address balance :") - fmt.Println(state.GetBalance(StakingAddress)) -} - -func playStaking(chain *core.BlockChain) { - depositFnSignature := []byte("deposit()") - - fnHash := hash.Keccak256(depositFnSignature) - methodID := fnHash[:4] - - var callEncl []byte - callEncl = append(callEncl, methodID...) - stake := 100000 - fmt.Println() - fmt.Println("--------- Staking Contract added to txns ---------") - fmt.Println() - fmt.Printf("-- Now Staking with stake: %d --\n", stake) - fmt.Println() - for i := 0; i <= 2; i++ { - //Deposit Does not take a argument, stake is transferred via amount. - tx, _ := types.SignTx(types.NewTransaction(0, stakeContractAddress, 0, big.NewInt(int64(stake)), params.TxGas*5, nil, callEncl), types.HomesteadSigner{}, allRandomUserKey[i]) - stakingtxns = append(stakingtxns, tx) - } - err = contractworker.CommitTransactions( - stakingtxns, - staking.StakingTransactions{}, - common.Address{}, - func([]types.RPCTransactionError) {}, - ) - - if err != nil { - fmt.Println(err) - } - block, _ := contractworker.FinalizeNewBlock([]byte{}, []byte{}, 0, common.Address{}, nil, nil) - _, err = chain.InsertChain(types.Blocks{block}) - if err != nil { - fmt.Println(err) - } - // receipts := contractworker.GetCurrentReceipts() - // fmt.Println(receipts[len(receipts)-4].ContractAddress) - state = contractworker.GetCurrentState() - - fmt.Printf("After Staking Balances (should be less by %d)\n", stake) - fmt.Println("user address balance") - fmt.Println(state.GetBalance(allRandomUserAddress[0])) - fmt.Println("The balances for 2 more users:") - fmt.Println(state.GetBalance(allRandomUserAddress[1])) - fmt.Println(state.GetBalance(allRandomUserAddress[2])) - fmt.Println("faucet contract balance (unchanged):") - fmt.Println(state.GetBalance(faucetContractAddress)) - fmt.Println("stake contract balance :") - fmt.Println(state.GetBalance(stakeContractAddress)) - fmt.Println("stake address balance :") - fmt.Println(state.GetBalance(StakingAddress)) -} - -func playWithdrawStaking(chain *core.BlockChain) { - fmt.Println() - fmt.Println("--------- Now Setting up Withdrawing Stakes ---------") - - withdrawFnSignature := []byte("withdraw(uint256)") - - fnHash := hash.Keccak256(withdrawFnSignature) - methodID := fnHash[:4] - - withdraw := "5000" - withdrawstake := new(big.Int) - withdrawstake.SetString(withdraw, 10) - paddedAmount := common.LeftPadBytes(withdrawstake.Bytes(), 32) - - var dataEncl []byte - dataEncl = append(dataEncl, methodID...) - dataEncl = append(dataEncl, paddedAmount...) - - var withdrawstakingtxns []*types.Transaction - - fmt.Println() - fmt.Printf("-- Withdrawing Stake by amount: %s --\n", withdraw) - fmt.Println() - - for i := 0; i <= 2; i++ { - cnonce := contractworker.GetCurrentState().GetNonce(allRandomUserAddress[i]) - tx, _ := types.SignTx(types.NewTransaction(cnonce, stakeContractAddress, 0, big.NewInt(0), params.TxGas*5, nil, dataEncl), types.HomesteadSigner{}, allRandomUserKey[i]) - withdrawstakingtxns = append(withdrawstakingtxns, tx) - } - - err = contractworker.CommitTransactions( - withdrawstakingtxns, - staking.StakingTransactions{}, - common.Address{}, - func([]types.RPCTransactionError) {}, - ) - if err != nil { - fmt.Println("error:") - fmt.Println(err) - } - - block, _ := contractworker.FinalizeNewBlock([]byte{}, []byte{}, 0, common.Address{}, nil, nil) - _, err = chain.InsertChain(types.Blocks{block}) - if err != nil { - fmt.Println(err) - } - state = contractworker.GetCurrentState() - fmt.Printf("Withdraw Staking Balances (should be up by %s)\n", withdraw) - fmt.Println(state.GetBalance(allRandomUserAddress[0])) - fmt.Println(state.GetBalance(allRandomUserAddress[1])) - fmt.Println(state.GetBalance(allRandomUserAddress[2])) - fmt.Println("faucet contract balance (unchanged):") - fmt.Println(state.GetBalance(faucetContractAddress)) - fmt.Printf("stake contract balance (should downup by %s)\n", withdraw) - fmt.Println(state.GetBalance(stakeContractAddress)) - fmt.Println("stake address balance :") - fmt.Println(state.GetBalance(StakingAddress)) -} - -func playStakingContract(chain *core.BlockChain) { - playSetupStakingContract(chain) - playStaking(chain) - playWithdrawStaking(chain) -} - func main() { genesis := gspec.MustCommit(database) chain, _ := core.NewBlockChain(database, nil, gspec.Config, chain.Engine(), vm.Config{}, nil) @@ -377,5 +228,4 @@ func main() { } playFaucetContract(chain) - playStakingContract(chain) }