2015-10-12 17:02:59 +00:00
|
|
|
/*
|
2015-10-12 17:48:40 +00:00
|
|
|
* 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.
|
2015-10-12 17:02:59 +00:00
|
|
|
*
|
|
|
|
|
* 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.
|
|
|
|
|
*
|
|
|
|
|
*/
|
2015-07-08 20:39:23 +00:00
|
|
|
#include <graphene/chain/protocol/asset_ops.hpp>
|
|
|
|
|
|
|
|
|
|
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 )
|
|
|
|
|
{
|
2015-10-08 18:59:43 +00:00
|
|
|
if( (isalpha( c ) && isupper( c )) || isdigit(c) )
|
2015-07-08 20:39:23 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if( c == '.' )
|
|
|
|
|
{
|
|
|
|
|
if( dot_already_present )
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
dot_already_present = true;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2015-10-08 18:59:43 +00:00
|
|
|
|
2015-07-08 22:45:53 +00:00
|
|
|
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 );
|
|
|
|
|
}
|
2015-07-08 20:39:23 +00:00
|
|
|
|
|
|
|
|
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) )
|
2015-07-09 17:50:47 +00:00
|
|
|
FC_ASSERT( bitasset_opts.valid() );
|
2015-07-08 20:39:23 +00:00
|
|
|
if( is_prediction_market )
|
|
|
|
|
{
|
2015-07-09 17:50:47 +00:00
|
|
|
FC_ASSERT( bitasset_opts.valid(), "Cannot have a User-Issued Asset implement a prediction market." );
|
2015-07-08 20:39:23 +00:00
|
|
|
FC_ASSERT( common_options.issuer_permissions & global_settle );
|
|
|
|
|
}
|
2015-07-09 17:50:47 +00:00
|
|
|
if( bitasset_opts ) bitasset_opts->validate();
|
2015-07-08 20:39:23 +00:00
|
|
|
|
|
|
|
|
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();
|
2015-10-15 21:59:46 +00:00
|
|
|
|
|
|
|
|
// 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.quote.asset_id == feed.core_exchange_rate.quote.asset_id );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FC_ASSERT( !feed.settlement_price.is_null() );
|
|
|
|
|
FC_ASSERT( !feed.core_exchange_rate.is_null() );
|
|
|
|
|
FC_ASSERT( feed.is_for( asset_id ) );
|
2015-07-08 20:39:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 );
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-09 17:50:47 +00:00
|
|
|
void asset_global_settle_operation::validate()const
|
2015-07-08 20:39:23 +00:00
|
|
|
{
|
|
|
|
|
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) );
|
2015-10-09 15:07:10 +00:00
|
|
|
// 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) );
|
2015-07-08 20:39:23 +00:00
|
|
|
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() );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} } // namespace graphene::chain
|