diff --git a/contrib/fullnode.config.ini b/contrib/fullnode.config.ini index fa6182cac4..9a126801a3 100644 --- a/contrib/fullnode.config.ini +++ b/contrib/fullnode.config.ini @@ -1,5 +1,5 @@ # Set larger shared-file-size than default -shared-file-size = 48G +shared-file-size = 54G # Set an API to be publicly available, may be specified multiple times public-api = database_api login_api account_by_key_api network_broadcast_api tag_api follow_api market_history_api raw_block_api diff --git a/contrib/healthcheck.conf.template b/contrib/healthcheck.conf.template index 9df3d404ae..3b690f4cb6 100644 --- a/contrib/healthcheck.conf.template +++ b/contrib/healthcheck.conf.template @@ -13,6 +13,7 @@ proxy_set_header Host $host; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_http_version 1.1; + proxy_buffering off; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection "upgrade"; add_header 'Access-Control-Allow-Origin' '*'; diff --git a/contrib/sync-sv-run.sh b/contrib/sync-sv-run.sh index fe5948f49b..b6b0c11f51 100644 --- a/contrib/sync-sv-run.sh +++ b/contrib/sync-sv-run.sh @@ -38,9 +38,13 @@ if [[ ! -z "$BLOCKCHAIN_TIME" ]]; then echo steemdsync: waiting for steemd to exit cleanly while [ -e /proc/$STEEMD_PID ]; do sleep 0.1; done echo steemdsync: starting a new blockchainstate upload operation - cd $HOME + cd ${COMPRESSPATH:-$HOME} echo steemdsync: compressing blockchainstate... - tar cf blockchain.tar.bz2 --use-compress-prog=pbzip2 blockchain + tar cf blockchain.tar.bz2 --use-compress-prog=pbzip2 -C $HOME blockchain + if [[ ! $? -eq 0 ]]; then + echo NOTIFYALERT! steemdsync was unable to compress shared memory file, check the logs. + exit 1 + fi FILE_NAME=blockchain-$VERSION-`date '+%Y%m%d-%H%M%S'`.tar.bz2 echo steemdsync: uploading $FILE_NAME to $S3_BUCKET aws s3 cp blockchain.tar.bz2 s3://$S3_BUCKET/$FILE_NAME @@ -56,6 +60,7 @@ if [[ ! -z "$BLOCKCHAIN_TIME" ]]; then exit 1 fi # upload a current block_log + cd $HOME aws s3 cp blockchain/block_log s3://$S3_BUCKET/block_log-intransit aws s3 cp s3://$S3_BUCKET/block_log-intransit s3://$S3_BUCKET/block_log-latest # kill the container starting the process over again diff --git a/doc/quickstart.md b/doc/quickstart.md new file mode 100644 index 0000000000..2931ef497d --- /dev/null +++ b/doc/quickstart.md @@ -0,0 +1,57 @@ +Quickstart +---------- + +### Get current steemd +Use docker: +``` +docker run \ + -d -p 2001:2001 -p 8090:8090 --name steemd-default \ + steemit/steem +``` +#### Low memory node? +Above runs low memory node, which is suitable for: +- seed nodes +- witness nodes +- exchanges, etc. +For full api node use: + +``` +docker run \ + --env USE_WAY_TOO_MUCH_RAM=1 --env USE_FULL_WEB_NODE=1 \ + -d -p 2001:2001 -p 8090:8090 --name steemd-full \ + steemit/steem +``` +### Configure for your use case +#### Full API node +You need to use `USE_WAY_TOO_MUCH_RAM=1` and `USE_FULL_WEB_NODE=1` as stated above. +You can Use `contrib/fullnode.config.ini` as a base for your `config.ini` file. + +#### Exchanges +Use low memory node. + +Also make sure that your `config.ini` contains: +``` +enable-plugin = account_history +public-api = database_api login_api +track-account-range = ["yourexchangeid", "yourexchangeid"] +``` +Do not add other APIs or plugins unless you know what you are doing. + +### Resources usage + +Please make sure that you have enough resources available. +Check `shared-file-size =` in your `config.ini` to reflect your needs. +Set it to at least 20% more than current size. + +Provided values are expected to grow significantly over time. + +Blockchain data takes over **12GB** of storage space. + +#### Full node +Shared memory file for full node uses over **45GB** + +#### Seed node +Shared memory file for seed node uses over **3GB** + +#### Other use cases +Shared memory file size varies, depends on your specific configuration but it is expected to be somewhere between "seed node" and "full node" usage. diff --git a/doc/seednodes.txt b/doc/seednodes.txt index 5a5383392b..8f9866e34f 100644 --- a/doc/seednodes.txt +++ b/doc/seednodes.txt @@ -1,36 +1,29 @@ -seed.riversteem.com:2001 # riverhead steem-seed1.abit-more.com:2001 # abit 52.74.152.79:2001 # smooth.witness seed.steemd.com:34191 # roadscape steemwitness.matthewniemerg.com:2001 # complexring -steemd.pharesim.me:2001 # pharesim seed.jesta.us:2001 # jesta seed.liondani.com:2016 # liondani anyx.co:2001 # anyx seed.xeldal.com:12150 # xeldal seed.steemnodes.com:2001 # wackou -steem.clawmap.com:2001 # steempty gtg.steem.house:2001 # gtg 192.99.3.29:2001 # joseph 5.9.18.213:2001 # pfunk lafonasteem.com:2001 # lafona seed.rossco99.com:2001 # rossco99 -212.47.249.84:40696 # ihashfury +steem-seed.altcap.io:40696 # ihashfury seed.steemfeeder.com:2001 # au1nethyb1 -52.175.211.168:2001 # aizensou seed.roelandp.nl:2001 # roelandp 81.89.101.133:2001 # cyrano.witness steem.global:2001 # klye seed.esteem.ws:2001 # good-karma 176.31.126.187:2001 # timcliff -seed.royaltiffany.me:2001 # royaltiffany seed.thecryptodrive.com:2001 # thecryptodrive steem-id.altexplorer.xyz:2001 # steem-id seed.bitcoiner.me:2001 # bitcoiner 104.199.118.92:2001 # clayop 192.99.4.226:2001 # dele-puppy -89.36.26.82:2001 # dragosroua -104.198.222.18:2001 # bitcoinparadise -138.197.17.188:2001 # chitty 88.99.33.113:2001 # cervantes seed.bhuz.info:2001 # bhuz +78.46.95.157:2001 # pcste diff --git a/libraries/app/application.cpp b/libraries/app/application.cpp index e44f19387c..d744670682 100644 --- a/libraries/app/application.cpp +++ b/libraries/app/application.cpp @@ -970,7 +970,7 @@ void application::set_program_options(boost::program_options::options_descriptio ("seed-node,s", bpo::value>()->composing(), "P2P nodes to connect to on startup (may specify multiple times)") ("checkpoint,c", bpo::value>()->composing(), "Pairs of [BLOCK_NUM,BLOCK_ID] that should be enforced as checkpoints.") ("shared-file-dir", bpo::value(), "Location of the shared memory file. Defaults to data_dir/blockchain") - ("shared-file-size", bpo::value()->default_value("32G"), "Size of the shared memory file. Default: 32G") + ("shared-file-size", bpo::value()->default_value("54G"), "Size of the shared memory file. Default: 54G") ("rpc-endpoint", bpo::value()->implicit_value("127.0.0.1:8090"), "Endpoint for websocket RPC to listen on") ("rpc-tls-endpoint", bpo::value()->implicit_value("127.0.0.1:8089"), "Endpoint for TLS websocket RPC to listen on") ("read-forward-rpc", bpo::value(), "Endpoint to forward write API calls to for a read node" ) diff --git a/libraries/app/database_api.cpp b/libraries/app/database_api.cpp index b150e9fb85..753ffea8e7 100755 --- a/libraries/app/database_api.cpp +++ b/libraries/app/database_api.cpp @@ -1145,7 +1145,7 @@ void database_api::set_pending_payout( discussion& d )const vshares = d.net_rshares.value > 0 ? steemit::chain::util::evaluate_reward_curve( d.net_rshares.value, rf.author_reward_curve, rf.content_constant ) : 0; } else - vshares = steemit::chain::util::evaluate_reward_curve( d.net_rshares.value > 0 ? d.net_rshares.value : 0 ); + vshares = d.net_rshares.value > 0 ? steemit::chain::util::evaluate_reward_curve( d.net_rshares.value ) : 0; u256 r2 = to256(vshares); //to256(abs_net_rshares); r2 *= pot.amount.value; diff --git a/libraries/chain/database.cpp b/libraries/chain/database.cpp index 151a89fc6d..c48a327bae 100644 --- a/libraries/chain/database.cpp +++ b/libraries/chain/database.cpp @@ -123,6 +123,7 @@ void database::open( const fc::path& data_dir, const fc::path& shared_mem_dir, u undo_all(); FC_ASSERT( revision() == head_block_num(), "Chainbase revision does not match head block num", ("rev", revision())("head_block", head_block_num()) ); + validate_invariants(); }); if( head_block_num() ) @@ -1150,7 +1151,8 @@ void database::clear_witness_votes( const account_object& a ) remove(current); } - if( has_hardfork( STEEMIT_HARDFORK_0_6__104 ) ) // TODO: this check can be removed after hard fork + #warning( "TODO: Remove this check after HF 19" ) + if( has_hardfork( STEEMIT_HARDFORK_0_6__104 ) ) modify( a, [&](account_object& acc ) { acc.witnesses_voted_for = 0; @@ -1621,7 +1623,7 @@ void database::process_comment_cashout() { fc::microseconds decay_rate; - // TODO: Remove temp fund after HF 19 + #warning( "TODO: Remove temp reward fund after HF 19" ) if( rfo.name == STEEMIT_TEMP_LINEAR_REWARD_FUND_NAME || has_hardfork( STEEMIT_HARDFORK_0_19__1051 ) ) decay_rate = STEEMIT_RECENT_RSHARES_DECAY_RATE_HF19; else @@ -3086,7 +3088,7 @@ void database::update_last_irreversible_block() witness_time_pairs.push_back( std::make_pair( b->data.witness, b->data.timestamp ) ); } - ilog( "Encountered a block num collision due to a fork. Walking the current fork to determine the correct block. block_num:${n}", ("n", log_head_num + 1) ); // TODO: Delete when we know this code works as intended + ilog( "Encountered a block num collision due to a fork. Walking the current fork to determine the correct block. block_num:${n}", ("n", log_head_num + 1) ); ilog( "Colliding blocks produced by witnesses at times: ${w}", ("w", witness_time_pairs) ); auto next = _fork_db.head(); @@ -3747,8 +3749,8 @@ void database::apply_hardfork( uint32_t hardfork ) case STEEMIT_HARDFORK_0_17: { static_assert( - STEEMIT_MAX_VOTED_WITNESSES_HF0 + STEEMIT_MAX_MINER_WITNESSES_HF0 + STEEMIT_MAX_RUNNER_WITNESSES_HF0 == STEEMIT_MAX_WITNESSES, - "HF0 witness counts must add up to STEEMIT_MAX_WITNESSES" ); + STEEMIT_MAX_VOTED_WITNESSES_HF0 + STEEMIT_MAX_MINER_WITNESSES_HF0 + STEEMIT_MAX_RUNNER_WITNESSES_HF0 == STEEMIT_MAX_WITNESSES, + "HF0 witness counts must add up to STEEMIT_MAX_WITNESSES" ); static_assert( STEEMIT_MAX_VOTED_WITNESSES_HF17 + STEEMIT_MAX_MINER_WITNESSES_HF17 + STEEMIT_MAX_RUNNER_WITNESSES_HF17 == STEEMIT_MAX_WITNESSES, "HF17 witness counts must add up to STEEMIT_MAX_WITNESSES" ); @@ -3793,7 +3795,7 @@ void database::apply_hardfork( uint32_t hardfork ) // As a shortcut in payout processing, we use the id as an array index. // The IDs must be assigned this way. The assertion is a dummy check to ensure this happens. FC_ASSERT( post_rf.id._id == 0 ); - FC_ASSERT( linear_rf.id._id == 1 ); + FC_ASSERT( linear_rf.id._id == STEEMIT_TEMP_LINEAR_REWARD_FUND_ID ); modify( gpo, [&]( dynamic_global_property_object& g ) { @@ -3859,14 +3861,21 @@ void database::apply_hardfork( uint32_t hardfork ) const auto& linear = get< reward_fund_object, by_name >( STEEMIT_TEMP_LINEAR_REWARD_FUND_NAME ); modify( get< reward_fund_object, by_name >( STEEMIT_POST_REWARD_FUND_NAME ), [&]( reward_fund_object &rfo ) { + #warning( "TODO: Replace with constant after HF 19" ) rfo.recent_claims = linear.recent_claims; rfo.author_reward_curve = curve_id::linear; rfo.curation_reward_curve = curve_id::square_root; }); + #warning( "TODO: Remove weight conversion after HF 19" ) const auto& cidx = get_index< comment_index, by_cashout_time >(); const auto& vidx = get_index< comment_vote_index, by_comment_voter >(); - for( auto c_itr = cidx.begin(); c_itr != cidx.end() && c_itr->cashout_time != fc::time_point_sec::maximum(); ++c_itr ) + + /* + * Iterator through all comments that have not yet been paid, setting their total vote weight to the sqrt weight + * and update their votes as well. + */ + for( auto c_itr = cidx.begin(); c_itr != cidx.end() && c_itr->cashout_time < fc::time_point_sec::maximum(); ++c_itr ) { modify( *c_itr, [&]( comment_object& c ) { @@ -3882,6 +3891,8 @@ void database::apply_hardfork( uint32_t hardfork ) } } + #warning( "TODO: Remove if 0 delegation opjects are not created in pre HF19 consensus" ) + /* Remove all 0 delegation objects */ vector< const vesting_delegation_object* > to_remove; const auto& delegation_idx = get_index< vesting_delegation_index, by_id >(); auto delegation_itr = delegation_idx.begin(); @@ -3945,7 +3956,7 @@ void database::validate_invariants()const /// verify no witness has too many votes const auto& witness_idx = get_index< witness_index >().indices(); for( auto itr = witness_idx.begin(); itr != witness_idx.end(); ++itr ) - FC_ASSERT( itr->votes < gpo.total_vesting_shares.amount, "", ("itr",*itr) ); + FC_ASSERT( itr->votes <= gpo.total_vesting_shares.amount, "", ("itr",*itr) ); for( auto itr = account_idx.begin(); itr != account_idx.end(); ++itr ) { diff --git a/libraries/chain/hardfork.d/0_6.hf b/libraries/chain/hardfork.d/0_6.hf index aa86b802b5..31962cbd71 100644 --- a/libraries/chain/hardfork.d/0_6.hf +++ b/libraries/chain/hardfork.d/0_6.hf @@ -16,6 +16,10 @@ // Fri Jun 24 14:00:00 UTC 2016 // Fri Jun 24 10:00:00 EDT 2016 +#ifdef IS_TEST_NET +#define STEEMIT_HARDFORK_0_6_REVERSE_AUCTION_TIME (0) +#else #define STEEMIT_HARDFORK_0_6_REVERSE_AUCTION_TIME (1467295200-(60*60*24*6)) +#endif #define STEEMIT_HARDFORK_0_6_VERSION hardfork_version( 0, 6 ) #endif diff --git a/libraries/chain/include/steemit/chain/comment_object.hpp b/libraries/chain/include/steemit/chain/comment_object.hpp index 02f284741c..7ae09db327 100644 --- a/libraries/chain/include/steemit/chain/comment_object.hpp +++ b/libraries/chain/include/steemit/chain/comment_object.hpp @@ -78,6 +78,7 @@ namespace steemit { namespace chain { time_point_sec cashout_time; /// 24 hours from the weighted average of vote time time_point_sec max_cashout_time; uint64_t total_vote_weight = 0; /// the total weight of voting rewards, used to calculate pro-rata share of curation payouts + #warning( "Remove total_sqrt_vote_weight after HF 19" ) uint64_t total_sqrt_vote_weight = 0; /// Temporary total weight of votes for post hf19 curation reward curve uint16_t reward_weight = 0; @@ -121,6 +122,7 @@ namespace steemit { namespace chain { account_id_type voter; comment_id_type comment; uint64_t weight = 0; ///< defines the score this vote receives, used by vote payout calc. 0 if a negative vote or changed votes. + #warning( "Remove sqrt_weight after HF 19. It is a temporary field" ) uint64_t sqrt_weight = 0; ///< Temporary vote weight for post hf19 curation curve int64_t rshares = 0; ///< The number of rshares this vote is responsible for int16_t vote_percent = 0; ///< The percent weight of the vote diff --git a/libraries/chain/include/steemit/chain/immutable_chain_parameters.hpp b/libraries/chain/include/steemit/chain/immutable_chain_parameters.hpp deleted file mode 100644 index d3f2f68644..0000000000 --- a/libraries/chain/include/steemit/chain/immutable_chain_parameters.hpp +++ /dev/null @@ -1,26 +0,0 @@ -#pragma once - -#include - -#include - -#include - -namespace steemit { namespace chain { - -struct immutable_chain_parameters -{ - uint16_t min_committee_member_count = STEEMIT_DEFAULT_MIN_COMMITTEE_MEMBER_COUNT; - uint16_t min_witness_count = STEEMIT_DEFAULT_MIN_WITNESS_COUNT; - uint32_t num_special_accounts = 0; - uint32_t num_special_assets = 0; -}; - -} } // steemit::chain - -FC_REFLECT( steemit::chain::immutable_chain_parameters, - (min_committee_member_count) - (min_witness_count) - (num_special_accounts) - (num_special_assets) -) diff --git a/libraries/chain/steem_evaluator.cpp b/libraries/chain/steem_evaluator.cpp index 4d7be93ac5..a1296449cd 100644 --- a/libraries/chain/steem_evaluator.cpp +++ b/libraries/chain/steem_evaluator.cpp @@ -124,6 +124,7 @@ void account_create_evaluator::do_apply( const account_create_operation& o ) } else if( _db.has_hardfork( STEEMIT_HARDFORK_0_17__818 ) ) { + #warning( "TODO: Remove after HF19. Entire conditional might be able to be removed." ) if( _db.is_producing() ) FC_ASSERT( false, "account_create_operation is temporarily disabled. Please use account_create_with_delegation_operation instead" ); const witness_schedule_object& wso = _db.get_witness_schedule_object(); @@ -271,9 +272,7 @@ void account_create_with_delegation_evaluator::do_apply( const account_create_wi auth.last_owner_update = fc::time_point_sec::min(); }); - /* TODO: Check if not creating 0 delegation objects in HF19 passes consensus - * If it does we can remove the apply_hardfork logic for deleting 0 delegation objects - */ + #warning( "TODO: Check if not creating 0 delegation objects in HF19 passes consensus. Expectation is it should work." ) if( ( _db.has_hardfork( STEEMIT_HARDFORK_0_19__997 ) && o.delegation.amount > 0 ) || !_db.has_hardfork( STEEMIT_HARDFORK_0_19__997 ) ) { @@ -384,7 +383,7 @@ void delete_comment_evaluator::do_apply( const delete_comment_operation& o ) if( _db.has_hardfork( STEEMIT_HARDFORK_0_19__876 ) ) FC_ASSERT( comment.cashout_time != fc::time_point_sec::maximum() ); - // TODO: remove is_producing after HF 19. Check if we can remove conditional altogether after HF. + #warning( "TODO: Remove is_producing check after HF19. Conditional may be removed entirely after HF." ) if( _db.is_producing() || _db.has_hardfork( STEEMIT_HARDFORK_0_19__977 ) ) { FC_ASSERT( comment.net_rshares <= 0, "Cannot delete a comment with net positive votes." ); } @@ -460,8 +459,6 @@ void comment_options_evaluator::do_apply( const comment_options_operation& o ) if( !o.allow_curation_rewards || !o.allow_votes || o.max_accepted_payout < comment.max_accepted_payout ) FC_ASSERT( comment.abs_rshares == 0, "One of the included comment options requires the comment to have no rshares allocated to it." ); - if( !_db.has_hardfork( STEEMIT_HARDFORK_0_17__773) ) // TODO: Remove after hardfork 17 - FC_ASSERT( o.extensions.size() == 0, "Operation extensions for the comment_options_operation are not currently supported." ); FC_ASSERT( comment.allow_curation_rewards >= o.allow_curation_rewards, "Curation rewards cannot be re-enabled." ); FC_ASSERT( comment.allow_votes >= o.allow_votes, "Voting cannot be re-enabled." ); FC_ASSERT( comment.max_accepted_payout >= o.max_accepted_payout, "A comment cannot accept a greater payout." ); @@ -1364,7 +1361,15 @@ void vote_evaluator::do_apply( const vote_operation& o ) uint64_t old_weight = util::evaluate_reward_curve( old_vote_rshares.value, reward_fund.curation_reward_curve, reward_fund.content_constant ).to_uint64(); uint64_t new_weight = util::evaluate_reward_curve( comment.vote_rshares.value, reward_fund.curation_reward_curve, reward_fund.content_constant ).to_uint64(); cv.weight = new_weight - old_weight; + #warning( "TODO: After HF19 caclulate which comment was the first to be paid on sqrt curation and activate the logic based on time" ) + +#ifndef IS_TEST_NET +/* + * Disabling this check so we can test the precalculation logic. + * It is not needed on live because the precalculation is only needed pre HF19 + */ if( !_db.has_hardfork( STEEMIT_HARDFORK_0_19__1052 ) ) +#endif { old_weight = util::evaluate_reward_curve( old_vote_rshares.value, curve_id::square_root ).to_uint64(); new_weight = util::evaluate_reward_curve( comment.vote_rshares.value, curve_id::square_root ).to_uint64(); @@ -1397,8 +1402,9 @@ void vote_evaluator::do_apply( const vote_operation& o ) w *= delta_t; w /= STEEMIT_REVERSE_AUCTION_WINDOW_SECONDS; cv.weight = w.to_uint64(); - +#ifndef IS_TEST_NET if( _db.has_hardfork( STEEMIT_HARDFORK_0_17 ) ) +#endif { uint128_t w(sqrt_max_vote_weight); w *= delta_t; @@ -1418,7 +1424,7 @@ void vote_evaluator::do_apply( const vote_operation& o ) _db.modify( comment, [&]( comment_object& c ) { c.total_vote_weight += max_vote_weight; - c.total_sqrt_vote_weight = sqrt_max_vote_weight; + c.total_sqrt_vote_weight += sqrt_max_vote_weight; }); } if( !_db.has_hardfork( STEEMIT_HARDFORK_0_17__774) ) @@ -1517,6 +1523,7 @@ void vote_evaluator::do_apply( const vote_operation& o ) _db.modify( comment, [&]( comment_object& c ) { c.total_vote_weight -= itr->weight; + c.total_sqrt_vote_weight -= itr->sqrt_weight; }); _db.modify( *itr, [&]( comment_vote_object& cv ) @@ -1525,6 +1532,7 @@ void vote_evaluator::do_apply( const vote_operation& o ) cv.vote_percent = o.weight; cv.last_update = _db.head_block_time(); cv.weight = 0; + cv.sqrt_weight = 0; cv.num_changes += 1; }); @@ -2189,8 +2197,6 @@ void claim_reward_balance_evaluator::do_apply( const claim_reward_balance_operat void delegate_vesting_shares_evaluator::do_apply( const delegate_vesting_shares_operation& op ) { - FC_ASSERT( _db.has_hardfork( STEEMIT_HARDFORK_0_17__818 ), "delegate_vesting_shares_operation is not enabled until HF 17" ); //TODO: Delete after hardfork - const auto& delegator = _db.get_account( op.delegator ); const auto& delegatee = _db.get_account( op.delegatee ); auto delegation = _db.find< vesting_delegation_object, by_delegation >( boost::make_tuple( op.delegator, op.delegatee ) ); @@ -2231,7 +2237,7 @@ void delegate_vesting_shares_evaluator::do_apply( const delegate_vesting_shares_ { auto delta = op.vesting_shares - delegation->vesting_shares; - FC_ASSERT( delta >= min_update, "Steem Power increase is not enough of a different. min_update: ${min}", ("min", min_update) ); + FC_ASSERT( delta >= min_update, "Steem Power increase is not enough of a difference. min_update: ${min}", ("min", min_update) ); FC_ASSERT( available_shares >= op.vesting_shares - delegation->vesting_shares, "Account does not have enough vesting shares to delegate." ); _db.modify( delegator, [&]( account_object& a ) @@ -2256,15 +2262,24 @@ void delegate_vesting_shares_evaluator::do_apply( const delegate_vesting_shares_ if( _db.has_hardfork( STEEMIT_HARDFORK_0_19__971 ) ) { - FC_ASSERT( ( op.vesting_shares.amount == 0 ) - || ( delta >= min_update && op.vesting_shares >= min_delegation ), - "Delegation must be removed or be a significant change and leave a minimum delegation. min_update: ${min_update}, min_delegation: ${min_delegation}", - ("min_update", min_update)("min_delegation", min_delegation) ); + if( op.vesting_shares.amount > 0 ) + { + FC_ASSERT( delta >= min_update, "Steem Power decrease is not enough of a difference. min_update: ${min}", ("min", min_update) ); + FC_ASSERT( op.vesting_shares >= min_delegation, "Delegation must be removed or leave minimum delegation amount of ${v}", ("v", min_delegation) ); + } + else + { + FC_ASSERT( delegation->vesting_shares.amount > 0, "Delegation would set vesting_shares to zero, but it is already zero"); + } } - else // TODO: Check and remove after HF19 + #warning( "TODO: Check and remove conditional after HF19" ) + else { - FC_ASSERT( delta >= min_update, "Steem Power increase is not enough of a different. min_update: ${min}", ("min", min_update) ); - FC_ASSERT( op.vesting_shares >= min_delegation || op.vesting_shares.amount == 0, "Delegation must be removed or leave minimum delegation amount of ${v}", ("v", min_delegation) ); + FC_ASSERT( delta >= min_update, "Steem Power decrease is not enough of a difference. min_update: ${min}", ("min", min_update) ); + if( op.vesting_shares.amount > 0 ) + { + FC_ASSERT( op.vesting_shares >= min_delegation, "Delegation must be removed or leave minimum delegation amount of ${v}", ("v", min_delegation) ); + } } _db.create< vesting_delegation_expiration_object >( [&]( vesting_delegation_expiration_object& obj ) diff --git a/libraries/chain/util/reward.cpp b/libraries/chain/util/reward.cpp index 750bf9fc81..8ca508f856 100644 --- a/libraries/chain/util/reward.cpp +++ b/libraries/chain/util/reward.cpp @@ -84,9 +84,10 @@ uint128_t evaluate_reward_curve( const uint128_t& rshares, const curve_id& curve } break; case linear: - result = approx_sqrt( rshares ); + result = rshares; break; case square_root: + result = approx_sqrt( rshares ); break; } diff --git a/libraries/chain/witness_schedule.cpp b/libraries/chain/witness_schedule.cpp index e2cd2c7718..52ff45ba0c 100644 --- a/libraries/chain/witness_schedule.cpp +++ b/libraries/chain/witness_schedule.cpp @@ -259,7 +259,6 @@ void update_witness_schedule4( database& db ) db.modify( wso, [&]( witness_schedule_object& _wso ) { - // active witnesses has exactly STEEMIT_MAX_WITNESSES elements, asserted above for( size_t i = 0; i < active_witnesses.size(); i++ ) { _wso.current_shuffled_witnesses[i] = active_witnesses[i]; diff --git a/tests/tests/operation_tests.cpp b/tests/tests/operation_tests.cpp index 0494dee515..5f1258deb6 100644 --- a/tests/tests/operation_tests.cpp +++ b/tests/tests/operation_tests.cpp @@ -6651,5 +6651,97 @@ BOOST_AUTO_TEST_CASE( comment_beneficiaries_apply ) FC_LOG_AND_RETHROW() } +#warning( "TODO: This test is not needed after HF19" ) +BOOST_AUTO_TEST_CASE( vote_weight_test ) +{ + try + { + BOOST_TEST_MESSAGE( "Test Comment Beneficiaries" ); + ACTORS( (alice)(bob)(sam)(dave) ) + generate_block(); + + set_price_feed( price( ASSET( "1.000 TESTS" ), ASSET( "1.000 TBD" ) ) ); + + comment_operation comment; + vote_operation vote; + signed_transaction tx; + + comment.author = "alice"; + comment.permlink = "test"; + comment.parent_permlink = "test"; + comment.title = "test"; + comment.body = "foobar"; + + tx.operations.push_back( comment ); + tx.set_expiration( db.head_block_time() + STEEMIT_MAX_TIME_UNTIL_EXPIRATION ); + tx.sign( alice_private_key, db.get_chain_id() ); + db.push_transaction( tx, 0 ); + + generate_blocks( 10 ); + + BOOST_TEST_MESSAGE( "--- Testing initial vote" ); + + vote.voter = "alice"; + vote.author = "alice"; + vote.permlink = "test"; + vote.weight = STEEMIT_100_PERCENT; + + tx.clear(); + tx.operations.push_back( vote ); + tx.sign( alice_private_key, db.get_chain_id() ); + db.push_transaction( tx, 0 ); + + { + const auto& alice = db.get_account( "alice" ); + const auto& alice_comment = db.get_comment( comment.author, comment.permlink ); + const auto& alice_vote = db.get< comment_vote_object, by_comment_voter >( boost::make_tuple( alice_comment.id, alice.id ) ); + + BOOST_REQUIRE( alice_vote.weight == alice_vote.sqrt_weight ); + BOOST_REQUIRE( alice_comment.total_vote_weight == alice_comment.total_sqrt_vote_weight ); + } + + generate_blocks( 10 ); + + BOOST_TEST_MESSAGE( "--- Testing second vote" ); + + vote.voter = "bob"; + + tx.clear(); + tx.operations.push_back( vote ); + tx.sign( bob_private_key, db.get_chain_id() ); + db.push_transaction( tx, 0 ); + + { + const auto& bob = db.get_account( "bob" ); + const auto& alice_comment = db.get_comment( comment.author, comment.permlink ); + const auto& bob_vote = db.get< comment_vote_object, by_comment_voter >( boost::make_tuple( alice_comment.id, bob.id ) ); + + BOOST_REQUIRE( bob_vote.weight == bob_vote.sqrt_weight ); + BOOST_REQUIRE( alice_comment.total_vote_weight == alice_comment.total_sqrt_vote_weight ); + } + + generate_blocks( 10 ); + + BOOST_TEST_MESSAGE( "--- Testing removing a vote" ); + + vote.weight = 0; + + tx.clear(); + tx.operations.push_back( vote ); + tx.sign( bob_private_key, db.get_chain_id() ); + db.push_transaction( tx, 0 ); + + { + const auto& bob = db.get_account( "bob" ); + const auto& alice_comment = db.get_comment( comment.author, comment.permlink ); + const auto& bob_vote = db.get< comment_vote_object, by_comment_voter >( boost::make_tuple( alice_comment.id, bob.id ) ); + + BOOST_REQUIRE( bob_vote.weight == bob_vote.sqrt_weight ); + BOOST_REQUIRE( alice_comment.total_vote_weight == alice_comment.total_sqrt_vote_weight ); + } + } + FC_LOG_AND_RETHROW() +} + BOOST_AUTO_TEST_SUITE_END() #endif diff --git a/tests/tests/operation_time_tests.cpp b/tests/tests/operation_time_tests.cpp index 3f7b5df69e..f3af8da2b7 100644 --- a/tests/tests/operation_time_tests.cpp +++ b/tests/tests/operation_time_tests.cpp @@ -145,7 +145,7 @@ BOOST_AUTO_TEST_CASE( comment_payout_equalize ) const account_object& bob_account = db.get_account("bob"); const account_object& dave_account = db.get_account("dave"); - BOOST_CHECK( alice_account.reward_sbd_balance == ASSET( "10720.000 TBD" ) ); + BOOST_CHECK( alice_account.reward_sbd_balance == ASSET( "14288.000 TBD" ) ); BOOST_CHECK( bob_account.reward_sbd_balance == ASSET( "0.000 TBD" ) ); BOOST_CHECK( dave_account.reward_sbd_balance == alice_account.reward_sbd_balance ); }