2017-04-04 21:02:19 +00:00
/*
2018-10-09 14:33:31 +00:00
* Copyright ( c ) 2018 Peerplays Blockchain Standards Association , and contributors .
2017-04-04 21:02:19 +00:00
*
* 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/bookie/bookie_plugin.hpp>
2017-08-09 21:17:51 +00:00
# include <graphene/bookie/bookie_objects.hpp>
2017-04-04 21:02:19 +00:00
Merge Plugins and graphene update changes from beatrice TESTNET to master (#317)
* increase delay for node connection
* remove cache from cli get_account
* add cli tests framework
* Adjust newly merged code to new API
* Merged changes from Bitshares PR 1036
* GRPH-76 - Short-cut long sequences of missed blocks
Fixes database::update_global_dynamic_data to speed up counting missed blocks.
(This also fixes a minor issue with counting - the previous algorithm would skip missed blocks for the witness who signed the first block after the gap.)
* Improved resilience of block database against corruption
* Moved reindex logic into database / chain_database, make use of additional blocks in block_database
Fixed tests wrt db.open
* Enable undo + fork database for final blocks in a replay
Dont remove blocks from block db when popping blocks, handle edge case in replay wrt fork_db, adapted unit tests
* Log starting block number of replay
* Prevent unsigned integer underflow
* Fixed lock detection
* Dont leave _data_dir empty if db is locked
* Writing the object_database is now almost atomic
* Improved consistency check for block_log
* Cut back block_log index file if inconsistent
* Fixed undo_database
* Added test case for broken merge on empty undo_db
* exclude second undo_db.enable() call in some cases
* Add missing change
* change bitshares to core in message
* Merge pull request #938 from bitshares/fix-block-storing
Store correct block ID when switching forks
* Fixed integer overflow issue
* Fix for for history ID mismatch ( Bitshares PR #875 )
* Update the FC submodule with the changes for GRPH-4
* Merged Bitshares PR #1462 and compilation fixes
* Support/gitlab (#123)
* Updated gitlab process
* Fix undefined references in cli test
* Updated GitLab CI
* Fix #436 object_database created outside of witness data directory
* supplement more comments on database::_opened variable
* prevent segfault when destructing application obj
* Fixed test failures and compilation issue
* minor performance improvement
* Added comment
* Fix compilation in debug mode
* Fixed duplicate ops returned from get_account_history
* Fixed account_history_pagination test
* Removed unrelated comment
* Update to fixed version of fc
* Skip auth check when pushing self-generated blocks
* Extract public keys before pushing a transaction
* Dereference chain_database shared_ptr
* Updated transaction::signees to mutable
and
* updated get_signature_keys() to return a const reference,
* get_signature_keys() will update signees on first call,
* modified test cases and wallet.cpp accordingly,
* no longer construct a new signed_transaction object before pushing
* Added get_asset_count API
* No longer extract public keys before pushing a trx
and removed unused new added constructor and _get_signature_keys() function from signed_transaction struct
* changes to withdraw_vesting feature(for both cdd and GPOS)
* Comments update
* update to GPOS hardfork ref
* Remove leftover comment from merge
* fix for get_vesting_balance API call
* braces update
* Allow sufficient space for new undo_session
* Throw for deep nesting
* node.cpp: Check the attacker/buggy client before updating items ids
The peer is an attacker or buggy, which means the item_hashes_received is
not correct.
Move the check before updating items ids to save some time in this case.
* Create .gitlab-ci.yml
* Added cli_test to CI
* fixing build errors (#150)
* fixing build errors
vest type correction
* fixing build errors
vest type correction
* fixes
new Dockerfile
* vesting_balance_type correction
vesting_balance_type changed to normal
* gcc5 support to Dockerfile
gcc5 support to Dockerfile
* use random port numbers in app_test (#154)
* Changes to compiple with GCC 7(Ubuntu 18.04)
* proposal fail_reason bug fixed (#157)
* Added Sonarcloud code_quality to CI (#159)
* Added sonarcloud analysis (#158)
* changes to have separate methods and single withdrawl fee for multiple vest objects
* 163-fix, Return only non-zero vesting balances
* Support/gitlab develop (#168)
* Added code_quality to CI
* Update .gitlab-ci.yml
* Point to PBSA/peerplays-fc commit f13d063 (#167)
* [GRPH-3] Additional cli tests (#155)
* Additional cli tests
* Compatible with latest fc changes
* Fixed Spacing issues
* [GRPH-106] Added voting tests (#136)
* Added more voting tests
* Added additional option
* Adjust p2p log level (#180)
* Added submodule sync to peerplays compile process
* merge gpos to develop (#186)
* issue - 154: Don't allow to vote when vesting balance is 0
* changes to withdraw_vesting feature(for both cdd and GPOS)
* Comments update
* update to GPOS hardfork ref
* fix for get_vesting_balance API call
* braces update
* Create .gitlab-ci.yml
* fixing build errors (#150)
* fixing build errors
vest type correction
* fixing build errors
vest type correction
* fixes
new Dockerfile
* vesting_balance_type correction
vesting_balance_type changed to normal
* gcc5 support to Dockerfile
gcc5 support to Dockerfile
* Changes to compiple with GCC 7(Ubuntu 18.04)
* changes to have separate methods and single withdrawl fee for multiple vest objects
* 163-fix, Return only non-zero vesting balances
* Revert "Revert "GPOS protocol""
This reverts commit 67616417b7f0b5d087b9862de0e48b2d8ccc1bca.
* add new line needed to gpos hardfork file
* comment temporally cli_vote_for_2_witnesses until refactor or delete
* fix gpos tests
* fix gitlab-ci conflict
* Fixed few error messages
* error message corrections at other places
* Updated FC repository to peerplays-network/peerplays-fc (#189)
Point to fc commit hash 6096e94 [latest-fc branch]
* Project name update in Doxyfile (#146)
* changes to allow user to vote in each sub-period
* Fixed GPOS vesting factor issue when proxy is set
* Added unit test for proxy voting
* Review changes
* changes to update last voting time
* resolve merge conflict
* unit test changes and also separated GPOS test suite
* delete unused variables
* removed witness check
* eliminate time gap between two consecutive vesting periods
* deleted GPOS specific test suite and updated gpos tests
* updated GPOS hf
* Fixed dividend distribution issue and added test case
* fix flag
* clean newlines gpos_tests
* adapt gpos_tests to changed flag
* Fix to roll in GPOS rules, carry votes from 6th sub-period
* check was already modified
* comments updated
* updated comments to the benefit of reviewer
* Added token symbol name in error messages
* Added token symbol name in error messages (#204)
* case 1: Fixed last voting time issue
* get_account bug fixed
* Fixed flag issue
* Fixed spelling issue
* remove non needed gcc5 changes to dockerfile
* GRPH134- High CPU Issue, websocket changes (#213)
* update submodule branch to refer to the latest commit on latest-fc branch (#214)
* Improve account maintenance performance (#130)
* Improve account maintenance performance
* merge fixes
* Fixed merge issue
* Fixed indentations and extra ';'
* Update CI for syncing gitmodules (#216)
* Added logging for the old update_expired_feeds bug
The old bug is https://github.com/cryptonomex/graphene/issues/615 .
Due to the bug, `update_median_feeds()` and `check_call_orders()`
will be called when a feed is not actually expired, normally this
should not affect consensus since calling them should not change
any data in the state.
However, the logging indicates that `check_call_orders()` did
change some data under certain circumstances, specifically, when
multiple limit order matching issue (#453) occurred at same block.
* https://github.com/bitshares/bitshares-core/issues/453
* Minor performance improvement for price::is_null()
* Use static refs in db_getter for immutable objects
* Minor performance improvement for db_maint
* Minor code updates for asset_evaluator.cpp
* changed an `assert()` to `FC_ASSERT()`
* replaced one `db.get(asset_id_type())` with `db.get_core_asset()`
* capture only required variables for lambda
* Improve update_expired_feeds performance #1093
* Change static refs to member pointers of db class
* Added getter for witness schedule object
* Added getter for core dynamic data object
* Use getters
* Removed unused variable
* Add comments for update_expired_feeds in db_block
* Minor refactory asset_create_evaluator::do_apply()
* Added FC_ASSERT for dynamic data id of core asset
* Added header inclusions in db_management.cpp
* fix global objects usage during replay
* Logging config parsing issue
* added new files
* compilation fix
* Simplified code in database::pay_workers()
* issue with withdrawl
* Added unit test for empty account history
* set extensions default values
* Update GPOS hardfork date and don't allow GPOS features before hardfork time
* refer to latest commit of latest-fc branch (#224)
* account name or id support in all database api
* asset id or name support in all asset APIs
* Fixed compilation issues
* Fixed alignment issues
* Externalized some API templates
* Externalize serialization of blocks, tx, ops
* Externalized db objects
* Externalized genesis serialization
* Externalized serialization in protocol library
* Undo superfluous change
* remove default value for extension parameter
* fix compilation issues
* GRPH-46-Quit_command_cliwallet
* removed multiple function definition
* Fixed chainparameter update proposal issue
* Move GPOS withdraw logic to have single transaction(also single fee) and update API
* Added log for authorization failure of proposal operations
* Votes consideration on GPOS activation
* bump fc version
* fix gpos tests
* Bump fc version
* Updated gpos/voting_tests
* Fixed withdraw vesting bug
* Added unit test
* Update hardfork date for TESTNET, sync fc module and update logs
* avoid wlog as it filling up space
* Beatrice hot fix(sync issue fix)
* gpos tests fix
* Set hardfork date to Jan5th on TESTNET
* Merge Elasticplugin, snapshot plugin and graphene updates to beatrice (#304)
* check witness signature before adding block to fork db
* Replace verify_no_send_in_progress with no_parallel_execution_guard
* fixed cli_wallet log issue
* Port plugin sanitization code
* avoid directly overwriting wallet file
* Implemented "plugins" config variable
* allow plugin to have descriptions
* Merge pull request #444 from oxarbitrage/elasticsearch
Elasticsearch plugin
* Merge pull request #500 from oxarbitrage/elasticsearch-extras
es_objects plugin
* Merge pull request #873 from pmconrad/585_fix_history_ids
Fix history ids
* Merge pull request #1201 from oxarbitrage/elasticsearch_tests2
Elasticsearch refactor
* Merge pull request #1271 from oxarbitrage/es_objects
refine es_objects plugin
* Merge pull request #1429 from oxarbitrage/es_objects_templates
Add an adaptor to es_objects and template function to reduce code
* Merge pull request #1458 from oxarbitrage/issue1455
add option elasticsearch-start-es-after-block to es plugin
* Merge pull request #1541 from oxarbitrage/es_objects_start_after_block
add es-objects-start-es-after-block option
* explicitly cleanup external library facilities
* Merge pull request #1717 from oxarbitrage/issue1652
add genesis data to es_objects
* Merge pull request #1073 from xiangxn/merge-impacted
merge impacted into db_notify
* Merge pull request #1725 from oxarbitrage/issue1682
elasticsearch history api #1682
* change ES index prefixes to Peerplays-specific
* sync develop with beatrice
* fix the data writing to ES during sync issues
* fix CLI tests
* brought updates from mainnet branch (#285)
* Fix unit test failures (#289)
* fixed unit test failures from the recent merges
* fixed unit test failures from the recent merges
* enable snapshot plugin (#288)
* sync fc branch(build optimization changes)
* update to es plugin
* fix verify witness signature method (#295)
* enable mandatory plugins to have smooth transition for next release
* updated tests to keep in-line with plugin changes
Co-authored-by: Sandip Patel <sandip@knackroot.com>
Co-authored-by: Peter Conrad <conrad@quisquis.de>
Co-authored-by: Alfredo <oxarbitrage@gmail.com>
Co-authored-by: Abit <abitmore@users.noreply.github.com>
Co-authored-by: crypto-ape <43807588+crypto-ape@users.noreply.github.com>
Co-authored-by: gladcow <s.gladkov@pbsa.info>
* sync latest fc commit on beatrice
* sweeps winner_ticket_id changes
Co-authored-by: Bobinson K B <bobinson@gmail.com>
Co-authored-by: gladcow <s.gladkov@pbsa.info>
Co-authored-by: Alfredo Garcia <oxarbitrage@gmail.com>
Co-authored-by: Miha Čančula <miha@noughmad.eu>
Co-authored-by: Ronak Patel <r.patel@pbsa.info>
Co-authored-by: Srdjan Obucina <obucinac@gmail.com>
Co-authored-by: Peter Conrad <conrad@quisquis.de>
Co-authored-by: Peter Conrad <cyrano@quisquis.de>
Co-authored-by: Abit <abitmore@users.noreply.github.com>
Co-authored-by: Roshan Syed <r.syed@pbsa.info>
Co-authored-by: cifer <maintianyu@gmail.com>
Co-authored-by: John Jones <jmjatlanta@gmail.com>
Co-authored-by: Sandip Patel <sandip@knackroot.com>
Co-authored-by: Wei Yang <richard.weiyang@gmail.com>
Co-authored-by: gladcow <jahr@yandex.ru>
Co-authored-by: satyakoneru <satyakoneru.iiith@gmail.com>
Co-authored-by: crypto-ape <43807588+crypto-ape@users.noreply.github.com>
2020-04-15 15:04:15 +00:00
# include <graphene/chain/impacted.hpp>
2017-04-04 21:02:19 +00:00
# include <graphene/chain/account_evaluator.hpp>
# include <graphene/chain/account_object.hpp>
# include <graphene/chain/config.hpp>
# include <graphene/chain/database.hpp>
# include <graphene/chain/evaluator.hpp>
# include <graphene/chain/operation_history_object.hpp>
# include <graphene/chain/transaction_evaluation_state.hpp>
2017-07-31 12:26:20 +00:00
# include <boost/algorithm/string/case_conv.hpp>
2017-07-28 16:23:57 +00:00
2017-04-04 21:02:19 +00:00
# include <fc/smart_ref_impl.hpp>
# include <fc/thread/thread.hpp>
# include <boost/polymorphic_cast.hpp>
#if 0
# ifdef DEFAULT_LOGGER
# undef DEFAULT_LOGGER
# endif
# define DEFAULT_LOGGER "bookie_plugin"
# endif
namespace graphene { namespace bookie {
namespace detail
{
2017-08-09 15:12:59 +00:00
/* As a plugin, we get notified of new/changed objects at the end of every block processed.
* For most objects , that ' s fine , because we expect them to always be around until the end of
* the block . However , with bet objects , it ' s possible that the user places a bet and it fills
* and is removed during the same block , so need another strategy to detect them immediately after
* they are created .
* We do this by creating a secondary index on bet_object . We don ' t actually use it
* to index any property of the bet , we just use it to register for callbacks .
*/
class persistent_bet_object_helper : public secondary_index
{
public :
virtual ~ persistent_bet_object_helper ( ) { }
virtual void object_inserted ( const object & obj ) override ;
//virtual void object_removed( const object& obj ) override;
//virtual void about_to_modify( const object& before ) override;
virtual void object_modified ( const object & after ) override ;
void set_plugin_instance ( bookie_plugin * instance ) { _bookie_plugin = instance ; }
private :
bookie_plugin * _bookie_plugin ;
} ;
void persistent_bet_object_helper : : object_inserted ( const object & obj )
{
const bet_object & bet_obj = * boost : : polymorphic_downcast < const bet_object * > ( & obj ) ;
_bookie_plugin - > database ( ) . create < persistent_bet_object > ( [ & ] ( persistent_bet_object & saved_bet_obj ) {
saved_bet_obj . ephemeral_bet_object = bet_obj ;
} ) ;
}
void persistent_bet_object_helper : : object_modified ( const object & after )
{
database & db = _bookie_plugin - > database ( ) ;
auto & persistent_bets_by_bet_id = db . get_index_type < persistent_bet_index > ( ) . indices ( ) . get < by_bet_id > ( ) ;
const bet_object & bet_obj = * boost : : polymorphic_downcast < const bet_object * > ( & after ) ;
auto iter = persistent_bets_by_bet_id . find ( bet_obj . id ) ;
assert ( iter ! = persistent_bets_by_bet_id . end ( ) ) ;
if ( iter ! = persistent_bets_by_bet_id . end ( ) )
db . modify ( * iter , [ & ] ( persistent_bet_object & saved_bet_obj ) {
saved_bet_obj . ephemeral_bet_object = bet_obj ;
} ) ;
}
//////////// end bet_object ///////////////////
2018-02-07 15:12:58 +00:00
class persistent_betting_market_object_helper : public secondary_index
{
public :
virtual ~ persistent_betting_market_object_helper ( ) { }
virtual void object_inserted ( const object & obj ) override ;
//virtual void object_removed( const object& obj ) override;
//virtual void about_to_modify( const object& before ) override;
virtual void object_modified ( const object & after ) override ;
void set_plugin_instance ( bookie_plugin * instance ) { _bookie_plugin = instance ; }
private :
bookie_plugin * _bookie_plugin ;
} ;
void persistent_betting_market_object_helper : : object_inserted ( const object & obj )
{
const betting_market_object & betting_market_obj = * boost : : polymorphic_downcast < const betting_market_object * > ( & obj ) ;
_bookie_plugin - > database ( ) . create < persistent_betting_market_object > ( [ & ] ( persistent_betting_market_object & saved_betting_market_obj ) {
saved_betting_market_obj . ephemeral_betting_market_object = betting_market_obj ;
} ) ;
}
void persistent_betting_market_object_helper : : object_modified ( const object & after )
{
database & db = _bookie_plugin - > database ( ) ;
auto & persistent_betting_markets_by_betting_market_id = db . get_index_type < persistent_betting_market_index > ( ) . indices ( ) . get < by_betting_market_id > ( ) ;
const betting_market_object & betting_market_obj = * boost : : polymorphic_downcast < const betting_market_object * > ( & after ) ;
auto iter = persistent_betting_markets_by_betting_market_id . find ( betting_market_obj . id ) ;
assert ( iter ! = persistent_betting_markets_by_betting_market_id . end ( ) ) ;
if ( iter ! = persistent_betting_markets_by_betting_market_id . end ( ) )
db . modify ( * iter , [ & ] ( persistent_betting_market_object & saved_betting_market_obj ) {
saved_betting_market_obj . ephemeral_betting_market_object = betting_market_obj ;
} ) ;
}
//////////// end betting_market_object ///////////////////
class persistent_betting_market_group_object_helper : public secondary_index
{
public :
virtual ~ persistent_betting_market_group_object_helper ( ) { }
virtual void object_inserted ( const object & obj ) override ;
//virtual void object_removed( const object& obj ) override;
//virtual void about_to_modify( const object& before ) override;
virtual void object_modified ( const object & after ) override ;
void set_plugin_instance ( bookie_plugin * instance ) { _bookie_plugin = instance ; }
private :
bookie_plugin * _bookie_plugin ;
} ;
void persistent_betting_market_group_object_helper : : object_inserted ( const object & obj )
{
const betting_market_group_object & betting_market_group_obj = * boost : : polymorphic_downcast < const betting_market_group_object * > ( & obj ) ;
_bookie_plugin - > database ( ) . create < persistent_betting_market_group_object > ( [ & ] ( persistent_betting_market_group_object & saved_betting_market_group_obj ) {
saved_betting_market_group_obj . ephemeral_betting_market_group_object = betting_market_group_obj ;
} ) ;
}
void persistent_betting_market_group_object_helper : : object_modified ( const object & after )
{
database & db = _bookie_plugin - > database ( ) ;
auto & persistent_betting_market_groups_by_betting_market_group_id = db . get_index_type < persistent_betting_market_group_index > ( ) . indices ( ) . get < by_betting_market_group_id > ( ) ;
const betting_market_group_object & betting_market_group_obj = * boost : : polymorphic_downcast < const betting_market_group_object * > ( & after ) ;
auto iter = persistent_betting_market_groups_by_betting_market_group_id . find ( betting_market_group_obj . id ) ;
assert ( iter ! = persistent_betting_market_groups_by_betting_market_group_id . end ( ) ) ;
if ( iter ! = persistent_betting_market_groups_by_betting_market_group_id . end ( ) )
db . modify ( * iter , [ & ] ( persistent_betting_market_group_object & saved_betting_market_group_obj ) {
saved_betting_market_group_obj . ephemeral_betting_market_group_object = betting_market_group_obj ;
} ) ;
}
//////////// end betting_market_group_object ///////////////////
class persistent_event_object_helper : public secondary_index
{
public :
virtual ~ persistent_event_object_helper ( ) { }
virtual void object_inserted ( const object & obj ) override ;
//virtual void object_removed( const object& obj ) override;
//virtual void about_to_modify( const object& before ) override;
virtual void object_modified ( const object & after ) override ;
void set_plugin_instance ( bookie_plugin * instance ) { _bookie_plugin = instance ; }
private :
bookie_plugin * _bookie_plugin ;
} ;
void persistent_event_object_helper : : object_inserted ( const object & obj )
{
const event_object & event_obj = * boost : : polymorphic_downcast < const event_object * > ( & obj ) ;
_bookie_plugin - > database ( ) . create < persistent_event_object > ( [ & ] ( persistent_event_object & saved_event_obj ) {
saved_event_obj . ephemeral_event_object = event_obj ;
} ) ;
}
void persistent_event_object_helper : : object_modified ( const object & after )
{
database & db = _bookie_plugin - > database ( ) ;
auto & persistent_events_by_event_id = db . get_index_type < persistent_event_index > ( ) . indices ( ) . get < by_event_id > ( ) ;
const event_object & event_obj = * boost : : polymorphic_downcast < const event_object * > ( & after ) ;
auto iter = persistent_events_by_event_id . find ( event_obj . id ) ;
assert ( iter ! = persistent_events_by_event_id . end ( ) ) ;
if ( iter ! = persistent_events_by_event_id . end ( ) )
db . modify ( * iter , [ & ] ( persistent_event_object & saved_event_obj ) {
saved_event_obj . ephemeral_event_object = event_obj ;
} ) ;
}
//////////// end event_object ///////////////////
2017-04-04 21:02:19 +00:00
class bookie_plugin_impl
{
public :
bookie_plugin_impl ( bookie_plugin & _plugin )
: _self ( _plugin )
{ }
virtual ~ bookie_plugin_impl ( ) ;
/**
* Called After a block has been applied and committed . The callback
* should not yield and should execute quickly .
*/
void on_objects_changed ( const vector < object_id_type > & changed_object_ids ) ;
2017-09-05 15:06:38 +00:00
void on_objects_new ( const vector < object_id_type > & new_object_ids ) ;
void on_objects_removed ( const vector < object_id_type > & removed_object_ids ) ;
2017-04-04 21:02:19 +00:00
/** this method is called as a callback after a block is applied
* and will process / index all operations that were applied in the block .
*/
void on_block_applied ( const signed_block & b ) ;
2017-07-28 16:23:57 +00:00
asset get_total_matched_bet_amount_for_betting_market_group ( betting_market_group_id_type group_id ) ;
2017-07-31 12:26:20 +00:00
void fill_localized_event_strings ( ) ;
2017-08-09 21:17:51 +00:00
std : : vector < event_object > get_events_containing_sub_string ( const std : : string & sub_string , const std : : string & language ) ;
2017-07-31 12:26:20 +00:00
2017-04-04 21:02:19 +00:00
graphene : : chain : : database & database ( )
{
return _self . database ( ) ;
}
2017-07-31 12:26:20 +00:00
// 1.18. "Washington Capitals/Chicago Blackhawks"
typedef std : : pair < event_id_type , std : : string > event_string ;
struct event_string_less : public std : : less < const event_string & >
{
bool operator ( ) ( const event_string & _left , const event_string & _right ) const
{
return ( _left . first . instance < _right . first . instance ) ;
}
} ;
typedef flat_set < event_string , event_string_less > event_string_set ;
// "en"
std : : map < std : : string , event_string_set > localized_event_strings ;
2017-04-04 21:02:19 +00:00
bookie_plugin & _self ;
flat_set < account_id_type > _tracked_accounts ;
} ;
bookie_plugin_impl : : ~ bookie_plugin_impl ( )
{
}
2017-09-05 15:06:38 +00:00
void bookie_plugin_impl : : on_objects_new ( const vector < object_id_type > & new_object_ids )
{
}
void bookie_plugin_impl : : on_objects_removed ( const vector < object_id_type > & removed_object_ids )
{
}
2017-04-04 21:02:19 +00:00
void bookie_plugin_impl : : on_objects_changed ( const vector < object_id_type > & changed_object_ids )
{
}
2017-09-06 20:52:58 +00:00
bool is_operation_history_object_stored ( operation_history_id_type id )
{
if ( id = = operation_history_id_type ( ) )
{
elog ( " Warning: the operation history object for an operation the bookie plugin needs to track "
" has id of ${id}, which means the account history plugin isn't storing this operation, or that "
" it is running after the bookie plugin. Make sure the account history plugin is tracking operations for "
" all accounts,, and that it is loaded before the bookie plugin " , ( " id " , id ) ) ;
return false ;
}
else
return true ;
}
2017-04-04 21:02:19 +00:00
void bookie_plugin_impl : : on_block_applied ( const signed_block & )
2018-02-07 15:12:58 +00:00
{ try {
2017-04-04 21:02:19 +00:00
graphene : : chain : : database & db = database ( ) ;
2017-07-27 23:35:13 +00:00
const vector < optional < operation_history_object > > & hist = db . get_applied_operations ( ) ;
2017-08-09 15:12:59 +00:00
for ( const optional < operation_history_object > & o_op : hist )
2017-07-28 16:23:57 +00:00
{
if ( ! o_op . valid ( ) )
continue ;
2017-09-05 15:06:38 +00:00
2017-07-28 16:23:57 +00:00
const operation_history_object & op = * o_op ;
2017-08-09 15:12:59 +00:00
if ( op . op . which ( ) = = operation : : tag < bet_matched_operation > : : value )
2017-07-28 16:23:57 +00:00
{
2017-09-05 15:06:38 +00:00
const bet_matched_operation & bet_matched_op = op . op . get < bet_matched_operation > ( ) ;
//idump((bet_matched_op));
const asset & amount_bet = bet_matched_op . amount_bet ;
// object may no longer exist
//const bet_object& bet = bet_matched_op.bet_id(db);
auto & persistent_bets_by_bet_id = db . get_index_type < persistent_bet_index > ( ) . indices ( ) . get < by_bet_id > ( ) ;
auto bet_iter = persistent_bets_by_bet_id . find ( bet_matched_op . bet_id ) ;
assert ( bet_iter ! = persistent_bets_by_bet_id . end ( ) ) ;
if ( bet_iter ! = persistent_bets_by_bet_id . end ( ) )
{
db . modify ( * bet_iter , [ & ] ( persistent_bet_object & obj ) {
obj . amount_matched + = amount_bet . amount ;
2017-09-06 20:52:58 +00:00
if ( is_operation_history_object_stored ( op . id ) )
obj . associated_operations . emplace_back ( op . id ) ;
2017-09-05 15:06:38 +00:00
} ) ;
const bet_object & bet_obj = bet_iter - > ephemeral_bet_object ;
2018-02-07 15:12:58 +00:00
auto & persistent_betting_market_idx = db . get_index_type < persistent_betting_market_index > ( ) . indices ( ) . get < by_betting_market_id > ( ) ;
auto persistent_betting_market_object_iter = persistent_betting_market_idx . find ( bet_obj . betting_market_id ) ;
FC_ASSERT ( persistent_betting_market_object_iter ! = persistent_betting_market_idx . end ( ) ) ;
const betting_market_object & betting_market = persistent_betting_market_object_iter - > ephemeral_betting_market_object ;
auto & persistent_betting_market_group_idx = db . get_index_type < persistent_betting_market_group_index > ( ) . indices ( ) . get < by_betting_market_group_id > ( ) ;
auto persistent_betting_market_group_object_iter = persistent_betting_market_group_idx . find ( betting_market . group_id ) ;
FC_ASSERT ( persistent_betting_market_group_object_iter ! = persistent_betting_market_group_idx . end ( ) ) ;
const betting_market_group_object & betting_market_group = persistent_betting_market_group_object_iter - > ephemeral_betting_market_group_object ;
// if the object is still in the main database, keep the running total there
// otherwise, add it directly to the persistent version
auto & betting_market_group_idx = db . get_index_type < betting_market_group_object_index > ( ) . indices ( ) . get < by_id > ( ) ;
auto betting_market_group_iter = betting_market_group_idx . find ( betting_market_group . id ) ;
if ( betting_market_group_iter ! = betting_market_group_idx . end ( ) )
db . modify ( * betting_market_group_iter , [ & ] ( betting_market_group_object & obj ) {
obj . total_matched_bets_amount + = amount_bet . amount ;
} ) ;
else
db . modify ( * persistent_betting_market_group_object_iter , [ & ] ( persistent_betting_market_group_object & obj ) {
obj . ephemeral_betting_market_group_object . total_matched_bets_amount + = amount_bet . amount ;
} ) ;
2017-09-05 15:06:38 +00:00
}
2017-07-28 16:23:57 +00:00
}
2017-08-09 15:12:59 +00:00
else if ( op . op . which ( ) = = operation : : tag < event_create_operation > : : value )
2017-07-31 12:26:20 +00:00
{
2017-09-05 15:06:38 +00:00
FC_ASSERT ( op . result . which ( ) = = operation_result : : tag < object_id_type > : : value ) ;
//object_id_type object_id = op.result.get<object_id_type>();
event_id_type object_id = op . result . get < object_id_type > ( ) ;
FC_ASSERT ( db . find_object ( object_id ) , " invalid event specified " ) ;
const event_create_operation & event_create_op = op . op . get < event_create_operation > ( ) ;
for ( const std : : pair < std : : string , std : : string > & pair : event_create_op . name )
localized_event_strings [ pair . first ] . insert ( event_string ( object_id , pair . second ) ) ;
2017-07-31 12:26:20 +00:00
}
2017-08-09 15:12:59 +00:00
else if ( op . op . which ( ) = = operation : : tag < event_update_operation > : : value )
2017-07-31 12:26:20 +00:00
{
2017-09-05 15:06:38 +00:00
const event_update_operation & event_create_op = op . op . get < event_update_operation > ( ) ;
if ( ! event_create_op . new_name . valid ( ) )
continue ;
event_id_type event_id = event_create_op . event_id ;
for ( const std : : pair < std : : string , std : : string > & pair : * event_create_op . new_name )
{
// try insert
std : : pair < event_string_set : : iterator , bool > result =
localized_event_strings [ pair . first ] . insert ( event_string ( event_id , pair . second ) ) ;
if ( ! result . second )
// update string only
result . first - > second = pair . second ;
}
2017-07-31 12:26:20 +00:00
}
2017-09-06 20:52:58 +00:00
else if ( op . op . which ( ) = = operation : : tag < bet_canceled_operation > : : value )
{
const bet_canceled_operation & bet_canceled_op = op . op . get < bet_canceled_operation > ( ) ;
auto & persistent_bets_by_bet_id = db . get_index_type < persistent_bet_index > ( ) . indices ( ) . get < by_bet_id > ( ) ;
auto bet_iter = persistent_bets_by_bet_id . find ( bet_canceled_op . bet_id ) ;
assert ( bet_iter ! = persistent_bets_by_bet_id . end ( ) ) ;
if ( bet_iter ! = persistent_bets_by_bet_id . end ( ) )
{
2020-02-10 13:51:50 +00:00
//ilog("Adding bet_canceled_operation ${canceled_id} to bet ${bet_id}'s associated operations",
// ("canceled_id", op.id)("bet_id", bet_canceled_op.bet_id));
2017-09-06 20:52:58 +00:00
if ( is_operation_history_object_stored ( op . id ) )
db . modify ( * bet_iter , [ & ] ( persistent_bet_object & obj ) {
obj . associated_operations . emplace_back ( op . id ) ;
} ) ;
}
}
else if ( op . op . which ( ) = = operation : : tag < bet_adjusted_operation > : : value )
{
const bet_adjusted_operation & bet_adjusted_op = op . op . get < bet_adjusted_operation > ( ) ;
auto & persistent_bets_by_bet_id = db . get_index_type < persistent_bet_index > ( ) . indices ( ) . get < by_bet_id > ( ) ;
auto bet_iter = persistent_bets_by_bet_id . find ( bet_adjusted_op . bet_id ) ;
assert ( bet_iter ! = persistent_bets_by_bet_id . end ( ) ) ;
if ( bet_iter ! = persistent_bets_by_bet_id . end ( ) )
{
2020-02-10 13:51:50 +00:00
//ilog("Adding bet_adjusted_operation ${adjusted_id} to bet ${bet_id}'s associated operations",
// ("adjusted_id", op.id)("bet_id", bet_adjusted_op.bet_id));
2017-09-06 20:52:58 +00:00
if ( is_operation_history_object_stored ( op . id ) )
db . modify ( * bet_iter , [ & ] ( persistent_bet_object & obj ) {
obj . associated_operations . emplace_back ( op . id ) ;
} ) ;
}
}
2017-07-28 16:23:57 +00:00
}
2018-02-07 15:12:58 +00:00
} FC_RETHROW_EXCEPTIONS ( warn , " " ) }
2017-07-28 16:23:57 +00:00
2017-07-31 12:26:20 +00:00
void bookie_plugin_impl : : fill_localized_event_strings ( )
{
graphene : : chain : : database & db = database ( ) ;
const auto & event_index = db . get_index_type < event_object_index > ( ) . indices ( ) . get < by_id > ( ) ;
auto event_itr = event_index . cbegin ( ) ;
while ( event_itr ! = event_index . cend ( ) )
{
const event_object & event_obj = * event_itr ;
+ + event_itr ;
for ( const std : : pair < std : : string , std : : string > & pair : event_obj . name )
{
localized_event_strings [ pair . first ] . insert ( event_string ( event_obj . id , pair . second ) ) ;
}
}
}
2017-08-09 21:17:51 +00:00
std : : vector < event_object > bookie_plugin_impl : : get_events_containing_sub_string ( const std : : string & sub_string , const std : : string & language )
2017-07-31 12:26:20 +00:00
{
2017-08-09 21:17:51 +00:00
graphene : : chain : : database & db = database ( ) ;
std : : vector < event_object > events ;
if ( localized_event_strings . find ( language ) ! = localized_event_strings . end ( ) )
{
std : : string lower_case_sub_string = boost : : algorithm : : to_lower_copy ( sub_string ) ;
const event_string_set & language_set = localized_event_strings [ language ] ;
for ( const event_string & pair : language_set )
{
std : : string lower_case_string = boost : : algorithm : : to_lower_copy ( pair . second ) ;
if ( lower_case_string . find ( lower_case_sub_string ) ! = std : : string : : npos )
events . push_back ( pair . first ( db ) ) ;
}
}
return events ;
2017-07-31 12:26:20 +00:00
}
2017-07-28 16:23:57 +00:00
asset bookie_plugin_impl : : get_total_matched_bet_amount_for_betting_market_group ( betting_market_group_id_type group_id )
{
2017-08-09 21:17:51 +00:00
graphene : : chain : : database & db = database ( ) ;
FC_ASSERT ( db . find_object ( group_id ) , " Invalid betting market group specified " ) ;
const betting_market_group_object & betting_market_group = group_id ( db ) ;
return asset ( betting_market_group . total_matched_bets_amount , betting_market_group . asset_id ) ;
2017-04-04 21:02:19 +00:00
}
} // end namespace detail
bookie_plugin : : bookie_plugin ( ) :
my ( new detail : : bookie_plugin_impl ( * this ) )
{
}
bookie_plugin : : ~ bookie_plugin ( )
{
}
std : : string bookie_plugin : : plugin_name ( ) const
{
return " bookie " ;
}
2017-08-09 15:12:59 +00:00
void bookie_plugin : : plugin_set_program_options ( boost : : program_options : : options_description & cli ,
boost : : program_options : : options_description & cfg )
2017-04-04 21:02:19 +00:00
{
2017-07-27 23:35:13 +00:00
//cli.add_options()
// ("track-account", boost::program_options::value<std::vector<std::string>>()->composing()->multitoken(), "Account ID to track history for (may specify multiple times)")
// ;
//cfg.add(cli);
2017-04-04 21:02:19 +00:00
}
void bookie_plugin : : plugin_initialize ( const boost : : program_options : : variables_map & options )
{
2017-07-31 12:26:20 +00:00
ilog ( " bookie plugin: plugin_startup() begin " ) ;
2017-09-07 19:55:44 +00:00
database ( ) . force_slow_replays ( ) ;
2017-07-31 12:26:20 +00:00
database ( ) . applied_block . connect ( [ & ] ( const signed_block & b ) { my - > on_block_applied ( b ) ; } ) ;
database ( ) . changed_objects . connect ( [ & ] ( const vector < object_id_type > & changed_object_ids , const fc : : flat_set < graphene : : chain : : account_id_type > & impacted_accounts ) { my - > on_objects_changed ( changed_object_ids ) ; } ) ;
2017-09-05 15:06:38 +00:00
database ( ) . new_objects . connect ( [ this ] ( const vector < object_id_type > & ids , const flat_set < account_id_type > & impacted_accounts ) { my - > on_objects_new ( ids ) ; } ) ;
database ( ) . removed_objects . connect ( [ this ] ( const vector < object_id_type > & ids , const vector < const object * > & objs , const flat_set < account_id_type > & impacted_accounts ) { my - > on_objects_removed ( ids ) ; } ) ;
2017-07-31 12:26:20 +00:00
//auto event_index =
2017-08-10 19:44:07 +00:00
database ( ) . add_index < primary_index < detail : : persistent_event_index > > ( ) ;
2017-08-09 15:12:59 +00:00
database ( ) . add_index < primary_index < detail : : persistent_betting_market_group_index > > ( ) ;
database ( ) . add_index < primary_index < detail : : persistent_betting_market_index > > ( ) ;
database ( ) . add_index < primary_index < detail : : persistent_bet_index > > ( ) ;
const primary_index < bet_object_index > & bet_object_idx = database ( ) . get_index_type < primary_index < bet_object_index > > ( ) ;
primary_index < bet_object_index > & nonconst_bet_object_idx = const_cast < primary_index < bet_object_index > & > ( bet_object_idx ) ;
detail : : persistent_bet_object_helper * persistent_bet_object_helper_index = nonconst_bet_object_idx . add_secondary_index < detail : : persistent_bet_object_helper > ( ) ;
persistent_bet_object_helper_index - > set_plugin_instance ( this ) ;
2017-07-31 12:26:20 +00:00
2018-02-07 15:12:58 +00:00
const primary_index < betting_market_object_index > & betting_market_object_idx = database ( ) . get_index_type < primary_index < betting_market_object_index > > ( ) ;
primary_index < betting_market_object_index > & nonconst_betting_market_object_idx = const_cast < primary_index < betting_market_object_index > & > ( betting_market_object_idx ) ;
detail : : persistent_betting_market_object_helper * persistent_betting_market_object_helper_index = nonconst_betting_market_object_idx . add_secondary_index < detail : : persistent_betting_market_object_helper > ( ) ;
persistent_betting_market_object_helper_index - > set_plugin_instance ( this ) ;
const primary_index < betting_market_group_object_index > & betting_market_group_object_idx = database ( ) . get_index_type < primary_index < betting_market_group_object_index > > ( ) ;
primary_index < betting_market_group_object_index > & nonconst_betting_market_group_object_idx = const_cast < primary_index < betting_market_group_object_index > & > ( betting_market_group_object_idx ) ;
detail : : persistent_betting_market_group_object_helper * persistent_betting_market_group_object_helper_index = nonconst_betting_market_group_object_idx . add_secondary_index < detail : : persistent_betting_market_group_object_helper > ( ) ;
persistent_betting_market_group_object_helper_index - > set_plugin_instance ( this ) ;
const primary_index < event_object_index > & event_object_idx = database ( ) . get_index_type < primary_index < event_object_index > > ( ) ;
primary_index < event_object_index > & nonconst_event_object_idx = const_cast < primary_index < event_object_index > & > ( event_object_idx ) ;
detail : : persistent_event_object_helper * persistent_event_object_helper_index = nonconst_event_object_idx . add_secondary_index < detail : : persistent_event_object_helper > ( ) ;
persistent_event_object_helper_index - > set_plugin_instance ( this ) ;
2017-07-31 12:26:20 +00:00
ilog ( " bookie plugin: plugin_startup() end " ) ;
}
2017-04-04 21:02:19 +00:00
void bookie_plugin : : plugin_startup ( )
{
2017-07-27 23:35:13 +00:00
ilog ( " bookie plugin: plugin_startup() " ) ;
2017-07-31 12:26:20 +00:00
my - > fill_localized_event_strings ( ) ;
2017-04-04 21:02:19 +00:00
}
flat_set < account_id_type > bookie_plugin : : tracked_accounts ( ) const
{
return my - > _tracked_accounts ;
}
2017-07-28 16:23:57 +00:00
asset bookie_plugin : : get_total_matched_bet_amount_for_betting_market_group ( betting_market_group_id_type group_id )
{
2017-07-31 12:26:20 +00:00
ilog ( " bookie plugin: get_total_matched_bet_amount_for_betting_market_group($group_id) " , ( " group_d " , group_id ) ) ;
2017-07-28 16:23:57 +00:00
return my - > get_total_matched_bet_amount_for_betting_market_group ( group_id ) ;
}
2017-08-09 21:17:51 +00:00
std : : vector < event_object > bookie_plugin : : get_events_containing_sub_string ( const std : : string & sub_string , const std : : string & language )
2017-07-31 12:26:20 +00:00
{
2017-08-09 21:17:51 +00:00
ilog ( " bookie plugin: get_events_containing_sub_string(${sub_string}, ${language}) " , ( sub_string ) ( language ) ) ;
return my - > get_events_containing_sub_string ( sub_string , language ) ;
2017-07-31 12:26:20 +00:00
}
2017-07-28 16:23:57 +00:00
2017-04-04 21:02:19 +00:00
} }