Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(near-contract-standards): NEP-199 - Non-Fungible Token Royalties and Payouts #1077

Draft
wants to merge 50 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 35 commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
8a1c2da
payout
ymc182 Apr 4, 2022
da97136
d
ymc182 Mar 26, 2022
34c576f
included the sim
ymc182 Apr 4, 2022
7b58745
sim
ymc182 Apr 4, 2022
ca930d7
remove macros, add examples and docs
ruseinov Aug 20, 2023
0406030
add royalties storage prefix
ruseinov Aug 20, 2023
bd9c9fe
Merge branch 'master' into feat/payout
ruseinov Aug 20, 2023
c8bd137
examples
ruseinov Aug 20, 2023
1e62834
Merge branch 'master' into feat/payout
ruseinov Aug 25, 2023
8e5e777
remove near-sdk-sim
ruseinov Sep 10, 2023
14ac453
Merge branch 'master' into feat/payout
frol Sep 12, 2023
5da6f89
Merge branch 'master' into feat/payout
frol Sep 22, 2023
9470437
Merge branch 'master' into feat/payout
ruseinov Sep 26, 2023
98aeceb
update CHANGELOG.md
ruseinov Sep 26, 2023
3d7a4ad
fix test
ruseinov Sep 26, 2023
562b284
fixes and tests
ruseinov Sep 26, 2023
933c029
more tests
ruseinov Sep 26, 2023
2f2779d
better comment
ruseinov Sep 26, 2023
a5c67f7
Merge branch 'master' into feat/payout
ruseinov Sep 28, 2023
6f07b52
return the missing check
ruseinov Sep 29, 2023
e4da342
fix ci
ruseinov Oct 1, 2023
743eb06
Merge branch 'master' into feat/payout
ruseinov Oct 2, 2023
b3716bf
Merge branch 'master' into feat/payout
ruseinov Oct 7, 2023
d76ff5d
fix review comments
ruseinov Oct 7, 2023
f3b3594
use TreeMap for rolayties storage
ruseinov Oct 7, 2023
815a9ea
apply review suggestions
ruseinov Oct 13, 2023
d809519
Merge branch 'master' into feat/payout
ruseinov Oct 13, 2023
c2e990e
fixes
ruseinov Oct 13, 2023
c3d8b67
fixes
ruseinov Oct 13, 2023
ee59a72
fixes
ruseinov Oct 14, 2023
5ec54b3
fix
ruseinov Oct 14, 2023
71dc47b
Merge branch 'master' into feat/payout
ruseinov Oct 24, 2023
b7a0a22
add tests
ruseinov Oct 22, 2023
5c20528
fix lint
ruseinov Oct 24, 2023
2db048b
fix nits
ruseinov Oct 28, 2023
279a7ee
simplify royalties and more tests
ruseinov Nov 3, 2023
369faaa
add comment
ruseinov Nov 3, 2023
e080566
Merge branch 'master' into feat/payout
ruseinov Nov 3, 2023
9e01a09
validate apply_percent
ruseinov Nov 3, 2023
423bdc9
fix
ruseinov Nov 3, 2023
30175c3
Merge branch 'master' into feat/payout
ruseinov Nov 9, 2023
5829c77
fix tests
ruseinov Nov 10, 2023
b59fc92
Merge branch 'master' into feat/payout
ruseinov Nov 24, 2023
312f4ac
fix payouts
ruseinov Nov 24, 2023
f2e709c
fix NearToken
ruseinov Nov 24, 2023
f2e3e08
Merge branch 'master' into feat/payout
ruseinov Dec 11, 2023
66ccf2c
fix workflow name
ruseinov Dec 12, 2023
1fa86f7
fix
ruseinov Dec 12, 2023
5a574b4
fix compilation
ruseinov Dec 13, 2023
7745888
Merge branch 'master' into feat/payout
ruseinov Jan 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@

## [Unreleased]

### Added
- **BREAKING** Added [NEP-199 - Non-Fungible Token Royalties and Payouts](https://github.com/willemneal/NEPs/blob/18828873648eff1a2e8464db234aefd70918b3e0/neps/nep-0199.md) implementation. This
will need a state migration in case of upgrade for any contracts using NFT of previous versions due to an extra field.

### Fixed
- Exposed missing iterator types used in `near_sdk::store::UnorderedSet`. [PR 961](https://github.com/near/near-sdk-rs/pull/961)

Expand Down
178 changes: 164 additions & 14 deletions examples/non-fungible-token/nft/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,21 +15,25 @@ NOTES:
- To prevent the deployed contract from being modified or deleted, it should not have any access
keys on its account.
*/
use std::collections::HashMap;

use near_contract_standards::non_fungible_token::approval::NonFungibleTokenApproval;
use near_contract_standards::non_fungible_token::core::{
NonFungibleTokenCore, NonFungibleTokenResolver,
};
use near_contract_standards::non_fungible_token::enumeration::NonFungibleTokenEnumeration;
use near_contract_standards::non_fungible_token::metadata::{
NFTContractMetadata, NonFungibleTokenMetadataProvider, TokenMetadata, NFT_METADATA_SPEC,
};
use near_contract_standards::non_fungible_token::NonFungibleToken;
use near_contract_standards::non_fungible_token::payout::Payout;
use near_contract_standards::non_fungible_token::{NonFungibleToken, NonFungibleTokenPayout};
use near_contract_standards::non_fungible_token::{Token, TokenId};
use near_contract_standards::non_fungible_token::approval::NonFungibleTokenApproval;
use near_contract_standards::non_fungible_token::core::{NonFungibleTokenCore, NonFungibleTokenResolver};
use near_contract_standards::non_fungible_token::enumeration::NonFungibleTokenEnumeration;
use near_sdk::borsh::{BorshDeserialize, BorshSerialize};
use near_sdk::collections::LazyOption;
use near_sdk::json_types::U128;
use near_sdk::{
env, near_bindgen, require, AccountId, BorshStorageKey, PanicOnDefault, Promise, PromiseOrValue,
};
use near_sdk::json_types::U128;
use std::collections::HashMap;

#[near_bindgen]
#[derive(BorshDeserialize, BorshSerialize, PanicOnDefault)]
Expand All @@ -49,6 +53,7 @@ enum StorageKey {
TokenMetadata,
Enumeration,
Approval,
Royalties,
}

#[near_bindgen]
Expand Down Expand Up @@ -82,6 +87,7 @@ impl Contract {
Some(StorageKey::TokenMetadata),
Some(StorageKey::Enumeration),
Some(StorageKey::Approval),
Some(StorageKey::Royalties),
),
metadata: LazyOption::new(StorageKey::Metadata, Some(&metadata)),
}
Expand Down Expand Up @@ -110,12 +116,25 @@ impl Contract {
#[near_bindgen]
impl NonFungibleTokenCore for Contract {
#[payable]
fn nft_transfer(&mut self, receiver_id: AccountId, token_id: TokenId, approval_id: Option<u64>, memo: Option<String>) {
fn nft_transfer(
&mut self,
receiver_id: AccountId,
token_id: TokenId,
approval_id: Option<u64>,
memo: Option<String>,
) {
self.tokens.nft_transfer(receiver_id, token_id, approval_id, memo);
}

#[payable]
fn nft_transfer_call(&mut self, receiver_id: AccountId, token_id: TokenId, approval_id: Option<u64>, memo: Option<String>, msg: String) -> PromiseOrValue<bool> {
fn nft_transfer_call(
&mut self,
receiver_id: AccountId,
token_id: TokenId,
approval_id: Option<u64>,
memo: Option<String>,
msg: String,
) -> PromiseOrValue<bool> {
self.tokens.nft_transfer_call(receiver_id, token_id, approval_id, memo, msg)
}

Expand All @@ -127,15 +146,31 @@ impl NonFungibleTokenCore for Contract {
#[near_bindgen]
impl NonFungibleTokenResolver for Contract {
#[private]
fn nft_resolve_transfer(&mut self, previous_owner_id: AccountId, receiver_id: AccountId, token_id: TokenId, approved_account_ids: Option<HashMap<AccountId, u64>>) -> bool {
self.tokens.nft_resolve_transfer(previous_owner_id, receiver_id, token_id, approved_account_ids)
fn nft_resolve_transfer(
&mut self,
previous_owner_id: AccountId,
receiver_id: AccountId,
token_id: TokenId,
approved_account_ids: Option<HashMap<AccountId, u64>>,
) -> bool {
self.tokens.nft_resolve_transfer(
previous_owner_id,
receiver_id,
token_id,
approved_account_ids,
)
}
}

#[near_bindgen]
impl NonFungibleTokenApproval for Contract {
#[payable]
fn nft_approve(&mut self, token_id: TokenId, account_id: AccountId, msg: Option<String>) -> Option<Promise> {
fn nft_approve(
&mut self,
token_id: TokenId,
account_id: AccountId,
msg: Option<String>,
) -> Option<Promise> {
self.tokens.nft_approve(token_id, account_id, msg)
}

Expand All @@ -147,10 +182,14 @@ impl NonFungibleTokenApproval for Contract {
#[payable]
fn nft_revoke_all(&mut self, token_id: TokenId) {
self.tokens.nft_revoke_all(token_id);

}

fn nft_is_approved(&self, token_id: TokenId, approved_account_id: AccountId, approval_id: Option<u64>) -> bool {
fn nft_is_approved(
&self,
token_id: TokenId,
approved_account_id: AccountId,
approval_id: Option<u64>,
) -> bool {
self.tokens.nft_is_approved(token_id, approved_account_id, approval_id)
}
}
Expand All @@ -169,11 +208,44 @@ impl NonFungibleTokenEnumeration for Contract {
self.tokens.nft_supply_for_owner(account_id)
}

fn nft_tokens_for_owner(&self, account_id: AccountId, from_index: Option<U128>, limit: Option<u64>) -> Vec<Token> {
fn nft_tokens_for_owner(
&self,
account_id: AccountId,
from_index: Option<U128>,
limit: Option<u64>,
) -> Vec<Token> {
self.tokens.nft_tokens_for_owner(account_id, from_index, limit)
}
}

#[near_bindgen]
impl NonFungibleTokenPayout for Contract {
#[allow(unused_variables)]
fn nft_payout(&self, token_id: String, balance: U128, max_len_payout: Option<u32>) -> Payout {
self.tokens.nft_payout(token_id, balance, max_len_payout)
}

#[payable]
fn nft_transfer_payout(
&mut self,
receiver_id: AccountId,
token_id: String,
approval_id: Option<u64>,
memo: Option<String>,
balance: U128,
max_len_payout: Option<u32>,
) -> Payout {
self.tokens.nft_transfer_payout(
receiver_id,
token_id,
approval_id,
memo,
balance,
max_len_payout,
)
}
}

#[near_bindgen]
impl NonFungibleTokenMetadataProvider for Contract {
fn nft_metadata(&self) -> NFTContractMetadata {
Expand Down Expand Up @@ -291,6 +363,84 @@ mod tests {
}
}

#[test]
fn test_transfer_payout() {
let mut context = get_context(accounts(0));
testing_env!(context.build());
let mut contract = Contract::new_default_meta(accounts(0).into());

testing_env!(context
.storage_usage(env::storage_usage())
.attached_deposit(MINT_STORAGE_COST)
.predecessor_account_id(accounts(0))
.build());
let token_id = "0".to_string();
contract.nft_mint(token_id.clone(), accounts(0), sample_token_metadata());

testing_env!(context
.storage_usage(env::storage_usage())
.attached_deposit(1)
.predecessor_account_id(accounts(0))
.build());

let payout = contract.nft_transfer_payout(
accounts(1),
token_id.clone(),
None,
None,
U128::from(1),
None,
);

let mut expected_payout = HashMap::new();
expected_payout.insert(accounts(0), U128::from(1));

assert_eq!(payout.payout, expected_payout);

testing_env!(context
.storage_usage(env::storage_usage())
.account_balance(env::account_balance())
.is_view(true)
.attached_deposit(0)
.build());

if let Some(token) = contract.nft_token(token_id.clone()) {
assert_eq!(token.token_id, token_id);
assert_eq!(token.owner_id, accounts(1));
assert_eq!(token.metadata.unwrap(), sample_token_metadata());
assert_eq!(token.approved_account_ids.unwrap(), HashMap::new());
} else {
panic!("token not correctly created, or not found by nft_token");
}
}

#[test]
fn test_payout() {
let mut context = get_context(accounts(0));
testing_env!(context.build());
let mut contract = Contract::new_default_meta(accounts(0).into());

testing_env!(context
.storage_usage(env::storage_usage())
.attached_deposit(MINT_STORAGE_COST)
.predecessor_account_id(accounts(0))
.build());
let token_id = "0".to_string();
contract.nft_mint(token_id.clone(), accounts(0), sample_token_metadata());

testing_env!(context
.storage_usage(env::storage_usage())
.predecessor_account_id(accounts(0))
.build());

let payout = contract.nft_payout(token_id, U128::from(1), None);

let mut expected_payout = HashMap::new();
expected_payout.insert(accounts(0), U128::from(1));

assert_eq!(payout.payout, expected_payout);
}

#[test]
fn test_approve() {
let mut context = get_context(accounts(0));
Expand Down
1 change: 1 addition & 0 deletions examples/non-fungible-token/tests/workspaces/main.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
mod test_approval;
mod test_core;
mod test_enumeration;
mod test_payout;
mod utils;
16 changes: 4 additions & 12 deletions examples/non-fungible-token/tests/workspaces/test_approval.rs
Original file line number Diff line number Diff line change
Expand Up @@ -50,12 +50,8 @@ async fn simulate_simple_approve() -> anyhow::Result<()> {
assert!(!alice_approval_id_is_2);

// alternatively, one could check the data returned by nft_token
let token = nft_contract
.call("nft_token")
.args_json((TOKEN_ID,))
.view()
.await?
.json::<Token>()?;
let token =
nft_contract.call("nft_token").args_json((TOKEN_ID,)).view().await?.json::<Token>()?;
let mut expected_approvals: HashMap<AccountId, u64> = HashMap::new();
expected_approvals.insert(AccountId::try_from(alice.id().to_string())?, 1);
assert_eq!(token.approved_account_ids.unwrap(), expected_approvals);
Expand Down Expand Up @@ -157,12 +153,8 @@ async fn simulate_approved_account_transfers_token() -> anyhow::Result<()> {
assert!(res.is_success());

// token now owned by alice
let token = nft_contract
.call("nft_token")
.args_json((TOKEN_ID,))
.view()
.await?
.json::<Token>()?;
let token =
nft_contract.call("nft_token").args_json((TOKEN_ID,)).view().await?.json::<Token>()?;
assert_eq!(token.owner_id.to_string(), alice.id().to_string());

Ok(())
Expand Down
Loading
Loading