157 lines
6.2 KiB
C++
157 lines
6.2 KiB
C++
/*
|
|
* 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 <graphene/chain/exceptions.hpp>
|
|
#include <graphene/chain/protocol/confidential.hpp>
|
|
#include <graphene/chain/confidential_evaluator.hpp>
|
|
#include <graphene/chain/database.hpp>
|
|
|
|
namespace graphene { namespace chain {
|
|
|
|
void_result transfer_to_blind_evaluator::do_evaluate( const transfer_to_blind_operation& o )
|
|
{ try {
|
|
const auto& d = db();
|
|
|
|
const auto& atype = o.amount.asset_id(db());
|
|
FC_ASSERT( atype.allow_confidential() );
|
|
FC_ASSERT( !atype.is_transfer_restricted() );
|
|
FC_ASSERT( !(atype.options.flags & white_list) );
|
|
|
|
for( const auto& out : o.outputs )
|
|
{
|
|
for( const auto& a : out.owner.account_auths )
|
|
a.first(d); // verify all accounts exist and are valid
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
|
|
void_result transfer_to_blind_evaluator::do_apply( const transfer_to_blind_operation& o )
|
|
{ try {
|
|
db().adjust_balance( o.from, -o.amount );
|
|
|
|
const auto& add = o.amount.asset_id(db()).dynamic_asset_data_id(db()); // verify fee is a legit asset
|
|
db().modify( add, [&]( asset_dynamic_data_object& obj ){
|
|
obj.confidential_supply += o.amount.amount;
|
|
FC_ASSERT( obj.confidential_supply >= 0 );
|
|
});
|
|
for( const auto& out : o.outputs )
|
|
{
|
|
db().create<blinded_balance_object>( [&]( blinded_balance_object& obj ){
|
|
obj.asset_id = o.amount.asset_id;
|
|
obj.owner = out.owner;
|
|
obj.commitment = out.commitment;
|
|
});
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
|
|
void_result transfer_from_blind_evaluator::do_evaluate( const transfer_from_blind_operation& o )
|
|
{ try {
|
|
const auto& d = db();
|
|
o.fee.asset_id(d); // verify fee is a legit asset
|
|
const auto& bbi = d.get_index_type<blinded_balance_index>();
|
|
const auto& cidx = bbi.indices().get<by_commitment>();
|
|
for( const auto& in : o.inputs )
|
|
{
|
|
auto itr = cidx.find( in.commitment );
|
|
FC_ASSERT( itr != cidx.end() );
|
|
FC_ASSERT( itr->asset_id == o.fee.asset_id );
|
|
FC_ASSERT( itr->owner == in.owner );
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
void_result transfer_from_blind_evaluator::do_apply( const transfer_from_blind_operation& o )
|
|
{ try {
|
|
db().adjust_balance( o.fee_payer(), o.fee );
|
|
db().adjust_balance( o.to, o.amount );
|
|
const auto& bbi = db().get_index_type<blinded_balance_index>();
|
|
const auto& cidx = bbi.indices().get<by_commitment>();
|
|
for( const auto& in : o.inputs )
|
|
{
|
|
auto itr = cidx.find( in.commitment );
|
|
FC_ASSERT( itr != cidx.end() );
|
|
db().remove( *itr );
|
|
}
|
|
const auto& add = o.amount.asset_id(db()).dynamic_asset_data_id(db()); // verify fee is a legit asset
|
|
db().modify( add, [&]( asset_dynamic_data_object& obj ){
|
|
obj.confidential_supply -= o.amount.amount + o.fee.amount;
|
|
FC_ASSERT( obj.confidential_supply >= 0 );
|
|
});
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
|
|
|
|
|
|
|
|
void_result blind_transfer_evaluator::do_evaluate( const blind_transfer_operation& o )
|
|
{ try {
|
|
const auto& d = db();
|
|
o.fee.asset_id(db()); // verify fee is a legit asset
|
|
const auto& bbi = db().get_index_type<blinded_balance_index>();
|
|
const auto& cidx = bbi.indices().get<by_commitment>();
|
|
for( const auto& out : o.outputs )
|
|
{
|
|
for( const auto& a : out.owner.account_auths )
|
|
a.first(d); // verify all accounts exist and are valid
|
|
}
|
|
for( const auto& in : o.inputs )
|
|
{
|
|
auto itr = cidx.find( in.commitment );
|
|
GRAPHENE_ASSERT( itr != cidx.end(), blind_transfer_unknown_commitment, "", ("commitment",in.commitment) );
|
|
FC_ASSERT( itr->asset_id == o.fee.asset_id );
|
|
FC_ASSERT( itr->owner == in.owner );
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
void_result blind_transfer_evaluator::do_apply( const blind_transfer_operation& o )
|
|
{ try {
|
|
db().adjust_balance( o.fee_payer(), o.fee ); // deposit the fee to the temp account
|
|
const auto& bbi = db().get_index_type<blinded_balance_index>();
|
|
const auto& cidx = bbi.indices().get<by_commitment>();
|
|
for( const auto& in : o.inputs )
|
|
{
|
|
auto itr = cidx.find( in.commitment );
|
|
GRAPHENE_ASSERT( itr != cidx.end(), blind_transfer_unknown_commitment, "", ("commitment",in.commitment) );
|
|
db().remove( *itr );
|
|
}
|
|
for( const auto& out : o.outputs )
|
|
{
|
|
db().create<blinded_balance_object>( [&]( blinded_balance_object& obj ){
|
|
obj.asset_id = o.fee.asset_id;
|
|
obj.owner = out.owner;
|
|
obj.commitment = out.commitment;
|
|
});
|
|
}
|
|
const auto& add = o.fee.asset_id(db()).dynamic_asset_data_id(db());
|
|
db().modify( add, [&]( asset_dynamic_data_object& obj ){
|
|
obj.confidential_supply -= o.fee.amount;
|
|
FC_ASSERT( obj.confidential_supply >= 0 );
|
|
});
|
|
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (o) ) }
|
|
|
|
|
|
} } // graphene::chain
|