* private-key option update * ppy marketplace 1 - add evaluators and objects * NFT object and basic operations * ci: update .gitlab-ci.yml * ci: update .gitlab-ci.yml * NFT evaluators and basic tests, no evaluator checks * Evaluator checks in place * ppy marketplace 2 - batch sale, offer_object escrow * Database API * Wallet API * NFT metadata implemented * Fix NFT tests * Database API for NFT metadata and enumerables * ppy marketplace 4 - Add tests NFT+Marketplace * ppy marketplace 5 - Add revenue split * ppy marketplace 6 - Remove unnecessary files * ppy marketplace 7 - Add db, wallet changes and some NFT fixes * ppy marketplace 8 - Add pagination for list APIs * ci: update .gitlab-ci.yml * New DB API, list all NFTs, list NFTs by owner * Marketplace + NFT + RBAC (#368) * rbac1 - evaluators and op validators added * rbac2 - op_type hf checks * rbac3 - tx auth verify changes * Update .gitlab-ci.yml * rbac4 - basic op tests * rbac5 - clear expired and deleted permission linked auths * rbac6 - more tests * rbac7 - more tests * rbac8 - more tests * rbac9 - wallet and db api changes * rbac10 - db api changes for required signature fetch * rbac11 - add db_api tests * rbac12 - add missing code for key auths Co-authored-by: Roshan Syed <roshan.syed.rs@gmail.com> Co-authored-by: sierra19XX <15652887+sierra19XX@users.noreply.github.com> * Fix nft_get_token_uri returning empty string * Fix nft_mint_evaluator to save token_uri * Fix cli_wallet to properly pass metadata id for nft_create * ppy marketplace 9 - FC_REFLECT offer create op * Add stricter checks to NFTs * GPOS2 HF - Handle rolling period on missing blocks (#369) * Mainnet chain halt 5050 Issue (#370) * Unlisting offers, add result in offer history object * Reverting genesis.json wrong commit * Add non-transferable non-sellable properties to NFTs * Review comments - change variable names, use scoped enums * nft_metadata_update changes * NFT HF checks and op fee addition changes * NFT make revenue_split integer from double * revenue_split condition check allow zero or above * Peerplays Marketplace + NFT (#367) * ppy marketplace 1 - add evaluators and objects * NFT object and basic operations * ci: update .gitlab-ci.yml * ci: update .gitlab-ci.yml * NFT evaluators and basic tests, no evaluator checks * Evaluator checks in place * ppy marketplace 2 - batch sale, offer_object escrow * Database API * Wallet API * NFT metadata implemented * Fix NFT tests * Database API for NFT metadata and enumerables * ppy marketplace 4 - Add tests NFT+Marketplace * ppy marketplace 5 - Add revenue split * ppy marketplace 6 - Remove unnecessary files * ppy marketplace 7 - Add db, wallet changes and some NFT fixes * ppy marketplace 8 - Add pagination for list APIs * New DB API, list all NFTs, list NFTs by owner * Marketplace + NFT + RBAC (#368) * rbac1 - evaluators and op validators added * rbac2 - op_type hf checks * rbac3 - tx auth verify changes * Update .gitlab-ci.yml * rbac4 - basic op tests * rbac5 - clear expired and deleted permission linked auths * rbac6 - more tests * rbac7 - more tests * rbac8 - more tests * rbac9 - wallet and db api changes * rbac10 - db api changes for required signature fetch * rbac11 - add db_api tests * rbac12 - add missing code for key auths Co-authored-by: Roshan Syed <roshan.syed.rs@gmail.com> Co-authored-by: sierra19XX <15652887+sierra19XX@users.noreply.github.com> * Fix nft_get_token_uri returning empty string * Fix nft_mint_evaluator to save token_uri * Fix cli_wallet to properly pass metadata id for nft_create * ppy marketplace 9 - FC_REFLECT offer create op * Add stricter checks to NFTs * Unlisting offers, add result in offer history object * Reverting genesis.json wrong commit * Add non-transferable non-sellable properties to NFTs * Review comments - change variable names, use scoped enums * nft_metadata_update changes * NFT HF checks and op fee addition changes * NFT make revenue_split integer from double * revenue_split condition check allow zero or above Co-authored-by: Srdjan Obucina <obucinac@gmail.com> Co-authored-by: Roshan Syed <roshan.syed.rs@gmail.com> Co-authored-by: obucina <11353193+obucina@users.noreply.github.com> * Beatrice NFT HF Co-authored-by: pbattu123 <43043205+pbattu123@users.noreply.github.com> Co-authored-by: pbattu123 <p.battu@pbsa.info> Co-authored-by: Srdjan Obucina <obucinac@gmail.com> Co-authored-by: Roshan Syed <roshan.syed.rs@gmail.com> Co-authored-by: obucina <11353193+obucina@users.noreply.github.com>
238 lines
10 KiB
C++
238 lines
10 KiB
C++
#include <graphene/chain/nft_evaluator.hpp>
|
|
#include <graphene/chain/nft_object.hpp>
|
|
#include <graphene/chain/hardfork.hpp>
|
|
|
|
namespace graphene { namespace chain {
|
|
|
|
void_result nft_metadata_create_evaluator::do_evaluate( const nft_metadata_create_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
op.owner(db());
|
|
const auto& idx_nft_md_by_name = db().get_index_type<nft_metadata_index>().indices().get<by_name>();
|
|
FC_ASSERT( idx_nft_md_by_name.find(op.name) == idx_nft_md_by_name.end(), "NFT name already in use" );
|
|
const auto& idx_nft_md_by_symbol = db().get_index_type<nft_metadata_index>().indices().get<by_symbol>();
|
|
FC_ASSERT( idx_nft_md_by_symbol.find(op.symbol) == idx_nft_md_by_symbol.end(), "NFT symbol already in use" );
|
|
FC_ASSERT((op.revenue_partner && op.revenue_split) || (!op.revenue_partner && !op.revenue_split), "NFT revenue partner info invalid");
|
|
if (op.revenue_partner) {
|
|
(*op.revenue_partner)(db());
|
|
FC_ASSERT(*op.revenue_split >= 0 && *op.revenue_split <= GRAPHENE_100_PERCENT, "Revenue split percent invalid");
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
object_id_type nft_metadata_create_evaluator::do_apply( const nft_metadata_create_operation& op )
|
|
{ try {
|
|
const auto& new_nft_metadata_object = db().create<nft_metadata_object>( [&]( nft_metadata_object& obj ){
|
|
obj.owner = op.owner;
|
|
obj.name = op.name;
|
|
obj.symbol = op.symbol;
|
|
obj.base_uri = op.base_uri;
|
|
obj.revenue_partner = op.revenue_partner;
|
|
obj.revenue_split = op.revenue_split;
|
|
obj.is_transferable = op.is_transferable;
|
|
obj.is_sellable = op.is_sellable;
|
|
});
|
|
return new_nft_metadata_object.id;
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
|
|
void_result nft_metadata_update_evaluator::do_evaluate( const nft_metadata_update_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
op.owner(db());
|
|
const auto& idx_nft_md = db().get_index_type<nft_metadata_index>().indices().get<by_id>();
|
|
auto itr_nft_md = idx_nft_md.find(op.nft_metadata_id);
|
|
FC_ASSERT( itr_nft_md != idx_nft_md.end(), "NFT metadata not found" );
|
|
FC_ASSERT( itr_nft_md->owner == op.owner, "Only owner can modify NFT metadata" );
|
|
const auto& idx_nft_md_by_name = db().get_index_type<nft_metadata_index>().indices().get<by_name>();
|
|
const auto& idx_nft_md_by_symbol = db().get_index_type<nft_metadata_index>().indices().get<by_symbol>();
|
|
if (op.name.valid())
|
|
FC_ASSERT((itr_nft_md->name != *op.name) && (idx_nft_md_by_name.find(*op.name) == idx_nft_md_by_name.end()), "NFT name already in use");
|
|
if (op.symbol.valid())
|
|
FC_ASSERT((itr_nft_md->symbol != *op.symbol) && (idx_nft_md_by_symbol.find(*op.symbol) == idx_nft_md_by_symbol.end()), "NFT symbol already in use");
|
|
FC_ASSERT((op.revenue_partner && op.revenue_split) || (!op.revenue_partner && !op.revenue_split), "NFT revenue partner info invalid");
|
|
if (op.revenue_partner) {
|
|
(*op.revenue_partner)(db());
|
|
FC_ASSERT(*op.revenue_split >= 0 && *op.revenue_split <= GRAPHENE_100_PERCENT, "Revenue split percent invalid");
|
|
}
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
void_result nft_metadata_update_evaluator::do_apply( const nft_metadata_update_operation& op )
|
|
{ try {
|
|
db().modify(db().get(op.nft_metadata_id), [&] ( nft_metadata_object& obj ) {
|
|
if( op.name.valid() )
|
|
obj.name = *op.name;
|
|
if( op.symbol.valid() )
|
|
obj.symbol = *op.symbol;
|
|
if( op.base_uri.valid() )
|
|
obj.base_uri = *op.base_uri;
|
|
if( op.revenue_partner.valid() )
|
|
obj.revenue_partner = op.revenue_partner;
|
|
if( op.revenue_split.valid() )
|
|
obj.revenue_split = op.revenue_split;
|
|
if( op.is_transferable.valid() )
|
|
obj.is_transferable = *op.is_transferable;
|
|
if( op.is_sellable.valid() )
|
|
obj.is_sellable = *op.is_sellable;
|
|
});
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
|
|
void_result nft_mint_evaluator::do_evaluate( const nft_mint_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
op.payer(db());
|
|
op.owner(db());
|
|
op.approved(db());
|
|
for(const auto& op_iter: op.approved_operators) {
|
|
op_iter(db());
|
|
}
|
|
const auto& idx_nft_md = db().get_index_type<nft_metadata_index>().indices().get<by_id>();
|
|
auto itr_nft_md = idx_nft_md.find(op.nft_metadata_id);
|
|
FC_ASSERT( itr_nft_md != idx_nft_md.end(), "NFT metadata not found" );
|
|
FC_ASSERT( itr_nft_md->owner == op.payer, "Only metadata owner can mint NFT" );
|
|
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
object_id_type nft_mint_evaluator::do_apply( const nft_mint_operation& op )
|
|
{ try {
|
|
const auto& new_nft_object = db().create<nft_object>( [&]( nft_object& obj ){
|
|
obj.nft_metadata_id = op.nft_metadata_id;
|
|
obj.owner = op.owner;
|
|
obj.approved = op.approved;
|
|
obj.approved_operators = op.approved_operators;
|
|
obj.token_uri = op.token_uri;
|
|
});
|
|
return new_nft_object.id;
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
|
|
void_result nft_safe_transfer_from_evaluator::do_evaluate( const nft_safe_transfer_from_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
const auto& idx_nft = db().get_index_type<nft_index>().indices().get<by_id>();
|
|
const auto& idx_acc = db().get_index_type<account_index>().indices().get<by_id>();
|
|
|
|
auto itr_nft = idx_nft.find(op.token_id);
|
|
FC_ASSERT( itr_nft != idx_nft.end(), "NFT does not exists" );
|
|
|
|
FC_ASSERT(!db().item_locked(op.token_id), "Item(s) is already on sale on market, transfer is not allowed");
|
|
|
|
auto itr_operator = idx_acc.find(op.operator_);
|
|
FC_ASSERT( itr_operator != idx_acc.end(), "Operator account does not exists" );
|
|
|
|
auto itr_owner = idx_acc.find(itr_nft->owner);
|
|
FC_ASSERT( itr_owner != idx_acc.end(), "Owner account does not exists" );
|
|
|
|
auto itr_from = idx_acc.find(op.from);
|
|
FC_ASSERT( itr_from != idx_acc.end(), "Sender account does not exists" );
|
|
FC_ASSERT( itr_from->id == itr_owner->id, "Sender account is not owner of this NFT" );
|
|
|
|
auto itr_to = idx_acc.find(op.to);
|
|
FC_ASSERT( itr_to != idx_acc.end(), "Receiver account does not exists" );
|
|
|
|
auto itr_approved_op = std::find(itr_nft->approved_operators.begin(), itr_nft->approved_operators.end(), op.operator_);
|
|
FC_ASSERT( (itr_nft->owner == op.operator_) || (itr_nft->approved == itr_operator->id) || (itr_approved_op != itr_nft->approved_operators.end()), "Operator is not NFT owner or approved operator" );
|
|
|
|
const auto& nft_meta_obj = itr_nft->nft_metadata_id(db());
|
|
FC_ASSERT( nft_meta_obj.is_transferable == true, "NFT is not transferable");
|
|
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
object_id_type nft_safe_transfer_from_evaluator::do_apply( const nft_safe_transfer_from_operation& op )
|
|
{ try {
|
|
const auto& idx = db().get_index_type<nft_index>().indices().get<by_id>();
|
|
auto itr = idx.find(op.token_id);
|
|
if (itr != idx.end())
|
|
{
|
|
db().modify(*itr, [&op](nft_object &obj) {
|
|
obj.owner = op.to;
|
|
obj.approved = {};
|
|
obj.approved_operators.clear();
|
|
});
|
|
}
|
|
return op.token_id;
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
|
|
void_result nft_approve_evaluator::do_evaluate( const nft_approve_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
const auto& idx_nft = db().get_index_type<nft_index>().indices().get<by_id>();
|
|
const auto& idx_acc = db().get_index_type<account_index>().indices().get<by_id>();
|
|
|
|
auto itr_nft = idx_nft.find(op.token_id);
|
|
FC_ASSERT( itr_nft != idx_nft.end(), "NFT does not exists" );
|
|
|
|
auto itr_owner = idx_acc.find(op.operator_);
|
|
FC_ASSERT( itr_owner != idx_acc.end(), "Owner account does not exists" );
|
|
|
|
auto itr_approved = idx_acc.find(op.approved);
|
|
FC_ASSERT( itr_approved != idx_acc.end(), "Approved account does not exists" );
|
|
|
|
auto itr_approved_op = std::find(itr_nft->approved_operators.begin(), itr_nft->approved_operators.end(), op.operator_);
|
|
FC_ASSERT( (itr_nft->owner == itr_owner->id) || (itr_approved_op != itr_nft->approved_operators.end()), "Sender is not NFT owner or approved operator" );
|
|
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
object_id_type nft_approve_evaluator::do_apply( const nft_approve_operation& op )
|
|
{ try {
|
|
const auto& idx = db().get_index_type<nft_index>().indices().get<by_id>();
|
|
auto itr = idx.find(op.token_id);
|
|
if (itr != idx.end())
|
|
{
|
|
db().modify(*itr, [&op](nft_object &obj) {
|
|
obj.approved = op.approved;
|
|
//auto itr = std::find(obj.approved_operators.begin(), obj.approved_operators.end(), op.approved);
|
|
//if (itr == obj.approved_operators.end()) {
|
|
// obj.approved_operators.push_back(op.approved);
|
|
//}
|
|
});
|
|
}
|
|
return op.token_id;
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
|
|
void_result nft_set_approval_for_all_evaluator::do_evaluate( const nft_set_approval_for_all_operation& op )
|
|
{ try {
|
|
auto now = db().head_block_time();
|
|
FC_ASSERT(now >= HARDFORK_NFT_TIME, "Not allowed until NFT HF");
|
|
op.owner(db());
|
|
const auto& idx_acc = db().get_index_type<account_index>().indices().get<by_id>();
|
|
|
|
auto itr_operator = idx_acc.find(op.operator_);
|
|
FC_ASSERT( itr_operator != idx_acc.end(), "Operator account does not exists" );
|
|
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
void_result nft_set_approval_for_all_evaluator::do_apply( const nft_set_approval_for_all_operation& op )
|
|
{ try {
|
|
const auto &idx = db().get_index_type<nft_index>().indices().get<by_owner>();
|
|
const auto &idx_range = idx.equal_range(op.owner);
|
|
std::for_each(idx_range.first, idx_range.second, [&](const nft_object &obj) {
|
|
db().modify(obj, [&op](nft_object &obj) {
|
|
auto itr = std::find(obj.approved_operators.begin(), obj.approved_operators.end(), op.operator_);
|
|
if ((op.approved) && (itr == obj.approved_operators.end())) {
|
|
obj.approved_operators.push_back(op.operator_);
|
|
}
|
|
if ((!op.approved) && (itr != obj.approved_operators.end())) {
|
|
obj.approved_operators.erase(itr);
|
|
}
|
|
});
|
|
});
|
|
return void_result();
|
|
} FC_CAPTURE_AND_RETHROW( (op) ) }
|
|
|
|
} } // graphene::chain
|
|
|