1124 lines
50 KiB
C++
1124 lines
50 KiB
C++
/*
|
|
* Copyright (c) 2015 Cryptonomex, Inc., and contributors.
|
|
*
|
|
* The MIT License
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#include <graphene/chain/database.hpp>
|
|
#include <graphene/chain/fba_accumulator_id.hpp>
|
|
|
|
#include <graphene/chain/account_object.hpp>
|
|
#include <graphene/chain/asset_object.hpp>
|
|
#include <graphene/chain/balance_object.hpp>
|
|
#include <graphene/chain/block_summary_object.hpp>
|
|
#include <graphene/chain/budget_record_object.hpp>
|
|
#include <graphene/chain/buyback_object.hpp>
|
|
#include <graphene/chain/chain_property_object.hpp>
|
|
#include <graphene/chain/committee_member_object.hpp>
|
|
#include <graphene/chain/confidential_object.hpp>
|
|
#include <graphene/chain/fba_object.hpp>
|
|
#include <graphene/chain/global_property_object.hpp>
|
|
#include <graphene/chain/market_object.hpp>
|
|
#include <graphene/chain/operation_history_object.hpp>
|
|
#include <graphene/chain/proposal_object.hpp>
|
|
#include <graphene/chain/special_authority_object.hpp>
|
|
#include <graphene/chain/transaction_object.hpp>
|
|
#include <graphene/chain/vesting_balance_object.hpp>
|
|
#include <graphene/chain/withdraw_permission_object.hpp>
|
|
#include <graphene/chain/witness_object.hpp>
|
|
#include <graphene/chain/witness_schedule_object.hpp>
|
|
#include <graphene/chain/worker_object.hpp>
|
|
#include <graphene/chain/tournament_object.hpp>
|
|
#include <graphene/chain/match_object.hpp>
|
|
#include <graphene/chain/game_object.hpp>
|
|
#include <graphene/chain/custom_permission_object.hpp>
|
|
#include <graphene/chain/custom_account_authority_object.hpp>
|
|
#include <graphene/chain/offer_object.hpp>
|
|
#include <graphene/chain/account_role_object.hpp>
|
|
#include <graphene/chain/random_number_object.hpp>
|
|
|
|
#include <graphene/chain/nft_object.hpp>
|
|
|
|
#include <graphene/chain/sport_object.hpp>
|
|
#include <graphene/chain/event_group_object.hpp>
|
|
#include <graphene/chain/event_object.hpp>
|
|
#include <graphene/chain/betting_market_object.hpp>
|
|
#include <graphene/chain/global_betting_statistics_object.hpp>
|
|
#include <graphene/chain/son_object.hpp>
|
|
#include <graphene/chain/son_proposal_object.hpp>
|
|
#include <graphene/chain/son_wallet_object.hpp>
|
|
#include <graphene/chain/son_wallet_deposit_object.hpp>
|
|
#include <graphene/chain/son_wallet_withdraw_object.hpp>
|
|
#include <graphene/chain/sidechain_address_object.hpp>
|
|
#include <graphene/chain/sidechain_transaction_object.hpp>
|
|
|
|
#include <graphene/chain/account_evaluator.hpp>
|
|
#include <graphene/chain/asset_evaluator.hpp>
|
|
#include <graphene/chain/lottery_evaluator.hpp>
|
|
#include <graphene/chain/assert_evaluator.hpp>
|
|
#include <graphene/chain/balance_evaluator.hpp>
|
|
#include <graphene/chain/committee_member_evaluator.hpp>
|
|
#include <graphene/chain/confidential_evaluator.hpp>
|
|
#include <graphene/chain/custom_evaluator.hpp>
|
|
#include <graphene/chain/market_evaluator.hpp>
|
|
#include <graphene/chain/proposal_evaluator.hpp>
|
|
#include <graphene/chain/transfer_evaluator.hpp>
|
|
#include <graphene/chain/vesting_balance_evaluator.hpp>
|
|
#include <graphene/chain/withdraw_permission_evaluator.hpp>
|
|
#include <graphene/chain/witness_evaluator.hpp>
|
|
#include <graphene/chain/worker_evaluator.hpp>
|
|
#include <graphene/chain/sport_evaluator.hpp>
|
|
#include <graphene/chain/event_group_evaluator.hpp>
|
|
#include <graphene/chain/event_evaluator.hpp>
|
|
#include <graphene/chain/betting_market_evaluator.hpp>
|
|
#include <graphene/chain/tournament_evaluator.hpp>
|
|
#include <graphene/chain/custom_permission_evaluator.hpp>
|
|
#include <graphene/chain/custom_account_authority_evaluator.hpp>
|
|
#include <graphene/chain/offer_evaluator.hpp>
|
|
#include <graphene/chain/nft_evaluator.hpp>
|
|
#include <graphene/chain/account_role_evaluator.hpp>
|
|
#include <graphene/chain/nft_lottery_evaluator.hpp>
|
|
#include <graphene/chain/son_evaluator.hpp>
|
|
#include <graphene/chain/son_wallet_evaluator.hpp>
|
|
#include <graphene/chain/son_wallet_deposit_evaluator.hpp>
|
|
#include <graphene/chain/son_wallet_withdraw_evaluator.hpp>
|
|
#include <graphene/chain/sidechain_address_evaluator.hpp>
|
|
#include <graphene/chain/sidechain_transaction_evaluator.hpp>
|
|
#include <graphene/chain/random_number_evaluator.hpp>
|
|
|
|
#include <graphene/chain/protocol/fee_schedule.hpp>
|
|
|
|
#include <fc/uint128.hpp>
|
|
#include <fc/crypto/digest.hpp>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
namespace graphene { namespace chain {
|
|
|
|
// C++ requires that static class variables declared and initialized
|
|
// in headers must also have a definition in a single source file,
|
|
// else linker errors will occur [1].
|
|
//
|
|
// The purpose of this source file is to collect such definitions in
|
|
// a single place.
|
|
//
|
|
// [1] http://stackoverflow.com/questions/8016780/undefined-reference-to-static-constexpr-char
|
|
|
|
const uint8_t account_object::space_id;
|
|
const uint8_t account_object::type_id;
|
|
|
|
const uint8_t asset_object::space_id;
|
|
const uint8_t asset_object::type_id;
|
|
|
|
const uint8_t block_summary_object::space_id;
|
|
const uint8_t block_summary_object::type_id;
|
|
|
|
const uint8_t call_order_object::space_id;
|
|
const uint8_t call_order_object::type_id;
|
|
|
|
const uint8_t committee_member_object::space_id;
|
|
const uint8_t committee_member_object::type_id;
|
|
|
|
const uint8_t force_settlement_object::space_id;
|
|
const uint8_t force_settlement_object::type_id;
|
|
|
|
const uint8_t global_property_object::space_id;
|
|
const uint8_t global_property_object::type_id;
|
|
|
|
const uint8_t limit_order_object::space_id;
|
|
const uint8_t limit_order_object::type_id;
|
|
|
|
const uint8_t operation_history_object::space_id;
|
|
const uint8_t operation_history_object::type_id;
|
|
|
|
const uint8_t proposal_object::space_id;
|
|
const uint8_t proposal_object::type_id;
|
|
|
|
const uint8_t transaction_object::space_id;
|
|
const uint8_t transaction_object::type_id;
|
|
|
|
const uint8_t vesting_balance_object::space_id;
|
|
const uint8_t vesting_balance_object::type_id;
|
|
|
|
const uint8_t withdraw_permission_object::space_id;
|
|
const uint8_t withdraw_permission_object::type_id;
|
|
|
|
const uint8_t witness_object::space_id;
|
|
const uint8_t witness_object::type_id;
|
|
|
|
const uint8_t worker_object::space_id;
|
|
const uint8_t worker_object::type_id;
|
|
|
|
const uint8_t sport_object::space_id;
|
|
const uint8_t sport_object::type_id;
|
|
|
|
const uint8_t event_group_object::space_id;
|
|
const uint8_t event_group_object::type_id;
|
|
|
|
const uint8_t event_object::space_id;
|
|
const uint8_t event_object::type_id;
|
|
|
|
const uint8_t betting_market_rules_object::space_id;
|
|
const uint8_t betting_market_rules_object::type_id;
|
|
|
|
const uint8_t betting_market_group_object::space_id;
|
|
const uint8_t betting_market_group_object::type_id;
|
|
|
|
const uint8_t betting_market_object::space_id;
|
|
const uint8_t betting_market_object::type_id;
|
|
|
|
const uint8_t bet_object::space_id;
|
|
const uint8_t bet_object::type_id;
|
|
|
|
const uint8_t nft_object::space_id;
|
|
const uint8_t nft_object::type_id;
|
|
|
|
const uint8_t betting_market_position_object::space_id;
|
|
const uint8_t betting_market_position_object::type_id;
|
|
|
|
const uint8_t global_betting_statistics_object::space_id;
|
|
const uint8_t global_betting_statistics_object::type_id;
|
|
|
|
const uint8_t offer_object::space_id;
|
|
const uint8_t offer_object::type_id;
|
|
|
|
const uint8_t offer_history_object::space_id;
|
|
const uint8_t offer_history_object::type_id;
|
|
|
|
const uint8_t account_role_object::space_id;
|
|
const uint8_t account_role_object::type_id;
|
|
|
|
const uint8_t nft_lottery_balance_object::space_id;
|
|
const uint8_t nft_lottery_balance_object::type_id;
|
|
|
|
const uint8_t random_number_object::space_id;
|
|
const uint8_t random_number_object::type_id;
|
|
|
|
void database::initialize_evaluators()
|
|
{
|
|
_operation_evaluators.resize(255);
|
|
register_evaluator<account_create_evaluator>();
|
|
register_evaluator<account_update_evaluator>();
|
|
register_evaluator<account_upgrade_evaluator>();
|
|
register_evaluator<account_whitelist_evaluator>();
|
|
register_evaluator<committee_member_create_evaluator>();
|
|
register_evaluator<committee_member_update_evaluator>();
|
|
register_evaluator<committee_member_update_global_parameters_evaluator>();
|
|
register_evaluator<custom_evaluator>();
|
|
register_evaluator<asset_create_evaluator>();
|
|
register_evaluator<asset_issue_evaluator>();
|
|
register_evaluator<asset_reserve_evaluator>();
|
|
register_evaluator<asset_update_evaluator>();
|
|
register_evaluator<asset_update_bitasset_evaluator>();
|
|
register_evaluator<asset_update_dividend_evaluator>();
|
|
register_evaluator<asset_update_feed_producers_evaluator>();
|
|
register_evaluator<asset_settle_evaluator>();
|
|
register_evaluator<asset_global_settle_evaluator>();
|
|
register_evaluator<assert_evaluator>();
|
|
register_evaluator<limit_order_create_evaluator>();
|
|
register_evaluator<limit_order_cancel_evaluator>();
|
|
register_evaluator<call_order_update_evaluator>();
|
|
register_evaluator<transfer_evaluator>();
|
|
register_evaluator<override_transfer_evaluator>();
|
|
register_evaluator<asset_fund_fee_pool_evaluator>();
|
|
register_evaluator<asset_publish_feeds_evaluator>();
|
|
register_evaluator<proposal_create_evaluator>();
|
|
register_evaluator<proposal_update_evaluator>();
|
|
register_evaluator<proposal_delete_evaluator>();
|
|
register_evaluator<vesting_balance_create_evaluator>();
|
|
register_evaluator<vesting_balance_withdraw_evaluator>();
|
|
register_evaluator<witness_create_evaluator>();
|
|
register_evaluator<witness_update_evaluator>();
|
|
register_evaluator<withdraw_permission_create_evaluator>();
|
|
register_evaluator<withdraw_permission_claim_evaluator>();
|
|
register_evaluator<withdraw_permission_update_evaluator>();
|
|
register_evaluator<withdraw_permission_delete_evaluator>();
|
|
register_evaluator<worker_create_evaluator>();
|
|
register_evaluator<balance_claim_evaluator>();
|
|
register_evaluator<transfer_to_blind_evaluator>();
|
|
register_evaluator<transfer_from_blind_evaluator>();
|
|
register_evaluator<blind_transfer_evaluator>();
|
|
register_evaluator<asset_claim_fees_evaluator>();
|
|
register_evaluator<sport_create_evaluator>();
|
|
register_evaluator<sport_update_evaluator>();
|
|
register_evaluator<sport_delete_evaluator>();
|
|
register_evaluator<event_group_create_evaluator>();
|
|
register_evaluator<event_group_update_evaluator>();
|
|
register_evaluator<event_group_delete_evaluator>();
|
|
register_evaluator<event_create_evaluator>();
|
|
register_evaluator<event_update_evaluator>();
|
|
register_evaluator<event_update_status_evaluator>();
|
|
register_evaluator<betting_market_rules_create_evaluator>();
|
|
register_evaluator<betting_market_rules_update_evaluator>();
|
|
register_evaluator<betting_market_group_create_evaluator>();
|
|
register_evaluator<betting_market_group_update_evaluator>();
|
|
register_evaluator<betting_market_create_evaluator>();
|
|
register_evaluator<betting_market_update_evaluator>();
|
|
register_evaluator<bet_place_evaluator>();
|
|
register_evaluator<bet_cancel_evaluator>();
|
|
register_evaluator<betting_market_group_resolve_evaluator>();
|
|
register_evaluator<betting_market_group_cancel_unmatched_bets_evaluator>();
|
|
register_evaluator<tournament_create_evaluator>();
|
|
register_evaluator<tournament_join_evaluator>();
|
|
register_evaluator<game_move_evaluator>();
|
|
register_evaluator<tournament_leave_evaluator>();
|
|
register_evaluator<lottery_asset_create_evaluator>();
|
|
register_evaluator<ticket_purchase_evaluator>();
|
|
register_evaluator<lottery_reward_evaluator>();
|
|
register_evaluator<lottery_end_evaluator>();
|
|
register_evaluator<sweeps_vesting_claim_evaluator>();
|
|
register_evaluator<create_custom_permission_evaluator>();
|
|
register_evaluator<update_custom_permission_evaluator>();
|
|
register_evaluator<delete_custom_permission_evaluator>();
|
|
register_evaluator<create_custom_account_authority_evaluator>();
|
|
register_evaluator<update_custom_account_authority_evaluator>();
|
|
register_evaluator<delete_custom_account_authority_evaluator>();
|
|
register_evaluator<offer_evaluator>();
|
|
register_evaluator<bid_evaluator>();
|
|
register_evaluator<cancel_offer_evaluator>();
|
|
register_evaluator<finalize_offer_evaluator>();
|
|
register_evaluator<nft_metadata_create_evaluator>();
|
|
register_evaluator<nft_metadata_update_evaluator>();
|
|
register_evaluator<nft_mint_evaluator>();
|
|
register_evaluator<nft_safe_transfer_from_evaluator>();
|
|
register_evaluator<nft_approve_evaluator>();
|
|
register_evaluator<nft_set_approval_for_all_evaluator>();
|
|
register_evaluator<account_role_create_evaluator>();
|
|
register_evaluator<account_role_update_evaluator>();
|
|
register_evaluator<account_role_delete_evaluator>();
|
|
register_evaluator<nft_lottery_token_purchase_evaluator>();
|
|
register_evaluator<nft_lottery_reward_evaluator>();
|
|
register_evaluator<nft_lottery_end_evaluator>();
|
|
register_evaluator<create_son_evaluator>();
|
|
register_evaluator<update_son_evaluator>();
|
|
register_evaluator<deregister_son_evaluator>();
|
|
register_evaluator<son_heartbeat_evaluator>();
|
|
register_evaluator<son_report_down_evaluator>();
|
|
register_evaluator<son_maintenance_evaluator>();
|
|
register_evaluator<recreate_son_wallet_evaluator>();
|
|
register_evaluator<update_son_wallet_evaluator>();
|
|
register_evaluator<create_son_wallet_deposit_evaluator>();
|
|
register_evaluator<process_son_wallet_deposit_evaluator>();
|
|
register_evaluator<create_son_wallet_withdraw_evaluator>();
|
|
register_evaluator<process_son_wallet_withdraw_evaluator>();
|
|
register_evaluator<add_sidechain_address_evaluator>();
|
|
register_evaluator<update_sidechain_address_evaluator>();
|
|
register_evaluator<delete_sidechain_address_evaluator>();
|
|
register_evaluator<sidechain_transaction_create_evaluator>();
|
|
register_evaluator<sidechain_transaction_sign_evaluator>();
|
|
register_evaluator<sidechain_transaction_send_evaluator>();
|
|
register_evaluator<sidechain_transaction_settle_evaluator>();
|
|
register_evaluator<random_number_store_evaluator>();
|
|
}
|
|
|
|
void database::initialize_indexes()
|
|
{
|
|
reset_indexes();
|
|
_undo_db.set_max_size( GRAPHENE_MIN_UNDO_HISTORY );
|
|
|
|
//Protocol object indexes
|
|
add_index< primary_index<asset_index, 13> >(); // 8192 assets per chunk
|
|
add_index< primary_index<force_settlement_index> >();
|
|
|
|
auto acnt_index = add_index< primary_index<account_index, 20> >(); // ~1 million accounts per chunk
|
|
acnt_index->add_secondary_index<account_member_index>();
|
|
acnt_index->add_secondary_index<account_referrer_index>();
|
|
|
|
add_index< primary_index<committee_member_index, 8> >(); // 256 members per chunk
|
|
add_index< primary_index<son_index> >();
|
|
add_index< primary_index<witness_index, 10> >(); // 1024 witnesses per chunk
|
|
add_index< primary_index<limit_order_index > >();
|
|
add_index< primary_index<call_order_index > >();
|
|
|
|
auto prop_index = add_index< primary_index<proposal_index > >();
|
|
prop_index->add_secondary_index<required_approval_index>();
|
|
|
|
add_index< primary_index<withdraw_permission_index > >();
|
|
add_index< primary_index<vesting_balance_index> >();
|
|
add_index< primary_index<worker_index> >();
|
|
add_index< primary_index<balance_index> >();
|
|
add_index< primary_index<blinded_balance_index> >();
|
|
add_index< primary_index<sport_object_index > >();
|
|
add_index< primary_index<event_group_object_index > >();
|
|
add_index< primary_index<event_object_index > >();
|
|
add_index< primary_index<betting_market_rules_object_index > >();
|
|
add_index< primary_index<betting_market_group_object_index > >();
|
|
add_index< primary_index<betting_market_object_index > >();
|
|
add_index< primary_index<bet_object_index > >();
|
|
|
|
add_index< primary_index<tournament_index> >();
|
|
auto tournament_details_idx = add_index< primary_index<tournament_details_index> >();
|
|
tournament_details_idx->add_secondary_index<tournament_players_index>();
|
|
add_index< primary_index<match_index> >();
|
|
add_index< primary_index<game_index> >();
|
|
add_index< primary_index<custom_permission_index> >();
|
|
add_index< primary_index<custom_account_authority_index> >();
|
|
auto offer_idx = add_index< primary_index<offer_index> >();
|
|
offer_idx->add_secondary_index<offer_item_index>();
|
|
|
|
add_index< primary_index<nft_metadata_index > >();
|
|
add_index< primary_index<nft_index > >();
|
|
add_index< primary_index<account_role_index> >();
|
|
add_index< primary_index<son_proposal_index> >();
|
|
|
|
add_index< primary_index<son_wallet_index> >();
|
|
add_index< primary_index<son_wallet_deposit_index> >();
|
|
add_index< primary_index<son_wallet_withdraw_index> >();
|
|
|
|
add_index< primary_index<sidechain_address_index> >();
|
|
add_index< primary_index<sidechain_transaction_index> >();
|
|
|
|
//Implementation object indexes
|
|
add_index< primary_index<transaction_index > >();
|
|
|
|
auto bal_idx = add_index< primary_index<account_balance_index > >();
|
|
bal_idx->add_secondary_index<balances_by_account_index>();
|
|
|
|
add_index< primary_index<asset_bitasset_data_index, 13 > >(); // 8192
|
|
add_index< primary_index<asset_dividend_data_object_index > >();
|
|
add_index< primary_index<simple_index<global_property_object >> >();
|
|
add_index< primary_index<simple_index<dynamic_global_property_object >> >();
|
|
add_index< primary_index<account_stats_index > >();
|
|
add_index< primary_index<simple_index<asset_dynamic_data_object >> >();
|
|
add_index< primary_index<flat_index< block_summary_object >> >();
|
|
add_index< primary_index<simple_index<chain_property_object > > >();
|
|
add_index< primary_index<simple_index<witness_schedule_object > > >();
|
|
add_index< primary_index<simple_index<son_schedule_object > > >();
|
|
add_index< primary_index<simple_index<budget_record_object > > >();
|
|
add_index< primary_index< special_authority_index > >();
|
|
add_index< primary_index< buyback_index > >();
|
|
add_index< primary_index< simple_index< fba_accumulator_object > > >();
|
|
add_index< primary_index< betting_market_position_index > >();
|
|
add_index< primary_index< global_betting_statistics_object_index > >();
|
|
//add_index< primary_index<pending_dividend_payout_balance_object_index > >();
|
|
//add_index< primary_index<distributed_dividend_balance_object_index > >();
|
|
add_index< primary_index<pending_dividend_payout_balance_for_holder_object_index > >();
|
|
add_index< primary_index<total_distributed_dividend_balance_object_index > >();
|
|
|
|
add_index< primary_index<lottery_balance_index > >();
|
|
add_index< primary_index<sweeps_vesting_balance_index > >();
|
|
add_index< primary_index<offer_history_index > >();
|
|
add_index< primary_index<nft_lottery_balance_index > >();
|
|
add_index< primary_index<son_stats_index > >();
|
|
add_index< primary_index<random_number_index > >();
|
|
|
|
}
|
|
|
|
void database::init_genesis(const genesis_state_type& genesis_state)
|
|
{ try {
|
|
FC_ASSERT( genesis_state.initial_timestamp != time_point_sec(), "Must initialize genesis timestamp." );
|
|
FC_ASSERT( genesis_state.initial_timestamp.sec_since_epoch() % GRAPHENE_DEFAULT_BLOCK_INTERVAL == 0,
|
|
"Genesis timestamp must be divisible by GRAPHENE_DEFAULT_BLOCK_INTERVAL." );
|
|
FC_ASSERT(genesis_state.initial_witness_candidates.size() > 0,
|
|
"Cannot start a chain with zero witnesses.");
|
|
FC_ASSERT(genesis_state.initial_active_witnesses <= genesis_state.initial_witness_candidates.size(),
|
|
"initial_active_witnesses is larger than the number of candidate witnesses.");
|
|
|
|
_undo_db.disable();
|
|
struct auth_inhibitor {
|
|
auth_inhibitor(database& db) : db(db), old_flags(db.node_properties().skip_flags)
|
|
{ db.node_properties().skip_flags |= skip_authority_check; }
|
|
~auth_inhibitor()
|
|
{ db.node_properties().skip_flags = old_flags; }
|
|
private:
|
|
database& db;
|
|
uint32_t old_flags;
|
|
} inhibitor(*this);
|
|
|
|
transaction_evaluation_state genesis_eval_state(this);
|
|
|
|
flat_index<block_summary_object>& bsi = get_mutable_index_type< flat_index<block_summary_object> >();
|
|
bsi.resize(0xffff+1);
|
|
|
|
// Create blockchain accounts
|
|
fc::ecc::private_key null_private_key = fc::ecc::private_key::regenerate(fc::sha256::hash(string("null_key")));
|
|
create<account_balance_object>([](account_balance_object& b) {
|
|
b.balance = GRAPHENE_MAX_SHARE_SUPPLY;
|
|
});
|
|
const account_object& committee_account =
|
|
create<account_object>( [&](account_object& n) {
|
|
n.membership_expiration_date = time_point_sec::maximum();
|
|
n.network_fee_percentage = GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
n.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT - GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
n.owner.weight_threshold = 1;
|
|
n.active.weight_threshold = 1;
|
|
n.name = "committee-account";
|
|
n.statistics = create<account_statistics_object>( [&n](account_statistics_object& s){
|
|
s.owner = n.id;
|
|
s.name = n.name;
|
|
s.core_in_balance = GRAPHENE_MAX_SHARE_SUPPLY;
|
|
}).id;
|
|
});
|
|
FC_ASSERT(committee_account.get_id() == GRAPHENE_COMMITTEE_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "witness-account";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_WITNESS_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT - GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
}).get_id() == GRAPHENE_WITNESS_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "relaxed-committee-account";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_RELAXED_COMMITTEE_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT - GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
}).get_id() == GRAPHENE_RELAXED_COMMITTEE_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "null-account";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_NULL_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = 0;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT;
|
|
}).get_id() == GRAPHENE_NULL_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "temp-account";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 0;
|
|
a.active.weight_threshold = 0;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_TEMP_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT - GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
}).get_id() == GRAPHENE_TEMP_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "proxy-to-self";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_NULL_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = 0;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT;
|
|
}).get_id() == GRAPHENE_PROXY_TO_SELF_ACCOUNT);
|
|
FC_ASSERT(create<account_object>([this](account_object& a) {
|
|
a.name = "default-dividend-distribution";
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = GRAPHENE_PROXY_TO_SELF_ACCOUNT;
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = 0;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT;
|
|
}).get_id() == GRAPHENE_RAKE_FEE_ACCOUNT_ID);
|
|
// Create more special accounts
|
|
while( true )
|
|
{
|
|
uint64_t id = get_index<account_object>().get_next_id().instance();
|
|
if( id >= genesis_state.immutable_parameters.num_special_accounts )
|
|
break;
|
|
const account_object& acct = create<account_object>([this,id](account_object& a) {
|
|
a.name = "special-account-" + std::to_string(id);
|
|
a.statistics = create<account_statistics_object>([&a](account_statistics_object& s){
|
|
s.owner = a.id;
|
|
s.name = a.name;
|
|
}).id;
|
|
a.owner.weight_threshold = 1;
|
|
a.active.weight_threshold = 1;
|
|
a.registrar = a.lifetime_referrer = a.referrer = account_id_type(id);
|
|
a.membership_expiration_date = time_point_sec::maximum();
|
|
a.network_fee_percentage = GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
a.lifetime_referrer_fee_percentage = GRAPHENE_100_PERCENT - GRAPHENE_DEFAULT_NETWORK_PERCENT_OF_FEE;
|
|
});
|
|
FC_ASSERT( acct.get_id() == account_id_type(id) );
|
|
remove( acct );
|
|
}
|
|
|
|
// Create core asset
|
|
const asset_dynamic_data_object& dyn_asset =
|
|
create<asset_dynamic_data_object>([](asset_dynamic_data_object& a) {
|
|
a.current_supply = GRAPHENE_MAX_SHARE_SUPPLY;
|
|
});
|
|
|
|
const asset_dividend_data_object& div_asset =
|
|
create<asset_dividend_data_object>([&genesis_state](asset_dividend_data_object& a) {
|
|
a.options.minimum_distribution_interval = 3*24*60*60;
|
|
a.options.minimum_fee_percentage = 10*GRAPHENE_1_PERCENT;
|
|
a.options.next_payout_time = genesis_state.initial_timestamp + fc::days(1);
|
|
a.options.payout_interval = 30*24*60*60;
|
|
a.dividend_distribution_account = GRAPHENE_RAKE_FEE_ACCOUNT_ID;
|
|
});
|
|
|
|
const asset_object& core_asset =
|
|
create<asset_object>( [&genesis_state,&div_asset,&dyn_asset]( asset_object& a ) {
|
|
a.symbol = GRAPHENE_SYMBOL;
|
|
a.options.max_supply = genesis_state.max_core_supply;
|
|
a.precision = GRAPHENE_BLOCKCHAIN_PRECISION_DIGITS;
|
|
a.options.flags = 0;
|
|
a.options.issuer_permissions = 0;
|
|
a.issuer = GRAPHENE_COMMITTEE_ACCOUNT;
|
|
a.options.core_exchange_rate.base.amount = 1;
|
|
a.options.core_exchange_rate.base.asset_id = asset_id_type(0);
|
|
a.options.core_exchange_rate.quote.amount = 1;
|
|
a.options.core_exchange_rate.quote.asset_id = asset_id_type(0);
|
|
a.dynamic_asset_data_id = dyn_asset.id;
|
|
a.dividend_data_id = div_asset.id;
|
|
});
|
|
FC_ASSERT( dyn_asset.id == asset_dynamic_data_id_type() );
|
|
FC_ASSERT( asset_id_type(core_asset.id) == asset().asset_id );
|
|
FC_ASSERT( get_balance(account_id_type(), asset_id_type()) == asset(dyn_asset.current_supply) );
|
|
_p_core_asset_obj = &core_asset;
|
|
_p_core_dynamic_data_obj = &dyn_asset;
|
|
|
|
#ifdef _DEFAULT_DIVIDEND_ASSET
|
|
// Create default dividend asset
|
|
const asset_dynamic_data_object& dyn_asset1 =
|
|
create<asset_dynamic_data_object>([&](asset_dynamic_data_object& a) {
|
|
a.current_supply = GRAPHENE_MAX_SHARE_SUPPLY;
|
|
});
|
|
const asset_dividend_data_object& div_asset1 =
|
|
create<asset_dividend_data_object>([&](asset_dividend_data_object& a) {
|
|
a.options.minimum_distribution_interval = 3*24*60*60;
|
|
a.options.minimum_fee_percentage = 10*GRAPHENE_1_PERCENT;
|
|
a.options.next_payout_time = genesis_state.initial_timestamp + fc::hours(1);
|
|
a.options.payout_interval = 7*24*60*60;
|
|
a.dividend_distribution_account = GRAPHENE_RAKE_FEE_ACCOUNT_ID;
|
|
});
|
|
|
|
const asset_object& default_asset =
|
|
create<asset_object>( [&]( asset_object& a ) {
|
|
a.symbol = "DEF";
|
|
a.options.max_market_fee =
|
|
a.options.max_supply = genesis_state.max_core_supply;
|
|
a.precision = GRAPHENE_BLOCKCHAIN_PRECISION_DIGITS;
|
|
a.options.flags = 0;
|
|
a.options.issuer_permissions = 79;
|
|
a.issuer = GRAPHENE_RAKE_FEE_ACCOUNT_ID;
|
|
a.options.core_exchange_rate.base.amount = 1;
|
|
a.options.core_exchange_rate.base.asset_id = asset_id_type(0);
|
|
a.options.core_exchange_rate.quote.amount = 1;
|
|
a.options.core_exchange_rate.quote.asset_id = asset_id_type(1);
|
|
a.dynamic_asset_data_id = dyn_asset1.id;
|
|
a.dividend_data_id = div_asset1.id;
|
|
});
|
|
FC_ASSERT( default_asset.id == asset_id_type(1) );
|
|
#endif
|
|
|
|
// Create more special assets
|
|
while( true )
|
|
{
|
|
uint64_t id = get_index<asset_object>().get_next_id().instance();
|
|
if( id >= genesis_state.immutable_parameters.num_special_assets )
|
|
break;
|
|
const asset_dynamic_data_object& dyn_asset =
|
|
create<asset_dynamic_data_object>([](asset_dynamic_data_object& a) {
|
|
a.current_supply = 0;
|
|
});
|
|
const asset_object& asset_obj = create<asset_object>( [id,&dyn_asset]( asset_object& a ) {
|
|
a.symbol = "SPECIAL" + std::to_string( id );
|
|
a.options.max_supply = 0;
|
|
a.precision = GRAPHENE_BLOCKCHAIN_PRECISION_DIGITS;
|
|
a.options.flags = 0;
|
|
a.options.issuer_permissions = 0;
|
|
a.issuer = GRAPHENE_NULL_ACCOUNT;
|
|
a.options.core_exchange_rate.base.amount = 1;
|
|
a.options.core_exchange_rate.base.asset_id = asset_id_type(0);
|
|
a.options.core_exchange_rate.quote.amount = 1;
|
|
a.options.core_exchange_rate.quote.asset_id = asset_id_type(0);
|
|
a.dynamic_asset_data_id = dyn_asset.id;
|
|
});
|
|
FC_ASSERT( asset_obj.get_id() == asset_id_type(id) );
|
|
remove( asset_obj );
|
|
}
|
|
|
|
chain_id_type chain_id = genesis_state.compute_chain_id();
|
|
|
|
// Create global properties
|
|
_p_global_prop_obj = & create<global_property_object>([&genesis_state](global_property_object& p) {
|
|
p.parameters = genesis_state.initial_parameters;
|
|
// Set fees to zero initially, so that genesis initialization needs not pay them
|
|
// We'll fix it at the end of the function
|
|
p.parameters.current_fees->zero_all_fees();
|
|
|
|
});
|
|
_p_dyn_global_prop_obj = & create<dynamic_global_property_object>([&genesis_state](dynamic_global_property_object& p) {
|
|
p.time = genesis_state.initial_timestamp;
|
|
p.dynamic_flags = 0;
|
|
p.witness_budget = 0;
|
|
p.recent_slots_filled = fc::uint128::max_value();
|
|
});
|
|
create<global_betting_statistics_object>([&](global_betting_statistics_object& betting_statistics) {
|
|
betting_statistics.number_of_active_events = 0;
|
|
});
|
|
|
|
FC_ASSERT( (genesis_state.immutable_parameters.min_witness_count & 1) == 1, "min_witness_count must be odd" );
|
|
FC_ASSERT( (genesis_state.immutable_parameters.min_committee_member_count & 1) == 1, "min_committee_member_count must be odd" );
|
|
|
|
_p_chain_property_obj = & create<chain_property_object>([chain_id,&genesis_state](chain_property_object& p)
|
|
{
|
|
p.chain_id = chain_id;
|
|
p.immutable_parameters = genesis_state.immutable_parameters;
|
|
} );
|
|
create<block_summary_object>([&](block_summary_object&) {});
|
|
|
|
// Create initial accounts from graphene-based chains
|
|
// graphene accounts can refer to other accounts in their authorities, so
|
|
// we first create all accounts with dummy authorities, then go back and
|
|
// set up the authorities once the accounts all have ids assigned.
|
|
for( const auto& account : genesis_state.initial_bts_accounts )
|
|
{
|
|
account_create_operation cop;
|
|
cop.name = account.name;
|
|
cop.registrar = GRAPHENE_TEMP_ACCOUNT;
|
|
cop.owner = authority(1, GRAPHENE_TEMP_ACCOUNT, 1);
|
|
account_id_type account_id(apply_operation(genesis_eval_state, cop).get<object_id_type>());
|
|
}
|
|
|
|
// Create initial accounts
|
|
for( const auto& account : genesis_state.initial_accounts )
|
|
{
|
|
account_create_operation cop;
|
|
cop.name = account.name;
|
|
cop.registrar = GRAPHENE_TEMP_ACCOUNT;
|
|
cop.owner = authority(1, account.owner_key, 1);
|
|
if( account.active_key == public_key_type() )
|
|
{
|
|
cop.active = cop.owner;
|
|
cop.options.memo_key = account.owner_key;
|
|
}
|
|
else
|
|
{
|
|
cop.active = authority(1, account.active_key, 1);
|
|
cop.options.memo_key = account.active_key;
|
|
}
|
|
account_id_type account_id(apply_operation(genesis_eval_state, cop).get<object_id_type>());
|
|
|
|
if( account.is_lifetime_member )
|
|
{
|
|
account_upgrade_operation op;
|
|
op.account_to_upgrade = account_id;
|
|
op.upgrade_to_lifetime_member = true;
|
|
apply_operation(genesis_eval_state, op);
|
|
}
|
|
}
|
|
|
|
// Helper function to get account ID by name
|
|
const auto& accounts_by_name = get_index_type<account_index>().indices().get<by_name>();
|
|
auto get_account_id = [&accounts_by_name](const string& name) {
|
|
auto itr = accounts_by_name.find(name);
|
|
FC_ASSERT(itr != accounts_by_name.end(),
|
|
"Unable to find account '${acct}'. Did you forget to add a record for it to initial_accounts?",
|
|
("acct", name));
|
|
return itr->get_id();
|
|
};
|
|
|
|
for( const auto& account : genesis_state.initial_bts_accounts )
|
|
{
|
|
account_update_operation op;
|
|
op.account = get_account_id(account.name);
|
|
|
|
authority owner_authority;
|
|
owner_authority.weight_threshold = account.owner_authority.weight_threshold;
|
|
for (const auto& value : account.owner_authority.account_auths)
|
|
owner_authority.account_auths.insert(std::make_pair(get_account_id(value.first), value.second));
|
|
owner_authority.key_auths = account.owner_authority.key_auths;
|
|
owner_authority.address_auths = account.owner_authority.address_auths;
|
|
|
|
op.owner = std::move(owner_authority);
|
|
|
|
authority active_authority;
|
|
active_authority.weight_threshold = account.active_authority.weight_threshold;
|
|
for (const auto& value : account.active_authority.account_auths)
|
|
active_authority.account_auths.insert(std::make_pair(get_account_id(value.first), value.second));
|
|
active_authority.key_auths = account.active_authority.key_auths;
|
|
active_authority.address_auths = account.active_authority.address_auths;
|
|
|
|
op.active = std::move(active_authority);
|
|
|
|
apply_operation(genesis_eval_state, op);
|
|
}
|
|
|
|
// Helper function to get asset ID by symbol
|
|
const auto& assets_by_symbol = get_index_type<asset_index>().indices().get<by_symbol>();
|
|
const auto get_asset_id = [&assets_by_symbol](const string& symbol) {
|
|
auto itr = assets_by_symbol.find(symbol);
|
|
|
|
// TODO: This is temporary for handling BTS snapshot
|
|
if( symbol == "BTS" )
|
|
itr = assets_by_symbol.find(GRAPHENE_SYMBOL);
|
|
|
|
FC_ASSERT(itr != assets_by_symbol.end(),
|
|
"Unable to find asset '${sym}'. Did you forget to add a record for it to initial_assets?",
|
|
("sym", symbol));
|
|
return itr->get_id();
|
|
};
|
|
|
|
map<asset_id_type, share_type> total_supplies;
|
|
map<asset_id_type, share_type> total_debts;
|
|
|
|
// Create initial assets
|
|
for( const genesis_state_type::initial_asset_type& asset : genesis_state.initial_assets )
|
|
{
|
|
asset_id_type new_asset_id = get_index_type<asset_index>().get_next_id();
|
|
total_supplies[ new_asset_id ] = 0;
|
|
|
|
asset_dynamic_data_id_type dynamic_data_id;
|
|
optional<asset_bitasset_data_id_type> bitasset_data_id;
|
|
if( asset.is_bitasset )
|
|
{
|
|
int collateral_holder_number = 0;
|
|
total_debts[ new_asset_id ] = 0;
|
|
for( const auto& collateral_rec : asset.collateral_records )
|
|
{
|
|
account_create_operation cop;
|
|
cop.name = asset.symbol + "-collateral-holder-" + std::to_string(collateral_holder_number);
|
|
boost::algorithm::to_lower(cop.name);
|
|
cop.registrar = GRAPHENE_TEMP_ACCOUNT;
|
|
cop.owner = authority(1, collateral_rec.owner, 1);
|
|
cop.active = cop.owner;
|
|
account_id_type owner_account_id = apply_operation(genesis_eval_state, cop).get<object_id_type>();
|
|
|
|
modify( owner_account_id(*this).statistics(*this), [&collateral_rec]( account_statistics_object& o ) {
|
|
o.total_core_in_orders = collateral_rec.collateral;
|
|
});
|
|
|
|
create<call_order_object>([&](call_order_object& c) {
|
|
c.borrower = owner_account_id;
|
|
c.collateral = collateral_rec.collateral;
|
|
c.debt = collateral_rec.debt;
|
|
c.call_price = price::call_price(chain::asset(c.debt, new_asset_id),
|
|
chain::asset(c.collateral, core_asset.id),
|
|
GRAPHENE_DEFAULT_MAINTENANCE_COLLATERAL_RATIO);
|
|
});
|
|
|
|
total_supplies[ asset_id_type(0) ] += collateral_rec.collateral;
|
|
total_debts[ new_asset_id ] += collateral_rec.debt;
|
|
++collateral_holder_number;
|
|
}
|
|
|
|
bitasset_data_id = create<asset_bitasset_data_object>([&](asset_bitasset_data_object& b) {
|
|
b.options.short_backing_asset = core_asset.id;
|
|
b.options.minimum_feeds = GRAPHENE_DEFAULT_MINIMUM_FEEDS;
|
|
}).id;
|
|
}
|
|
|
|
dynamic_data_id = create<asset_dynamic_data_object>([&](asset_dynamic_data_object& d) {
|
|
d.accumulated_fees = asset.accumulated_fees;
|
|
}).id;
|
|
|
|
total_supplies[ new_asset_id ] += asset.accumulated_fees;
|
|
|
|
create<asset_object>([&](asset_object& a) {
|
|
a.symbol = asset.symbol;
|
|
a.options.description = asset.description;
|
|
a.precision = asset.precision;
|
|
string issuer_name = asset.issuer_name;
|
|
a.issuer = get_account_id(issuer_name);
|
|
a.options.max_supply = asset.max_supply;
|
|
a.options.flags = witness_fed_asset;
|
|
a.options.issuer_permissions = charge_market_fee | override_authority | white_list | transfer_restricted | disable_confidential |
|
|
( asset.is_bitasset ? disable_force_settle | global_settle | witness_fed_asset | committee_fed_asset : 0 );
|
|
a.dynamic_asset_data_id = dynamic_data_id;
|
|
a.bitasset_data_id = bitasset_data_id;
|
|
});
|
|
}
|
|
|
|
// Create balances for all bts accounts
|
|
for( const auto& account : genesis_state.initial_bts_accounts ) {
|
|
if (account.core_balance != share_type()) {
|
|
total_supplies[asset_id_type()] += account.core_balance;
|
|
|
|
create<account_balance_object>([&](account_balance_object& b) {
|
|
b.owner = get_account_id(account.name);
|
|
b.balance = account.core_balance;
|
|
});
|
|
}
|
|
|
|
// create any vesting balances for this account
|
|
if (account.vesting_balances)
|
|
for (const auto& vesting_balance : *account.vesting_balances) {
|
|
create<vesting_balance_object>([&](vesting_balance_object& vbo) {
|
|
vbo.owner = get_account_id(account.name);
|
|
vbo.balance = asset(vesting_balance.amount, get_asset_id(vesting_balance.asset_symbol));
|
|
if (vesting_balance.policy_type == "linear") {
|
|
auto initial_linear_vesting_policy = vesting_balance.policy.as<genesis_state_type::initial_bts_account_type::initial_linear_vesting_policy>( 20 );
|
|
linear_vesting_policy new_vesting_policy;
|
|
new_vesting_policy.begin_timestamp = initial_linear_vesting_policy.begin_timestamp;
|
|
new_vesting_policy.vesting_cliff_seconds = initial_linear_vesting_policy.vesting_cliff_seconds;
|
|
new_vesting_policy.vesting_duration_seconds = initial_linear_vesting_policy.vesting_duration_seconds;
|
|
new_vesting_policy.begin_balance = initial_linear_vesting_policy.begin_balance;
|
|
vbo.policy = new_vesting_policy;
|
|
} else if (vesting_balance.policy_type == "cdd") {
|
|
auto initial_cdd_vesting_policy = vesting_balance.policy.as<genesis_state_type::initial_bts_account_type::initial_cdd_vesting_policy>( 20 );
|
|
cdd_vesting_policy new_vesting_policy;
|
|
new_vesting_policy.vesting_seconds = initial_cdd_vesting_policy.vesting_seconds;
|
|
new_vesting_policy.coin_seconds_earned = initial_cdd_vesting_policy.coin_seconds_earned;
|
|
new_vesting_policy.start_claim = initial_cdd_vesting_policy.start_claim;
|
|
new_vesting_policy.coin_seconds_earned_last_update = initial_cdd_vesting_policy.coin_seconds_earned_last_update;
|
|
vbo.policy = new_vesting_policy;
|
|
}
|
|
total_supplies[get_asset_id(vesting_balance.asset_symbol)] += vesting_balance.amount;
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
// Create initial balances
|
|
share_type total_allocation;
|
|
for( const auto& handout : genesis_state.initial_balances )
|
|
{
|
|
const auto asset_id = get_asset_id(handout.asset_symbol);
|
|
create<balance_object>([&handout,&get_asset_id,total_allocation,asset_id](balance_object& b) {
|
|
b.balance = asset(handout.amount, asset_id);
|
|
b.owner = handout.owner;
|
|
});
|
|
|
|
total_supplies[ asset_id ] += handout.amount;
|
|
}
|
|
|
|
// Create initial vesting balances
|
|
for( const genesis_state_type::initial_vesting_balance_type& vest : genesis_state.initial_vesting_balances )
|
|
{
|
|
const auto asset_id = get_asset_id(vest.asset_symbol);
|
|
create<balance_object>([&](balance_object& b) {
|
|
b.owner = vest.owner;
|
|
b.balance = asset(vest.amount, asset_id);
|
|
|
|
linear_vesting_policy policy;
|
|
policy.begin_timestamp = vest.begin_timestamp;
|
|
policy.vesting_cliff_seconds = vest.vesting_cliff_seconds ? *vest.vesting_cliff_seconds : 0;
|
|
policy.vesting_duration_seconds = vest.vesting_duration_seconds;
|
|
policy.begin_balance = vest.begin_balance;
|
|
|
|
b.vesting_policy = std::move(policy);
|
|
});
|
|
|
|
total_supplies[ asset_id ] += vest.amount;
|
|
}
|
|
|
|
if( total_supplies[ asset_id_type(0) ] > 0 )
|
|
{
|
|
adjust_balance(GRAPHENE_COMMITTEE_ACCOUNT, -get_balance(GRAPHENE_COMMITTEE_ACCOUNT,{}));
|
|
}
|
|
else
|
|
{
|
|
total_supplies[ asset_id_type(0) ] = GRAPHENE_MAX_SHARE_SUPPLY;
|
|
}
|
|
#ifdef _DEFAULT_DIVIDEND_ASSET
|
|
total_debts[ asset_id_type(1) ] =
|
|
total_supplies[ asset_id_type(1) ] = 0;
|
|
#endif
|
|
// it is workaround, should be clarified
|
|
total_debts[ asset_id_type() ] = total_supplies[ asset_id_type() ];
|
|
|
|
const auto& idx = get_index_type<asset_index>().indices().get<by_symbol>();
|
|
auto it = idx.begin();
|
|
|
|
while( it != idx.end() )
|
|
{
|
|
if( it->bitasset_data_id.valid() )
|
|
{
|
|
auto supply_itr = total_supplies.find( it->id );
|
|
auto debt_itr = total_debts.find( it->id );
|
|
FC_ASSERT( supply_itr != total_supplies.end() );
|
|
FC_ASSERT( debt_itr != total_debts.end() );
|
|
if( supply_itr->second != debt_itr->second )
|
|
{
|
|
elog( "Genesis for asset ${aname} is not balanced\n"
|
|
" Debt is ${debt}\n"
|
|
" Supply is ${supply}\n",
|
|
("aname", debt_itr->first)
|
|
("debt", debt_itr->second)
|
|
("supply", supply_itr->second)
|
|
);
|
|
}
|
|
}
|
|
++it;
|
|
}
|
|
|
|
// Save tallied supplies
|
|
for( const auto& item : total_supplies )
|
|
{
|
|
const auto asset_id = item.first;
|
|
const auto total_supply = item.second;
|
|
|
|
modify( get( asset_id ), [ & ]( asset_object& asset ) {
|
|
modify( get( asset.dynamic_asset_data_id ), [ & ]( asset_dynamic_data_object& asset_data ) {
|
|
asset_data.current_supply = total_supply;
|
|
} );
|
|
} );
|
|
}
|
|
|
|
// Create special witness account
|
|
const witness_object& wit = create<witness_object>([&](witness_object& w) {});
|
|
FC_ASSERT( wit.id == GRAPHENE_NULL_WITNESS );
|
|
remove(wit);
|
|
|
|
// Create initial witnesses
|
|
std::for_each(genesis_state.initial_witness_candidates.begin(), genesis_state.initial_witness_candidates.end(),
|
|
[&](const genesis_state_type::initial_witness_type& witness) {
|
|
witness_create_operation op;
|
|
op.initial_secret = secret_hash_type();
|
|
op.witness_account = get_account_id(witness.owner_name);
|
|
op.block_signing_key = witness.block_signing_key;
|
|
apply_operation(genesis_eval_state, op);
|
|
});
|
|
|
|
// Create initial committee members
|
|
std::for_each(genesis_state.initial_committee_candidates.begin(), genesis_state.initial_committee_candidates.end(),
|
|
[&](const genesis_state_type::initial_committee_member_type& member) {
|
|
committee_member_create_operation op;
|
|
op.committee_member_account = get_account_id(member.owner_name);
|
|
apply_operation(genesis_eval_state, op);
|
|
});
|
|
|
|
// Create initial workers
|
|
std::for_each(genesis_state.initial_worker_candidates.begin(), genesis_state.initial_worker_candidates.end(),
|
|
[&](const genesis_state_type::initial_worker_type& worker)
|
|
{
|
|
worker_create_operation op;
|
|
op.owner = get_account_id(worker.owner_name);
|
|
op.work_begin_date = genesis_state.initial_timestamp;
|
|
op.work_end_date = time_point_sec::maximum();
|
|
op.daily_pay = worker.daily_pay;
|
|
op.name = "Genesis-Worker-" + worker.owner_name;
|
|
op.initializer = vesting_balance_worker_initializer{uint16_t(0)};
|
|
|
|
apply_operation(genesis_eval_state, std::move(op));
|
|
});
|
|
|
|
// Set active witnesses
|
|
modify(get_global_properties(), [&genesis_state](global_property_object& p) {
|
|
for( uint32_t i = 1; i <= genesis_state.initial_active_witnesses; ++i )
|
|
{
|
|
p.active_witnesses.insert(witness_id_type(i));
|
|
}
|
|
});
|
|
|
|
// Initialize witness schedule
|
|
_p_witness_schedule_obj = & create<witness_schedule_object>([this](witness_schedule_object& _wso)
|
|
{
|
|
// for scheduled
|
|
memset(_wso.rng_seed.begin(), 0, _wso.rng_seed.size());
|
|
|
|
witness_scheduler_rng rng(_wso.rng_seed.begin(), GRAPHENE_NEAR_SCHEDULE_CTR_IV);
|
|
|
|
auto init_witnesses = get_global_properties().active_witnesses;
|
|
|
|
_wso.scheduler = witness_scheduler();
|
|
_wso.scheduler._min_token_count = std::max(int(init_witnesses.size()) / 2, 1);
|
|
_wso.scheduler.update(init_witnesses);
|
|
|
|
for( size_t i=0; i<init_witnesses.size(); ++i )
|
|
_wso.scheduler.produce_schedule(rng);
|
|
|
|
_wso.last_scheduling_block = 0;
|
|
|
|
_wso.recent_slots_filled = fc::uint128::max_value();
|
|
|
|
// for shuffled
|
|
for( const witness_id_type& wid : get_global_properties().active_witnesses )
|
|
_wso.current_shuffled_witnesses.push_back( wid );
|
|
});
|
|
FC_ASSERT( _p_witness_schedule_obj->id == witness_schedule_id_type() );
|
|
|
|
// Initialize witness schedule
|
|
#ifndef NDEBUG
|
|
const son_schedule_object& sso =
|
|
#endif
|
|
create<son_schedule_object>([&](son_schedule_object& _sso)
|
|
{
|
|
// for scheduled
|
|
memset(_sso.rng_seed.begin(), 0, _sso.rng_seed.size());
|
|
|
|
witness_scheduler_rng rng(_sso.rng_seed.begin(), GRAPHENE_NEAR_SCHEDULE_CTR_IV);
|
|
|
|
auto init_witnesses = get_global_properties().active_witnesses;
|
|
|
|
_sso.scheduler = son_scheduler();
|
|
_sso.scheduler._min_token_count = std::max(int(init_witnesses.size()) / 2, 1);
|
|
|
|
|
|
_sso.last_scheduling_block = 0;
|
|
|
|
_sso.recent_slots_filled = fc::uint128::max_value();
|
|
});
|
|
assert( sso.id == son_schedule_id_type() );
|
|
|
|
// Enable fees
|
|
modify(get_global_properties(), [&genesis_state](global_property_object& p) {
|
|
p.parameters.current_fees = genesis_state.initial_parameters.current_fees;
|
|
});
|
|
|
|
|
|
// Create FBA counters
|
|
create<fba_accumulator_object>([&]( fba_accumulator_object& acc )
|
|
{
|
|
FC_ASSERT( acc.id == fba_accumulator_id_type( fba_accumulator_id_transfer_to_blind ) );
|
|
acc.accumulated_fba_fees = 0;
|
|
#ifdef GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET
|
|
acc.designated_asset = GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET;
|
|
#endif
|
|
});
|
|
|
|
create<fba_accumulator_object>([&]( fba_accumulator_object& acc )
|
|
{
|
|
FC_ASSERT( acc.id == fba_accumulator_id_type( fba_accumulator_id_blind_transfer ) );
|
|
acc.accumulated_fba_fees = 0;
|
|
#ifdef GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET
|
|
acc.designated_asset = GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET;
|
|
#endif
|
|
});
|
|
|
|
create<fba_accumulator_object>([&]( fba_accumulator_object& acc )
|
|
{
|
|
FC_ASSERT( acc.id == fba_accumulator_id_type( fba_accumulator_id_transfer_from_blind ) );
|
|
acc.accumulated_fba_fees = 0;
|
|
#ifdef GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET
|
|
acc.designated_asset = GRAPHENE_FBA_STEALTH_DESIGNATED_ASSET;
|
|
#endif
|
|
});
|
|
|
|
FC_ASSERT( get_index<fba_accumulator_object>().get_next_id() == fba_accumulator_id_type( fba_accumulator_id_count ) );
|
|
|
|
debug_dump();
|
|
|
|
_undo_db.enable();
|
|
} FC_CAPTURE_AND_RETHROW() }
|
|
|
|
} }
|