/* * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include "../common/genesis_file_util.hpp" #define BOOST_TEST_MODULE Test Application #include using namespace graphene; BOOST_AUTO_TEST_CASE( two_node_network ) { using namespace graphene::chain; using namespace graphene::app; try { BOOST_TEST_MESSAGE( "Creating temporary files" ); fc::temp_directory app_dir( graphene::utilities::temp_directory_path() ); fc::temp_directory app2_dir( graphene::utilities::temp_directory_path() ); fc::temp_file genesis_json; BOOST_TEST_MESSAGE( "Creating and initializing app1" ); graphene::app::application app1; app1.register_plugin(); app1.register_plugin(); app1.register_plugin(); app1.register_plugin(); app1.register_plugin(); app1.register_plugin(); boost::program_options::variables_map cfg; cfg.emplace("p2p-endpoint", boost::program_options::variable_value(string("127.0.0.1:0"), false)); app1.initialize(app_dir.path(), cfg); cfg.emplace("genesis-json", boost::program_options::variable_value(create_genesis_file(app_dir), false)); BOOST_TEST_MESSAGE( "Starting app1 and waiting 1500 ms" ); app1.startup(); fc::usleep(fc::milliseconds(500)); string endpoint1 = app1.p2p_node()->get_actual_listening_endpoint(); BOOST_TEST_MESSAGE( "Creating and initializing app2" ); auto cfg2 = cfg; graphene::app::application app2; app2.register_plugin(); app2.register_plugin(); app2.register_plugin(); app2.register_plugin(); app2.register_plugin(); app2.register_plugin(); cfg2.erase("p2p-endpoint"); cfg2.emplace("p2p-endpoint", boost::program_options::variable_value(string("127.0.0.1:0"), false)); cfg2.emplace("seed-node", boost::program_options::variable_value(vector{endpoint1}, false)); app2.initialize(app2_dir.path(), cfg2); cfg2.emplace("genesis-json", boost::program_options::variable_value(create_genesis_file(app2_dir), false)); BOOST_TEST_MESSAGE( "Starting app2 and waiting 1500 ms" ); app2.startup(); int counter = 0; while(!app2.p2p_node()->is_connected()) { fc::usleep(fc::milliseconds(500)); if(counter++ >= 100) break; } BOOST_REQUIRE_EQUAL(app1.p2p_node()->get_connection_count(), 1); BOOST_CHECK_EQUAL(std::string(app1.p2p_node()->get_connected_peers().front().host.get_address()), "127.0.0.1"); BOOST_TEST_MESSAGE( "app1 and app2 successfully connected" ); std::shared_ptr db1 = app1.chain_database(); std::shared_ptr db2 = app2.chain_database(); BOOST_CHECK_EQUAL( db1->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 0 ); BOOST_CHECK_EQUAL( db2->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 0 ); BOOST_TEST_MESSAGE( "Creating transfer tx" ); graphene::chain::signed_transaction trx; { account_id_type nathan_id = db2->get_index_type().indices().get().find( "nathan" )->id; fc::ecc::private_key nathan_key = fc::ecc::private_key::regenerate(fc::sha256::hash(string("nathan"))); balance_claim_operation claim_op; balance_id_type bid = balance_id_type(); claim_op.deposit_to_account = nathan_id; claim_op.balance_to_claim = bid; claim_op.balance_owner_key = nathan_key.get_public_key(); claim_op.total_claimed = bid(*db1).balance; trx.operations.push_back( claim_op ); db1->current_fee_schedule().set_fee( trx.operations.back() ); transfer_operation xfer_op; xfer_op.from = nathan_id; xfer_op.to = GRAPHENE_NULL_ACCOUNT; xfer_op.amount = asset( 1000000 ); trx.operations.push_back( xfer_op ); db1->current_fee_schedule().set_fee( trx.operations.back() ); trx.set_expiration( db1->get_slot_time( 10 ) ); trx.sign( nathan_key, db1->get_chain_id() ); trx.validate(); } BOOST_TEST_MESSAGE( "Pushing tx locally on db1" ); processed_transaction ptrx = db1->push_transaction(trx); BOOST_CHECK_EQUAL( db1->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 1000000 ); BOOST_CHECK_EQUAL( db2->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 0 ); BOOST_TEST_MESSAGE( "Broadcasting tx" ); app1.p2p_node()->broadcast(graphene::net::trx_message(trx)); fc::usleep(fc::milliseconds(500)); BOOST_CHECK_EQUAL( db1->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 1000000 ); BOOST_CHECK_EQUAL( db2->get_balance( GRAPHENE_NULL_ACCOUNT, asset_id_type() ).amount.value, 1000000 ); BOOST_TEST_MESSAGE( "Generating block on db2" ); fc::ecc::private_key committee_key = fc::ecc::private_key::regenerate(fc::sha256::hash(string("nathan"))); auto block_1 = db2->generate_block( db2->get_slot_time(1), db2->get_scheduled_witness(1), committee_key, database::skip_nothing); BOOST_TEST_MESSAGE( "Broadcasting block" ); app2.p2p_node()->broadcast(graphene::net::block_message( block_1 )); fc::usleep(fc::milliseconds(500)); BOOST_TEST_MESSAGE( "Verifying nodes are still connected" ); BOOST_CHECK_EQUAL(app1.p2p_node()->get_connection_count(), 1); BOOST_CHECK_EQUAL(app1.chain_database()->head_block_num(), 1); BOOST_TEST_MESSAGE( "Checking GRAPHENE_NULL_ACCOUNT has balance" ); } catch( fc::exception& e ) { edump((e.to_detail_string())); throw; } }