/* * Copyright (c) 2015 Cryptonomex, Inc., and contributors. * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * * 1. Any modified source or binaries are used only with the BitShares network. * * 2. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * * 3. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #include namespace graphene { namespace chain { /** * Valid symbols can contain [A-Z0-9], and '.' * They must start with [A, Z] * They must end with [A, Z] * They can contain a maximum of one '.' */ bool is_valid_symbol( const string& symbol ) { if( symbol.size() < GRAPHENE_MIN_ASSET_SYMBOL_LENGTH ) return false; if( symbol.size() > GRAPHENE_MAX_ASSET_SYMBOL_LENGTH ) return false; if( !isalpha( symbol.front() ) ) return false; if( !isalpha( symbol.back() ) ) return false; bool dot_already_present = false; for( const auto c : symbol ) { if( (isalpha( c ) && isupper( c )) || isdigit(c) ) continue; if( c == '.' ) { if( dot_already_present ) return false; dot_already_present = true; continue; } return false; } return true; } share_type asset_issue_operation::calculate_fee(const fee_parameters_type& k)const { return k.fee + calculate_data_fee( fc::raw::pack_size(memo), k.price_per_kbyte ); } share_type asset_create_operation::calculate_fee(const asset_create_operation::fee_parameters_type& param)const { auto core_fee_required = param.long_symbol; switch(symbol.size()) { case 3: core_fee_required = param.symbol3; break; case 4: core_fee_required = param.symbol4; break; default: break; } // common_options contains several lists and a string. Charge fees for its size core_fee_required += calculate_data_fee( fc::raw::pack_size(*this), param.price_per_kbyte ); return core_fee_required; } void asset_create_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( is_valid_symbol(symbol) ); common_options.validate(); if( common_options.issuer_permissions & (disable_force_settle|global_settle) ) FC_ASSERT( bitasset_opts.valid() ); if( is_prediction_market ) { FC_ASSERT( bitasset_opts.valid(), "Cannot have a User-Issued Asset implement a prediction market." ); FC_ASSERT( common_options.issuer_permissions & global_settle ); } if( bitasset_opts ) bitasset_opts->validate(); asset dummy = asset(1) * common_options.core_exchange_rate; FC_ASSERT(dummy.asset_id == asset_id_type(1)); FC_ASSERT(precision <= 12); } void asset_update_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); if( new_issuer ) FC_ASSERT(issuer != *new_issuer); new_options.validate(); asset dummy = asset(1, asset_to_update) * new_options.core_exchange_rate; FC_ASSERT(dummy.asset_id == asset_id_type()); } share_type asset_update_operation::calculate_fee(const asset_update_operation::fee_parameters_type& k)const { return k.fee + calculate_data_fee( fc::raw::pack_size(*this), k.price_per_kbyte ); } void asset_publish_feed_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); feed.validate(); // maybe some of these could be moved to feed.validate() if( !feed.core_exchange_rate.is_null() ) { feed.core_exchange_rate.validate(); } if( (!feed.settlement_price.is_null()) && (!feed.core_exchange_rate.is_null()) ) { FC_ASSERT( feed.settlement_price.base.asset_id == feed.core_exchange_rate.base.asset_id ); } FC_ASSERT( !feed.settlement_price.is_null() ); FC_ASSERT( !feed.core_exchange_rate.is_null() ); FC_ASSERT( feed.is_for( asset_id ) ); } void asset_reserve_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( amount_to_reserve.amount.value <= GRAPHENE_MAX_SHARE_SUPPLY ); FC_ASSERT( amount_to_reserve.amount.value > 0 ); } void asset_issue_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( asset_to_issue.amount.value <= GRAPHENE_MAX_SHARE_SUPPLY ); FC_ASSERT( asset_to_issue.amount.value > 0 ); FC_ASSERT( asset_to_issue.asset_id != 0 ); } void asset_fund_fee_pool_operation::validate() const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( fee.asset_id == asset_id_type() ); FC_ASSERT( amount > 0 ); } void asset_settle_operation::validate() const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( amount.amount >= 0 ); } void asset_update_bitasset_operation::validate() const { FC_ASSERT( fee.amount >= 0 ); new_options.validate(); } void asset_update_feed_producers_operation::validate() const { FC_ASSERT( fee.amount >= 0 ); } void asset_global_settle_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( asset_to_settle == settle_price.base.asset_id ); } void bitasset_options::validate() const { FC_ASSERT(minimum_feeds > 0); FC_ASSERT(force_settlement_offset_percent <= GRAPHENE_100_PERCENT); FC_ASSERT(maximum_force_settlement_volume <= GRAPHENE_100_PERCENT); } void asset_options::validate()const { FC_ASSERT( max_supply > 0 ); FC_ASSERT( max_supply <= GRAPHENE_MAX_SHARE_SUPPLY ); FC_ASSERT( market_fee_percent <= GRAPHENE_100_PERCENT ); FC_ASSERT( max_market_fee >= 0 && max_market_fee <= GRAPHENE_MAX_SHARE_SUPPLY ); // There must be no high bits in permissions whose meaning is not known. FC_ASSERT( !(issuer_permissions & ~ASSET_ISSUER_PERMISSION_MASK) ); // The global_settle flag may never be set (this is a permission only) FC_ASSERT( !(flags & global_settle) ); // the witness_fed and committee_fed flags cannot be set simultaneously FC_ASSERT( (flags & (witness_fed_asset | committee_fed_asset)) != (witness_fed_asset | committee_fed_asset) ); core_exchange_rate.validate(); FC_ASSERT( core_exchange_rate.base.asset_id.instance.value == 0 || core_exchange_rate.quote.asset_id.instance.value == 0 ); if(!whitelist_authorities.empty() || !blacklist_authorities.empty()) FC_ASSERT( flags & white_list ); for( auto item : whitelist_markets ) { FC_ASSERT( blacklist_markets.find(item) == blacklist_markets.end() ); } for( auto item : blacklist_markets ) { FC_ASSERT( whitelist_markets.find(item) == whitelist_markets.end() ); } } void asset_claim_fees_operation::validate()const { FC_ASSERT( fee.amount >= 0 ); FC_ASSERT( amount_to_claim.amount > 0 ); } } } // namespace graphene::chain