peerplays_migrated/tests/tests/affiliate_tests.cpp
2019-09-12 19:13:36 +05:30

732 lines
35 KiB
C++

/*
* Copyright (c) 2018 Peerplays Blockchain Standards Association, 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 <boost/test/unit_test.hpp>
#include "../common/betting_test_markets.hpp"
#include "../common/tournament_helper.hpp"
#include <graphene/chain/affiliate_payout.hpp>
#include <graphene/chain/game_object.hpp>
#include <graphene/chain/hardfork.hpp>
#include <graphene/chain/match_object.hpp>
#include <graphene/chain/tournament_object.hpp>
#include <graphene/affiliate_stats/affiliate_stats_api.hpp>
#include <graphene/affiliate_stats/affiliate_stats_objects.hpp>
using namespace graphene::chain;
using namespace graphene::db;
class affiliate_test_helper
{
public:
affiliate_test_helper( database_fixture& f ) : fixture(f), accounts(&fixture.db.get_index_type<account_index>())
{
fixture.generate_blocks( HARDFORK_999_TIME );
fixture.generate_block();
test::set_expiration( fixture.db, fixture.trx );
fixture.trx.clear();
alice_id = find_account( "alice" );
if( alice_id == account_id_type() )
{
ACTOR(alice);
this->alice_id = alice_id;
}
ann_id = find_account( "ann" );
if( ann_id == account_id_type() )
{
ACTOR(ann);
this->ann_id = ann_id;
}
audrey_id = find_account( "audrey" );
if( audrey_id == account_id_type() )
{
ACTOR(audrey);
this->audrey_id = audrey_id;
}
paula_id = find_account("paula");
if( paula_id == account_id_type() )
{
// Paula: 100% to Alice for Bookie / nothing for RPS
account_create_operation aco = fixture.make_account( "paula", paula_private_key.get_public_key() );
aco.extensions.value.affiliate_distributions = affiliate_reward_distributions();
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[alice_id] = GRAPHENE_100_PERCENT;
fixture.trx.operations.push_back( aco );
processed_transaction op_result = fixture.db.push_transaction( fixture.trx, ~0 );
paula_id = op_result.operation_results[0].get<object_id_type>();
fixture.trx.clear();
fixture.fund( paula_id(fixture.db), asset(20000000) );
}
penny_id = find_account("penny");
if( penny_id == account_id_type() )
{
// Penny: For Bookie 60% to Alice + 40% to Ann / For RPS 20% to Alice, 30% to Ann, 50% to Audrey
account_create_operation aco = fixture.make_account( "penny", penny_private_key.get_public_key() );
aco.extensions.value.affiliate_distributions = affiliate_reward_distributions();
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[alice_id] = GRAPHENE_100_PERCENT * 3 / 5;
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[ann_id] = GRAPHENE_100_PERCENT
- aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[alice_id];
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[alice_id] = GRAPHENE_100_PERCENT / 5;
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[audrey_id] = GRAPHENE_100_PERCENT / 2;
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[ann_id] = GRAPHENE_100_PERCENT
- aco.extensions.value.affiliate_distributions->_dists[rps]._dist[alice_id]
- aco.extensions.value.affiliate_distributions->_dists[rps]._dist[audrey_id];
fixture.trx.operations.push_back( aco );
processed_transaction op_result = fixture.db.push_transaction( fixture.trx, ~0 );
penny_id = op_result.operation_results[0].get<object_id_type>();
fixture.trx.clear();
fixture.fund( penny_id(fixture.db), asset(30000000) );
}
petra_id = find_account("petra");
if( petra_id == account_id_type() )
{
// Petra: nothing for Bookie / For RPS 10% to Ann, 90% to Audrey
account_create_operation aco = fixture.make_account( "petra", petra_private_key.get_public_key() );
aco.extensions.value.affiliate_distributions = affiliate_reward_distributions();
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[ann_id] = GRAPHENE_100_PERCENT / 10;
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[audrey_id] = GRAPHENE_100_PERCENT
- aco.extensions.value.affiliate_distributions->_dists[rps]._dist[ann_id];
fixture.trx.operations.push_back( aco );
processed_transaction op_result = fixture.db.push_transaction( fixture.trx, ~0 );
petra_id = op_result.operation_results[0].get<object_id_type>();
fixture.trx.clear();
fixture.fund( petra_id(fixture.db), asset(40000000) );
}
update_balances();
}
void update_balances()
{
alice_ppy = fixture.get_balance( alice_id, asset_id_type() );
ann_ppy = fixture.get_balance( ann_id, asset_id_type() );
audrey_ppy = fixture.get_balance( audrey_id, asset_id_type() );
paula_ppy = fixture.get_balance( paula_id, asset_id_type() );
penny_ppy = fixture.get_balance( penny_id, asset_id_type() );
petra_ppy = fixture.get_balance( petra_id, asset_id_type() );
}
database_fixture& fixture;
account_id_type alice_id;
account_id_type ann_id;
account_id_type audrey_id;
account_id_type paula_id;
account_id_type penny_id;
account_id_type petra_id;
int64_t alice_ppy;
int64_t ann_ppy;
int64_t audrey_ppy;
int64_t paula_ppy;
int64_t penny_ppy;
int64_t petra_ppy;
private:
const account_index* accounts;
account_id_type find_account( const string& name )
{
auto itr = accounts->indices().get<by_name>().find( name );
if( itr == accounts->indices().get<by_name>().end() ) return account_id_type();
return itr->id;
}
static fc::ecc::private_key generate_private_key( const string& seed )
{
return database_fixture::generate_private_key( seed );
}
const account_object& create_account( const string& name, const fc::ecc::public_key& key )
{
return fixture.create_account( name, key );
}
public:
const fc::ecc::private_key alice_private_key = generate_private_key( "alice" );
const fc::ecc::private_key ann_private_key = generate_private_key( "ann" );
const fc::ecc::private_key audrey_private_key = generate_private_key( "audrey" );
const fc::ecc::private_key paula_private_key = generate_private_key( "paula" );
const fc::ecc::private_key penny_private_key = generate_private_key( "penny" );
const fc::ecc::private_key petra_private_key = generate_private_key( "petra" );
};
BOOST_FIXTURE_TEST_SUITE( affiliate_tests, database_fixture )
BOOST_AUTO_TEST_CASE( account_test )
{
ACTORS( (alice)(ann)(audrey) );
fund( alice );
const fc::ecc::private_key paula_private_key = generate_private_key( "paula" );
account_create_operation aco = make_account( "paula", paula_private_key.get_public_key() );
aco.extensions.value.affiliate_distributions = affiliate_reward_distributions();
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[alice_id] = GRAPHENE_100_PERCENT;
test::set_expiration( db, trx );
trx.clear();
trx.operations.push_back( aco );
// not allowed before hardfork
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
proposal_create_operation pco;
pco.fee_paying_account = alice_id;
aco.registrar = ann_id;
pco.proposed_ops.emplace_back( aco );
aco.registrar = account_id_type();
pco.expiration_time = db.head_block_time() + fc::days(1);
trx.clear();
trx.operations.push_back( pco );
// not allowed before hardfork
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
generate_blocks( HARDFORK_999_TIME );
generate_block();
// Proposal is now allowed
pco.expiration_time = db.head_block_time() + fc::days(1);
trx.clear();
trx.operations.push_back( pco );
test::set_expiration( db, trx );
db.push_transaction(trx, ~0);
// Must contain at least one app-tag
aco.extensions.value.affiliate_distributions->_dists.clear();
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// Distribution for app-tag must be non-empty
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist.clear();
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// If more than one app-tag given, neither can be empty
aco.extensions.value.affiliate_distributions->_dists[rps]._dist[alice_id] = GRAPHENE_100_PERCENT;
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist.clear();
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// Sum of percentage must be 100%
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[ann_id] = 1;
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// Individual percentages cannot exceed 100%
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[ann_id] = -1;
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[audrey_id] = 1 + GRAPHENE_100_PERCENT;
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// Sum of percentage must be 100%
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[ann_id] = GRAPHENE_100_PERCENT - 10;
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[audrey_id] = 9;
trx.clear();
trx.operations.push_back( aco );
GRAPHENE_REQUIRE_THROW( db.push_transaction(trx, ~0), fc::assert_exception );
// Ok
aco.extensions.value.affiliate_distributions->_dists[bookie]._dist[audrey_id] = 10;
trx.clear();
trx.operations.push_back( aco );
db.push_transaction(trx, ~0);
const auto& paula = get_account( aco.name );
BOOST_CHECK( paula.affiliate_distributions.valid() );
BOOST_CHECK_EQUAL( 2, paula.affiliate_distributions->_dists.size() );
const auto& app_rps = paula.affiliate_distributions->_dists.find( rps );
BOOST_CHECK( app_rps != paula.affiliate_distributions->_dists.end() );
BOOST_CHECK_EQUAL( 1, app_rps->second._dist.size() );
const auto& share_alice = app_rps->second._dist.find( alice_id );
BOOST_CHECK( share_alice != app_rps->second._dist.end() );
BOOST_CHECK_EQUAL( GRAPHENE_100_PERCENT, share_alice->second );
const auto& app_bookie = paula.affiliate_distributions->_dists.find( bookie );
BOOST_CHECK( app_bookie != paula.affiliate_distributions->_dists.end() );
BOOST_CHECK_EQUAL( 2, app_bookie->second._dist.size() );
const auto& share_ann = app_bookie->second._dist.find( ann_id );
BOOST_CHECK( share_ann != app_bookie->second._dist.end() );
BOOST_CHECK_EQUAL( GRAPHENE_100_PERCENT - 10, share_ann->second );
const auto& share_audrey = app_bookie->second._dist.find( audrey_id );
BOOST_CHECK( share_audrey != app_bookie->second._dist.end() );
BOOST_CHECK_EQUAL( 10, share_audrey->second );
}
BOOST_AUTO_TEST_CASE( affiliate_payout_helper_test )
{
ACTORS( (irene) );
const asset_id_type btc_id = create_user_issued_asset( "BTC", irene, 0 ).id;
issue_uia( irene, asset( 100000, btc_id ) );
affiliate_test_helper ath( *this );
int64_t alice_btc = 0;
int64_t ann_btc = 0;
int64_t audrey_btc = 0;
{
const tournament_object& game = db.create<tournament_object>( []( tournament_object& t ) {
t.options.game_options = rock_paper_scissors_game_options();
t.options.buy_in = asset( 10 );
});
affiliate_payout_helper helper = affiliate_payout_helper( db, game );
// Alice has no distribution set
BOOST_CHECK_EQUAL( 0, helper.payout( ath.alice_id, 1000 ).value );
// Paula has nothing for Bookie
BOOST_CHECK_EQUAL( 0, helper.payout( ath.paula_id, 1000 ).value );
// 20% of 4 gets rounded down to 0
BOOST_CHECK_EQUAL( 0, helper.payout( ath.penny_id, 4 ).value );
// 20% of 5 = 1 is paid to Audrey
BOOST_CHECK_EQUAL( 1, helper.payout( ath.penny_id, 5 ).value );
ath.audrey_ppy++;
// 20% of 50 = 10: 2 to Alice, 3 to Ann, 5 to Audrey
BOOST_CHECK_EQUAL( 10, helper.payout( ath.penny_id, 50 ).value );
ath.alice_ppy += 2;
ath.ann_ppy += 3;
ath.audrey_ppy += 5;
// 20% of 59 = 11: 1 to Ann, 10 to Audrey
BOOST_CHECK_EQUAL( 11, helper.payout( ath.petra_id, 59 ).value );
ath.audrey_ppy += 10;
ath.ann_ppy += 1;
helper.commit();
BOOST_CHECK_EQUAL( ath.alice_ppy, get_balance( ath.alice_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.ann_ppy, get_balance( ath.ann_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.audrey_ppy, get_balance( ath.audrey_id, asset_id_type() ) );
}
{
const tournament_object& game = db.create<tournament_object>( [btc_id]( tournament_object& t ) {
t.options.game_options = rock_paper_scissors_game_options();
t.options.buy_in = asset( 10, btc_id );
});
affiliate_payout_helper helper = affiliate_payout_helper( db, game );
// 20% of 60 = 12: 2 to Alice, 3 to Ann, 7 to Audrey
BOOST_CHECK_EQUAL( 12, helper.payout( ath.penny_id, 60 ).value );
alice_btc += 2;
ann_btc += 3;
audrey_btc += 7;
helper.commit();
BOOST_CHECK_EQUAL( alice_btc, get_balance( ath.alice_id, btc_id ) );
BOOST_CHECK_EQUAL( ann_btc, get_balance( ath.ann_id, btc_id ) );
BOOST_CHECK_EQUAL( audrey_btc, get_balance( ath.audrey_id, btc_id ) );
}
{
const betting_market_group_object& game = db.create<betting_market_group_object>( []( betting_market_group_object& b ) {
b.asset_id = asset_id_type();
} );
affiliate_payout_helper helper = affiliate_payout_helper( db, game );
// Alice has no distribution set
BOOST_CHECK_EQUAL( 0, helper.payout( ath.alice_id, 1000 ).value );
// Petra has nothing for Bookie
BOOST_CHECK_EQUAL( 0, helper.payout( ath.petra_id, 1000 ).value );
// 20% of 4 gets rounded down to 0
BOOST_CHECK_EQUAL( 0, helper.payout( ath.penny_id, 4 ).value );
// 20% of 5 = 1 is paid to Ann
BOOST_CHECK_EQUAL( 1, helper.payout( ath.penny_id, 5 ).value );
ath.ann_ppy++;
// 20% of 40 = 8: 8 to Alice
BOOST_CHECK_EQUAL( 8, helper.payout( ath.paula_id, 40 ).value );
ath.alice_ppy += 8;
// intermediate commit should clear internal accumulator
helper.commit();
// 20% of 59 = 11: 6 to Alice, 5 to Ann
BOOST_CHECK_EQUAL( 11, helper.payout( ath.penny_id, 59 ).value );
ath.alice_ppy += 6;
ath.ann_ppy += 5;
helper.commit();
BOOST_CHECK_EQUAL( ath.alice_ppy, get_balance( ath.alice_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.ann_ppy, get_balance( ath.ann_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.audrey_ppy, get_balance( ath.audrey_id, asset_id_type() ) );
}
{
const betting_market_group_object& game = db.create<betting_market_group_object>( [btc_id]( betting_market_group_object& b ) {
b.asset_id = btc_id;
} );
affiliate_payout_helper helper = affiliate_payout_helper( db, game );
// 20% of 60 = 12: 7 to Alice, 5 to Ann
BOOST_CHECK_EQUAL( 12, helper.payout( ath.penny_id, 60 ).value );
alice_btc += 7;
ann_btc += 5;
helper.commit();
BOOST_CHECK_EQUAL( alice_btc, get_balance( ath.alice_id, btc_id ) );
BOOST_CHECK_EQUAL( ann_btc, get_balance( ath.ann_id, btc_id ) );
BOOST_CHECK_EQUAL( audrey_btc, get_balance( ath.audrey_id, btc_id ) );
}
{
// Fix total supply
auto& index = db.get_index_type< primary_index< account_balance_index > >().get_secondary_index<balances_by_account_index>();
auto abo = index.get_account_balance( account_id_type(), asset_id_type() );
BOOST_CHECK( abo != nullptr );
db.modify( *abo, [&ath]( account_balance_object& bal ) {
bal.balance -= ath.alice_ppy + ath.ann_ppy + ath.audrey_ppy;
});
abo = index.get_account_balance( irene_id, btc_id );
BOOST_CHECK( abo != nullptr );
db.modify( *abo, [alice_btc,ann_btc,audrey_btc]( account_balance_object& bal ) {
bal.balance -= alice_btc + ann_btc + audrey_btc;
});
}
}
BOOST_AUTO_TEST_CASE( rps_tournament_payout_test )
{ try {
ACTORS( (martha) );
affiliate_test_helper ath( *this );
fund( martha_id(db), asset(1000000000) );
upgrade_to_lifetime_member( martha_id(db) );
tournaments_helper helper(*this);
account_id_type dividend_id = *helper.get_asset_dividend_account();
int64_t div_ppy = get_balance( dividend_id, asset_id_type() );
const asset buy_in = asset(12000);
tournament_id_type tournament_id = helper.create_tournament( martha_id, martha_private_key, buy_in, 3, 3, 1, 1 );
BOOST_REQUIRE(tournament_id == tournament_id_type());
helper.join_tournament( tournament_id, ath.paula_id, ath.paula_id, ath.paula_private_key, buy_in);
helper.join_tournament( tournament_id, ath.penny_id, ath.penny_id, ath.penny_private_key, buy_in);
helper.join_tournament( tournament_id, ath.petra_id, ath.petra_id, ath.petra_private_key, buy_in);
generate_block();
const tournament_object& tournament = db.get<tournament_object>( tournament_id );
BOOST_CHECK_EQUAL( ath.paula_ppy - 12000, get_balance( ath.paula_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.penny_ppy - 12000, get_balance( ath.penny_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.petra_ppy - 12000, get_balance( ath.petra_id, asset_id_type() ) );
const tournament_details_object& tournament_details = tournament.tournament_details_id(db);
BOOST_CHECK_EQUAL( 3, tournament_details.matches.size() );
{ // 3 players, match 1 is a bye for penny
const match_object& match = tournament_details.matches[0](db);
BOOST_CHECK_EQUAL( int64_t(match_state::match_complete), int64_t(match.get_state()) );
BOOST_CHECK_EQUAL( 1, match.players.size() );
BOOST_CHECK_EQUAL( object_id_type(ath.penny_id).instance(), object_id_type(match.players[0]).instance() );
}
{ // match 2 is paula vs. petra: paula wins
const match_object& match = tournament_details.matches[1](db);
BOOST_CHECK_EQUAL( int64_t(match_state::match_in_progress), int64_t(match.get_state()) );
BOOST_CHECK_EQUAL( 2, match.players.size() );
BOOST_CHECK_EQUAL( object_id_type(ath.paula_id).instance(), object_id_type(match.players[0]).instance() );
BOOST_CHECK_EQUAL( object_id_type(ath.petra_id).instance(), object_id_type(match.players[1]).instance() );
BOOST_CHECK_EQUAL( 1, match.games.size() );
const game_object& game = match.games[0](db);
BOOST_CHECK_EQUAL( int64_t(game_state::expecting_commit_moves), int64_t(game.get_state()) );
helper.rps_throw( game.id, ath.paula_id, rock_paper_scissors_gesture::paper, ath.paula_private_key );
helper.rps_throw( game.id, ath.petra_id, rock_paper_scissors_gesture::rock, ath.petra_private_key );
BOOST_CHECK_EQUAL( int64_t(game_state::expecting_reveal_moves), int64_t(game.get_state()) );
helper.rps_reveal( game.id, ath.paula_id, rock_paper_scissors_gesture::paper, ath.paula_private_key );
helper.rps_reveal( game.id, ath.petra_id, rock_paper_scissors_gesture::rock, ath.petra_private_key );
BOOST_CHECK_EQUAL( int64_t(game_state::game_complete), int64_t(game.get_state()) );
BOOST_CHECK_EQUAL( 1, match.games.size() );
BOOST_CHECK_EQUAL( int64_t(match_state::match_complete), int64_t(match.get_state()) );
}
{ // match 3 is penny vs. paula: penny wins
const match_object& match = tournament_details.matches[2](db);
BOOST_CHECK_EQUAL( int64_t(match_state::match_in_progress), int64_t(match.get_state()) );
BOOST_CHECK_EQUAL( 2, match.players.size() );
BOOST_CHECK_EQUAL( object_id_type(ath.penny_id).instance(), object_id_type(match.players[0]).instance() );
BOOST_CHECK_EQUAL( object_id_type(ath.paula_id).instance(), object_id_type(match.players[1]).instance() );
BOOST_CHECK_EQUAL( 1, match.games.size() );
const game_object& game = match.games[0](db);
BOOST_CHECK_EQUAL( int64_t(game_state::expecting_commit_moves), int64_t(game.get_state()) );
helper.rps_throw( game.id, ath.paula_id, rock_paper_scissors_gesture::paper, ath.paula_private_key );
helper.rps_throw( game.id, ath.penny_id, rock_paper_scissors_gesture::scissors, ath.penny_private_key );
BOOST_CHECK_EQUAL( int64_t(game_state::expecting_reveal_moves), int64_t(game.get_state()) );
helper.rps_reveal( game.id, ath.paula_id, rock_paper_scissors_gesture::paper, ath.paula_private_key );
helper.rps_reveal( game.id, ath.penny_id, rock_paper_scissors_gesture::scissors, ath.penny_private_key );
BOOST_CHECK_EQUAL( int64_t(game_state::game_complete), int64_t(game.get_state()) );
BOOST_CHECK_EQUAL( int64_t(match_state::match_complete), int64_t(match.get_state()) );
}
BOOST_CHECK_EQUAL( 3*GRAPHENE_1_PERCENT, db.get_global_properties().parameters.rake_fee_percentage );
// Penny wins net 3*buy_in minus rake = 36000 - 1080 = 34920
BOOST_CHECK_EQUAL( ath.paula_ppy - 12000, get_balance( ath.paula_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.penny_ppy - 12000 + 34920, get_balance( ath.penny_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.petra_ppy - 12000, get_balance( ath.petra_id, asset_id_type() ) );
// Dividend account receives 80% of rake = 864
BOOST_CHECK_EQUAL( div_ppy + 864, get_balance( dividend_id, asset_id_type() ) );
// 20% of rake = 216 is paid to Penny's affiliates: 43 to Alice, 64 to Ann, 109 to Audrey
BOOST_CHECK_EQUAL( ath.alice_ppy + 43, get_balance( ath.alice_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.ann_ppy + 64, get_balance( ath.ann_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.audrey_ppy + 109, get_balance( ath.audrey_id, asset_id_type() ) );
} FC_LOG_AND_RETHROW() }
BOOST_AUTO_TEST_CASE( bookie_payout_test )
{ try {
ACTORS( (irene) );
const asset_id_type btc_id = create_user_issued_asset( "BTC", irene, 0 ).id;
affiliate_test_helper ath( *this );
CREATE_ICE_HOCKEY_BETTING_MARKET(false, 0);
// place bets at 10:1
place_bet(ath.paula_id, capitals_win_market.id, bet_type::back, asset(10000, asset_id_type()), 11 * GRAPHENE_BETTING_ODDS_PRECISION);
place_bet(ath.penny_id, capitals_win_market.id, bet_type::lay, asset(100000, asset_id_type()), 11 * GRAPHENE_BETTING_ODDS_PRECISION);
// reverse positions at 1:1
place_bet(ath.paula_id, capitals_win_market.id, bet_type::lay, asset(110000, asset_id_type()), 2 * GRAPHENE_BETTING_ODDS_PRECISION);
place_bet(ath.penny_id, capitals_win_market.id, bet_type::back, asset(110000, asset_id_type()), 2 * GRAPHENE_BETTING_ODDS_PRECISION);
update_betting_market_group(moneyline_betting_markets.id, graphene::chain::keywords::_status = betting_market_group_status::closed);
resolve_betting_market_group(moneyline_betting_markets.id,
{{capitals_win_market.id, betting_market_resolution_type::win},
{blackhawks_win_market.id, betting_market_resolution_type::not_win}});
generate_block();
uint16_t rake_fee_percentage = db.get_global_properties().parameters.betting_rake_fee_percentage();
BOOST_CHECK_EQUAL( 3 * GRAPHENE_1_PERCENT, rake_fee_percentage );
uint32_t rake_value;
// rake_value = (-10000 + 110000 - 110000) * rake_fee_percentage / GRAPHENE_1_PERCENT / 100;
// paula starts with 20000000, pays 10000 (bet), wins 110000, then pays 110000 (bet), wins 0
BOOST_CHECK_EQUAL( get_balance( ath.paula_id, asset_id_type() ), ath.paula_ppy - 10000 + 110000 - 110000 + 0 );
// no wins -> no affiliate payouts
rake_value = (-100000 - 110000 + 220000) * rake_fee_percentage / GRAPHENE_1_PERCENT / 100;
// penny starts with 30000000, pays 100000 (bet), wins 0, then pays 110000 (bet), wins 220000
BOOST_CHECK_EQUAL( get_balance( ath.penny_id, asset_id_type() ), ath.penny_ppy - 100000 + 0 - 110000 + 220000 - rake_value );
// penny wins 10000 net, rake is 3% of that (=300)
// Affiliate pay is 20% of rake (=60). Of this, 60%=36 go to Alice, 40%=24 go to Ann
BOOST_CHECK_EQUAL( ath.alice_ppy + 36, get_balance( ath.alice_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.ann_ppy + 24, get_balance( ath.ann_id, asset_id_type() ) );
BOOST_CHECK_EQUAL( ath.audrey_ppy + 0, get_balance( ath.audrey_id, asset_id_type() ) );
{
test::set_expiration( db, trx );
issue_uia( ath.paula_id, asset( 1000000, btc_id ) );
issue_uia( ath.petra_id, asset( 1000000, btc_id ) );
create_event({{"en", "Washington Capitals/Chicago Blackhawks"}, {"zh_Hans", "華盛頓首都隊/芝加哥黑鷹"}, {"ja", "ワシントン・キャピタルズ/シカゴ・ブラックホークス"}}, {{"en", "2016-17"}}, nhl.id); \
generate_blocks(1); \
const event_object& capitals_vs_blackhawks2 = *db.get_index_type<event_object_index>().indices().get<by_id>().rbegin(); \
create_betting_market_group({{"en", "Moneyline"}}, capitals_vs_blackhawks2.id, betting_market_rules.id, btc_id, false, 0);
generate_blocks(1);
const betting_market_group_object& moneyline_betting_markets2 = *db.get_index_type<betting_market_group_object_index>().indices().get<by_id>().rbegin();
create_betting_market(moneyline_betting_markets2.id, {{"en", "Washington Capitals win"}});
generate_blocks(1);
const betting_market_object& capitals_win_market2 = *db.get_index_type<betting_market_object_index>().indices().get<by_id>().rbegin();
create_betting_market(moneyline_betting_markets2.id, {{"en", "Chicago Blackhawks win"}});
generate_blocks(1);
const betting_market_object& blackhawks_win_market2 = *db.get_index_type<betting_market_object_index>().indices().get<by_id>().rbegin();
// place bets at 10:1
place_bet(ath.paula_id, capitals_win_market2.id, bet_type::back, asset(10000, btc_id), 11 * GRAPHENE_BETTING_ODDS_PRECISION);
place_bet(ath.petra_id, capitals_win_market2.id, bet_type::lay, asset(100000, btc_id), 11 * GRAPHENE_BETTING_ODDS_PRECISION);
// reverse positions at 1:1
place_bet(ath.paula_id, capitals_win_market2.id, bet_type::lay, asset(110000, btc_id), 2 * GRAPHENE_BETTING_ODDS_PRECISION);
place_bet(ath.petra_id, capitals_win_market2.id, bet_type::back, asset(110000, btc_id), 2 * GRAPHENE_BETTING_ODDS_PRECISION);
update_betting_market_group(moneyline_betting_markets2.id, graphene::chain::keywords::_status = betting_market_group_status::closed);
resolve_betting_market_group(moneyline_betting_markets2.id,
{{capitals_win_market2.id, betting_market_resolution_type::not_win},
{blackhawks_win_market2.id, betting_market_resolution_type::win}});
generate_block();
rake_value = (-10000 + 0 - 110000 + 220000) * rake_fee_percentage / GRAPHENE_1_PERCENT / 100;
// paula starts with 1000000, pays 10000 (bet), wins 0, then pays 110000 (bet), wins 220000
BOOST_CHECK_EQUAL( get_balance( ath.paula_id, btc_id ), 1000000 - 10000 + 0 - 110000 + 220000 - rake_value );
// net win 100000, 3% rake = 3000, 20% of that is 600, 100% of that goes to Alice
BOOST_CHECK_EQUAL( 600, get_balance( ath.alice_id, btc_id ) );
BOOST_CHECK_EQUAL( 0, get_balance( ath.ann_id, btc_id ) );
BOOST_CHECK_EQUAL( 0, get_balance( ath.audrey_id, btc_id ) );
// rake_value = (-100000 + 110000 - 110000) * rake_fee_percentage / GRAPHENE_1_PERCENT / 100;
rake_value = 0;
// petra starts with 1000000, pays 100000 (bet), wins 110000, then pays 110000 (bet), wins 0
BOOST_CHECK_EQUAL( get_balance( ath.petra_id, btc_id ), 1000000 - 100000 + 110000 - 110000 + 0 - rake_value );
// petra wins nothing -> no payout
}
} FC_LOG_AND_RETHROW() }
BOOST_AUTO_TEST_CASE( statistics_test )
{ try {
INVOKE(rps_tournament_payout_test);
affiliate_test_helper ath( *this );
transfer( ath.audrey_id, ath.alice_id, asset( 10 ), asset(0) );
transfer( ath.audrey_id, ath.ann_id, asset( 10 ), asset(0) );
INVOKE(bookie_payout_test);
const asset_id_type btc_id = get_asset( "BTC" ).id;
transfer( ath.alice_id, ath.ann_id, asset( 100, btc_id ), asset(0) );
transfer( ath.alice_id, ath.audrey_id, asset( 100, btc_id ), asset(0) );
generate_block();
{
const auto& idx = db.get_index_type<graphene::affiliate_stats::referral_reward_index>().indices().get<graphene::affiliate_stats::by_asset>();
BOOST_CHECK_EQUAL( 2, idx.size() ); // penny 216+60 CORE, paula 600 BTC
}
{
const auto& idx = db.get_index_type<graphene::affiliate_stats::app_reward_index>().indices().get<graphene::affiliate_stats::by_asset>();
BOOST_CHECK_EQUAL( 3, idx.size() ); // rps 216 CORE, bookie 60 CORE + 600 BTC
}
graphene::affiliate_stats::affiliate_stats_api stats( app );
{
std::vector<graphene::affiliate_stats::referral_payment> refs = stats.list_historic_referral_rewards( ath.alice_id, operation_history_id_type() );
BOOST_CHECK_EQUAL( 3, refs.size() );
BOOST_REQUIRE_LE( 1, refs.size() );
BOOST_CHECK_EQUAL( app_tag::rps, refs[0].tag );
BOOST_CHECK_EQUAL( 43, refs[0].payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[0].payout.asset_id.instance.value );
BOOST_REQUIRE_LE( 2, refs.size() );
BOOST_CHECK_EQUAL( app_tag::bookie, refs[1].tag );
BOOST_CHECK_EQUAL( 36, refs[1].payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[1].payout.asset_id.instance.value );
BOOST_REQUIRE_LE( 3, refs.size() );
BOOST_CHECK_EQUAL( app_tag::bookie, refs[2].tag );
BOOST_CHECK_EQUAL( 600, refs[2].payout.amount.value );
BOOST_CHECK_EQUAL( btc_id.instance.value, refs[2].payout.asset_id.instance.value );
BOOST_CHECK_EQUAL( 3, stats.list_historic_referral_rewards( ath.alice_id, refs[0].id ).size() );
BOOST_CHECK_EQUAL( 2, stats.list_historic_referral_rewards( ath.alice_id, refs[1].id ).size() );
BOOST_CHECK_EQUAL( 1, stats.list_historic_referral_rewards( ath.alice_id, refs[2].id ).size() );
refs = stats.list_historic_referral_rewards( ath.ann_id, operation_history_id_type() );
BOOST_CHECK_EQUAL( 2, refs.size() );
BOOST_REQUIRE_LE( 1, refs.size() );
BOOST_CHECK_EQUAL( app_tag::rps, refs[0].tag );
BOOST_CHECK_EQUAL( 64, refs[0].payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[0].payout.asset_id.instance.value );
BOOST_REQUIRE_LE( 2, refs.size() );
BOOST_CHECK_EQUAL( app_tag::bookie, refs[1].tag );
BOOST_CHECK_EQUAL( 24, refs[1].payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[1].payout.asset_id.instance.value );
BOOST_CHECK_EQUAL( 2, stats.list_historic_referral_rewards( ath.ann_id, refs[0].id ).size() );
BOOST_CHECK_EQUAL( 1, stats.list_historic_referral_rewards( ath.ann_id, refs[1].id ).size() );
refs = stats.list_historic_referral_rewards( ath.audrey_id, operation_history_id_type() );
BOOST_CHECK_EQUAL( 1, refs.size() );
BOOST_REQUIRE_LE( 1, refs.size() );
BOOST_CHECK_EQUAL( app_tag::rps, refs[0].tag );
BOOST_CHECK_EQUAL( 109, refs[0].payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[0].payout.asset_id.instance.value );
BOOST_CHECK_EQUAL( 0, stats.list_historic_referral_rewards( ath.alice_id, operation_history_id_type(9990) ).size() );
BOOST_CHECK_EQUAL( 1, stats.list_historic_referral_rewards( ath.alice_id, operation_history_id_type(), 1 ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_historic_referral_rewards( ath.paula_id, operation_history_id_type() ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_historic_referral_rewards( ath.penny_id, operation_history_id_type() ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_historic_referral_rewards( ath.petra_id, operation_history_id_type() ).size() );
}
{
std::vector<graphene::affiliate_stats::top_referred_account> refs = stats.list_top_referred_accounts( asset_id_type() );
BOOST_CHECK_EQUAL( 1, refs.size() );
BOOST_REQUIRE_LE( 1, refs.size() );
BOOST_CHECK_EQUAL( ath.penny_id.instance.value, refs[0].referral.instance.value );
BOOST_CHECK_EQUAL( 276, refs[0].total_payout.amount.value );
BOOST_CHECK_EQUAL( 0, refs[0].total_payout.asset_id.instance.value );
refs = stats.list_top_referred_accounts( btc_id );
BOOST_CHECK_EQUAL( 1, refs.size() );
BOOST_REQUIRE_LE( 1, refs.size() );
BOOST_CHECK_EQUAL( ath.paula_id.instance.value, refs[0].referral.instance.value );
BOOST_CHECK_EQUAL( 600, refs[0].total_payout.amount.value );
BOOST_CHECK_EQUAL( btc_id.instance.value, refs[0].total_payout.asset_id.instance.value );
BOOST_CHECK_EQUAL( 1, stats.list_top_referred_accounts( btc_id, 1 ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_top_referred_accounts( btc_id, 0 ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_top_referred_accounts( asset_id_type(9999) ).size() );
}
{
std::vector<graphene::affiliate_stats::top_app> top = stats.list_top_rewards_per_app( asset_id_type() );
BOOST_CHECK_EQUAL( 2, top.size() );
BOOST_REQUIRE_LE( 1, top.size() );
BOOST_CHECK_EQUAL( app_tag::rps, top[0].app );
BOOST_CHECK_EQUAL( 216, top[0].total_payout.amount.value );
BOOST_CHECK_EQUAL( 0, top[0].total_payout.asset_id.instance.value );
BOOST_REQUIRE_LE( 2, top.size() );
BOOST_CHECK_EQUAL( app_tag::bookie, top[1].app );
BOOST_CHECK_EQUAL( 60, top[1].total_payout.amount.value );
BOOST_CHECK_EQUAL( 0, top[1].total_payout.asset_id.instance.value );
top = stats.list_top_rewards_per_app( btc_id );
BOOST_CHECK_EQUAL( 1, top.size() );
BOOST_REQUIRE_LE( 1, top.size() );
BOOST_CHECK_EQUAL( app_tag::bookie, top[0].app );
BOOST_CHECK_EQUAL( 600, top[0].total_payout.amount.value );
BOOST_CHECK_EQUAL( btc_id.instance.value, top[0].total_payout.asset_id.instance.value );
BOOST_CHECK_EQUAL( 1, stats.list_top_rewards_per_app( asset_id_type(), 1 ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_top_referred_accounts( btc_id, 0 ).size() );
BOOST_CHECK_EQUAL( 0, stats.list_top_rewards_per_app( asset_id_type(9999) ).size() );
}
} FC_LOG_AND_RETHROW() }
BOOST_AUTO_TEST_SUITE_END()