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(nfts): AccountBalance deposit #413

Merged
merged 45 commits into from
Jan 9, 2025
Merged
Show file tree
Hide file tree
Changes from 44 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
8a341d9
refactor: account balance deposit
evilrobot-01 Dec 18, 2024
e391888
chore: revert test changes
chungquantin Dec 18, 2024
31368ca
refactor: account balance deposit & tests
chungquantin Dec 19, 2024
3f9e07e
refactor: account balance deposit & tests
chungquantin Dec 19, 2024
99b31d8
Merge branch 'chungquantin/feat-nfts' into frank/nfts-balance-deposit
chungquantin Dec 19, 2024
5ecb661
chore: benchmarking
chungquantin Dec 19, 2024
ee364fb
fix: clippy warning
chungquantin Dec 19, 2024
5015e90
chore: comments
chungquantin Dec 19, 2024
94f0c74
chore: resolve comments (exclude weights)
chungquantin Dec 21, 2024
d3ae740
chore: resolve review comments
chungquantin Dec 22, 2024
ed6d6a2
fix: deposit bytes
chungquantin Dec 22, 2024
bda3e47
chore: update test & weights
chungquantin Dec 22, 2024
460859f
chore: benchmarking
chungquantin Dec 23, 2024
663066e
chore: update comments
chungquantin Dec 23, 2024
d975170
chore: reorder type
chungquantin Dec 23, 2024
f316e2f
chore: resolve comments
chungquantin Dec 26, 2024
28159d0
chore: comments
chungquantin Dec 26, 2024
911c5a9
chore: use balance deposit variable
chungquantin Dec 26, 2024
a16ecdc
chore: update comments
chungquantin Dec 26, 2024
02db7a5
chore: apply suggestion
chungquantin Dec 26, 2024
78a3815
chore: apply review suggestions
chungquantin Dec 30, 2024
26b43f3
chore: revert back test case order
chungquantin Dec 30, 2024
8210e9c
chore: format
chungquantin Dec 30, 2024
ff8d560
chore: remove redudant checks
chungquantin Dec 30, 2024
569aae7
chore: reorder
chungquantin Dec 30, 2024
439ad31
Merge branch 'chungquantin/feat-nfts' into frank/nfts-balance-deposit
chungquantin Dec 30, 2024
40a8bd8
chore: rebase
chungquantin Dec 30, 2024
3d1e367
chore: update tests
chungquantin Dec 30, 2024
b6c2529
chore: apply suggestion
chungquantin Dec 30, 2024
e1a8c4c
chore: test
chungquantin Dec 30, 2024
e2aedba
chore: update tests
chungquantin Dec 31, 2024
1a805f5
chore: update tests
chungquantin Dec 31, 2024
c76d469
Merge branch 'chungquantin/feat-nfts' into frank/nfts-balance-deposit
chungquantin Dec 31, 2024
d3b3141
chore: reformat
chungquantin Dec 31, 2024
7820d79
chore: update tests
chungquantin Dec 31, 2024
df965e1
chore: update tests
chungquantin Dec 31, 2024
6abe359
Merge branch 'chungquantin/feat-nfts' into frank/nfts-balance-deposit
chungquantin Dec 31, 2024
1159074
chore: format
chungquantin Dec 31, 2024
c5fd9e3
chore: resolve review comments
chungquantin Jan 6, 2025
af0b71e
fix: account balance logic (#418)
chungquantin Jan 7, 2025
270039e
chore: update benchmarking.rs
chungquantin Jan 7, 2025
b1719a6
chore: update tests
chungquantin Jan 7, 2025
e4020db
Merge branch 'chungquantin/feat-nfts' into frank/nfts-balance-deposit
chungquantin Jan 7, 2025
986a5b6
refactor: account balance (#421)
Daanvdplas Jan 8, 2025
1aa79f3
chore: resolve feedback comments
chungquantin Jan 9, 2025
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
6 changes: 4 additions & 2 deletions pallets/nfts/src/benchmarking.rs
Original file line number Diff line number Diff line change
Expand Up @@ -274,6 +274,7 @@ benchmarks_instance_pallet! {
mint {
let (collection, caller, caller_lookup) = create_collection::<T, I>();
let item = T::Helper::item(0);
T::Currency::make_free_balance_be(&caller, T::Currency::minimum_balance() + T::CollectionDeposit::get() + T::CollectionBalanceDeposit::get());
}: _(SystemOrigin::Signed(caller.clone()), collection, item, caller_lookup, None)
verify {
assert_last_event::<T, I>(Event::Issued { collection, item, owner: caller }.into());
Expand All @@ -282,6 +283,7 @@ benchmarks_instance_pallet! {
force_mint {
let (collection, caller, caller_lookup) = create_collection::<T, I>();
let item = T::Helper::item(0);
T::Currency::make_free_balance_be(&caller, T::Currency::minimum_balance() + T::CollectionDeposit::get() + T::CollectionBalanceDeposit::get());
}: _(SystemOrigin::Signed(caller.clone()), collection, item, caller_lookup, default_item_config())
verify {
assert_last_event::<T, I>(Event::Issued { collection, item, owner: caller }.into());
Expand Down Expand Up @@ -667,7 +669,7 @@ benchmarks_instance_pallet! {
let price = ItemPrice::<T, I>::from(0u32);
let origin = SystemOrigin::Signed(seller.clone()).into();
Nfts::<T, I>::set_price(origin, collection, item, Some(price), Some(buyer_lookup))?;
T::Currency::make_free_balance_be(&buyer, DepositBalanceOf::<T, I>::max_value());
T::Currency::make_free_balance_be(&buyer, T::Currency::minimum_balance() + price + T::CollectionBalanceDeposit::get());
}: _(SystemOrigin::Signed(buyer.clone()), collection, item, price)
verify {
assert_last_event::<T, I>(Event::ItemBought {
Expand Down Expand Up @@ -757,7 +759,7 @@ benchmarks_instance_pallet! {
let duration = T::MaxDeadlineDuration::get();
let target: T::AccountId = account("target", 0, SEED);
let target_lookup = T::Lookup::unlookup(target.clone());
T::Currency::make_free_balance_be(&target, T::Currency::minimum_balance());
T::Currency::make_free_balance_be(&target, T::Currency::minimum_balance() + T::CollectionBalanceDeposit::get());
let origin = SystemOrigin::Signed(caller.clone());
frame_system::Pallet::<T>::set_block_number(One::one());
Nfts::<T, I>::transfer(origin.clone().into(), collection, item2, target_lookup)?;
Expand Down
132 changes: 131 additions & 1 deletion pallets/nfts/src/common_functions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@

use alloc::vec::Vec;

use frame_support::pallet_prelude::*;
use frame_support::{pallet_prelude::*, sp_runtime::ArithmeticError};

use crate::*;

Expand Down Expand Up @@ -92,6 +92,52 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
Self::deposit_event(Event::NextCollectionIdIncremented { next_id });
}

/// Increment the number of items in the `collection` owned by the `owner`. If no entry exists
/// for the `owner` in `AccountBalance`, create a new record and reserve `deposit_amount` from
/// the `deposit_account`.
pub(crate) fn increment_account_balance(
collection: T::CollectionId,
owner: &T::AccountId,
(deposit_account, deposit_amount): (&T::AccountId, DepositBalanceOf<T, I>),
) -> DispatchResult {
AccountBalance::<T, I>::mutate(collection, owner, |maybe_balance| -> DispatchResult {
match maybe_balance {
None => {
T::Currency::reserve(deposit_account, deposit_amount)?;
*maybe_balance = Some((1, (deposit_account.clone(), deposit_amount)));
},
Some((balance, _deposit)) => {
balance.saturating_inc();
},
}
Ok(())
})
}

/// Decrement the number of `collection` items owned by the `owner`. If the `owner`'s item
/// count reaches zero after the reduction, remove the `AccountBalance` record and unreserve
/// the deposited funds.
pub(crate) fn decrement_account_balance(
collection: T::CollectionId,
owner: &T::AccountId,
) -> DispatchResult {
AccountBalance::<T, I>::try_mutate_exists(
collection,
owner,
|maybe_balance| -> DispatchResult {
let (balance, (deposit_account, deposit_amount)) =
maybe_balance.as_mut().ok_or(Error::<T, I>::NoItemOwned)?;

*balance = balance.checked_sub(1).ok_or(ArithmeticError::Underflow)?;
if balance.is_zero() {
T::Currency::unreserve(deposit_account, *deposit_amount);
*maybe_balance = None;
}
Ok(())
},
)
}

#[allow(missing_docs)]
#[cfg(any(test, feature = "runtime-benchmarks"))]
pub fn set_next_id(id: T::CollectionId) {
Expand All @@ -105,3 +151,87 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
.expect("Failed to get next collection ID")
}
}

#[cfg(test)]
mod tests {
use crate::{mock::*, tests::*, Currency, Error, ReservableCurrency};

#[test]
fn increment_account_balance_works() {
new_test_ext().execute_with(|| {
let collection_id = 0;
let deposit_account = account(1);
let deposit_amount = balance_deposit();
let owner = account(2);
assert_noop!(
Nfts::increment_account_balance(
collection_id,
&owner,
(&deposit_account, deposit_amount)
),
BalancesError::<Test>::InsufficientBalance
);
Balances::make_free_balance_be(&deposit_account, 100);
// Initialize `AccountBalance` and increase the collection item count for the new
// account.
assert_ok!(Nfts::increment_account_balance(
collection_id,
&owner,
(&deposit_account, deposit_amount)
));
assert_eq!(Balances::reserved_balance(&deposit_account), deposit_amount);
assert_eq!(
AccountBalance::get(collection_id, &owner),
Some((1, (deposit_account.clone(), deposit_amount)))
);
// Increment the balance of a non-zero balance account. No additional reserves.
assert_ok!(Nfts::increment_account_balance(
collection_id,
&owner,
(&deposit_account, deposit_amount)
));
assert_eq!(Balances::reserved_balance(&deposit_account), deposit_amount);
assert_eq!(
AccountBalance::get(collection_id, &owner),
Some((2, (deposit_account.clone(), deposit_amount)))
);
});
}

#[test]
fn decrement_account_balance_works() {
new_test_ext().execute_with(|| {
let collection_id = 0;
let balance = 2u32;
let deposit_account = account(1);
let deposit_amount = balance_deposit();
let owner = account(2);

Balances::make_free_balance_be(&deposit_account, 100);
// Decrement non-existing `AccountBalance` record.
assert_noop!(
Nfts::decrement_account_balance(collection_id, &deposit_account),
Error::<Test>::NoItemOwned
);
// Set account balance and reserve `DepositBalance`.
AccountBalance::insert(
collection_id,
&owner,
(&balance, (&deposit_account, deposit_amount)),
);
Balances::reserve(&deposit_account, deposit_amount).expect("should work");
// Successfully decrement the value of the `AccountBalance` entry.
assert_ok!(Nfts::decrement_account_balance(collection_id, &owner));
assert_eq!(
AccountBalance::get(collection_id, &owner),
Some((balance - 1, (deposit_account.clone(), deposit_amount)))
);
assert_eq!(Balances::reserved_balance(&deposit_account), deposit_amount);
// `AccountBalance` record is deleted, and the depositor's funds are unreserved if
// the `AccountBalance` value reaches zero after decrementing.
assert_ok!(Nfts::decrement_account_balance(collection_id, &owner));
assert_eq!(Balances::reserved_balance(&deposit_account), 0);
assert!(!AccountBalance::contains_key(collection_id, &owner));
});
}
}
7 changes: 6 additions & 1 deletion pallets/nfts/src/features/approvals.rs
Original file line number Diff line number Diff line change
Expand Up @@ -216,7 +216,12 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
Self::is_pallet_feature_enabled(PalletFeature::Approvals),
Error::<T, I>::MethodDisabled
);
ensure!(AccountBalance::<T, I>::get(collection, &owner) > 0, Error::<T, I>::NoItemOwned);
ensure!(
chungquantin marked this conversation as resolved.
Show resolved Hide resolved
AccountBalance::<T, I>::get(collection, &owner)
.filter(|(balance, _)| !balance.is_zero())
.is_some(),
Error::<T, I>::NoItemOwned
);

let collection_config = Self::get_collection_config(&collection)?;
ensure!(
Expand Down
13 changes: 10 additions & 3 deletions pallets/nfts/src/features/atomic_swap.rs
Original file line number Diff line number Diff line change
Expand Up @@ -209,10 +209,17 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
}

// This also removes the swap.
Self::do_transfer(send_collection_id, send_item_id, receive_item.owner.clone(), |_, _| {
Ok(())
})?;
Self::do_transfer(
Some(&receive_item.owner),
send_collection_id,
send_item_id,
receive_item.owner.clone(),
|_, _| Ok(()),
)?;
// Owner of `send_item` is responsible for the deposit if the collection balance
// went to zero due to the previous transfer.
Self::do_transfer(
Some(&send_item.owner),
receive_collection_id,
receive_item_id,
send_item.owner.clone(),
Expand Down
2 changes: 1 addition & 1 deletion pallets/nfts/src/features/buy_sell.rs
Original file line number Diff line number Diff line change
Expand Up @@ -158,7 +158,7 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {

let old_owner = details.owner.clone();

Self::do_transfer(collection, item, buyer.clone(), |_, _| Ok(()))?;
Self::do_transfer(Some(&buyer), collection, item, buyer.clone(), |_, _| Ok(()))?;

Self::deposit_event(Event::ItemBought {
collection,
Expand Down
41 changes: 19 additions & 22 deletions pallets/nfts/src/features/create_delete_item.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
//! This module contains helper methods to perform functionality associated with minting and burning
//! items for the NFTs pallet.

use frame_support::{pallet_prelude::*, sp_runtime::ArithmeticError, traits::ExistenceRequirement};
use frame_support::{pallet_prelude::*, traits::ExistenceRequirement};

use crate::*;

Expand Down Expand Up @@ -68,20 +68,20 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {

collection_details.items.saturating_inc();

AccountBalance::<T, I>::mutate(collection, &mint_to, |balance| {
balance.saturating_inc();
});

let collection_config = Self::get_collection_config(&collection)?;
let deposit_amount =
match collection_config.is_setting_enabled(CollectionSetting::DepositRequired) {
true => T::ItemDeposit::get(),
false => Zero::zero(),
};
let deposit_account = match maybe_depositor {
None => collection_details.owner.clone(),
Some(depositor) => depositor,
};
let deposit_required =
collection_config.is_setting_enabled(CollectionSetting::DepositRequired);
let deposit_account =
maybe_depositor.unwrap_or_else(|| collection_details.owner.clone());

let balance_deposit = deposit_required
.then_some(T::CollectionBalanceDeposit::get())
.unwrap_or_default();
Self::increment_account_balance(
chungquantin marked this conversation as resolved.
Show resolved Hide resolved
collection,
&mint_to,
(&deposit_account, balance_deposit),
)?;

let item_owner = mint_to.clone();
Account::<T, I>::insert((&item_owner, &collection, &item), ());
Expand All @@ -93,9 +93,11 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
collection_details.item_configs.saturating_inc();
}

T::Currency::reserve(&deposit_account, deposit_amount)?;
let item_deposit =
deposit_required.then_some(T::ItemDeposit::get()).unwrap_or_default();
T::Currency::reserve(&deposit_account, item_deposit)?;

let deposit = ItemDeposit { account: deposit_account, amount: deposit_amount };
let deposit = ItemDeposit { account: deposit_account, amount: item_deposit };
let details = ItemDetails {
owner: item_owner,
approvals: ApprovalsOf::<T, I>::default(),
Expand Down Expand Up @@ -264,12 +266,7 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
PendingSwapOf::<T, I>::remove(collection, item);
ItemAttributesApprovalsOf::<T, I>::remove(collection, item);

let balance = AccountBalance::<T, I>::take(collection, &owner)
.checked_sub(1)
.ok_or(ArithmeticError::Underflow)?;
if balance > 0 {
AccountBalance::<T, I>::insert(collection, &owner, balance);
}
Self::decrement_account_balance(collection, &owner)?;

if remove_config {
ItemConfigOf::<T, I>::remove(collection, item);
Expand Down
23 changes: 14 additions & 9 deletions pallets/nfts/src/features/transfer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ use crate::*;
impl<T: Config<I>, I: 'static> Pallet<T, I> {
/// Transfer an NFT to the specified destination account.
///
/// - `depositor`: The account reserving the `CollectionBalanceDeposit` from if `dest` holds no
/// items in the collection.
/// - `collection`: The ID of the collection to which the NFT belongs.
/// - `item`: The ID of the NFT to transfer.
/// - `dest`: The destination account to which the NFT will be transferred.
Expand All @@ -45,6 +47,7 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
/// - If the collection or item is non-transferable
/// ([`ItemsNonTransferable`](crate::Error::ItemsNonTransferable)).
pub fn do_transfer(
depositor: Option<&T::AccountId>,
chungquantin marked this conversation as resolved.
Show resolved Hide resolved
collection: T::CollectionId,
item: T::ItemId,
dest: T::AccountId,
Expand Down Expand Up @@ -87,16 +90,18 @@ impl<T: Config<I>, I: 'static> Pallet<T, I> {
with_details(&collection_details, &mut details)?;

// Update account balance of the owner.
let balance = AccountBalance::<T, I>::take(collection, &details.owner)
.checked_sub(1)
.ok_or(Error::<T, I>::NoItemOwned)?;
if balance > 0 {
AccountBalance::<T, I>::insert(collection, &details.owner, balance);
}
Self::decrement_account_balance(collection, &details.owner)?;

let deposit_amount = collection_config
.is_setting_enabled(CollectionSetting::DepositRequired)
.then_some(T::CollectionBalanceDeposit::get())
.unwrap_or_default();
// Reserve `CollectionBalanceDeposit` from the depositor if provided. Otherwise, reserve
// from the item's owner.
let deposit_account = depositor.unwrap_or(&details.owner);

// Update account balance of the destination account.
AccountBalance::<T, I>::mutate(collection, &dest, |balance| {
balance.saturating_inc();
});
Self::increment_account_balance(collection, &dest, (deposit_account, deposit_amount))?;

// Update account ownership information.
Account::<T, I>::remove((&details.owner, &collection, &item));
Expand Down
6 changes: 5 additions & 1 deletion pallets/nfts/src/impl_nonfungibles.rs
Original file line number Diff line number Diff line change
Expand Up @@ -411,7 +411,11 @@ impl<T: Config<I>, I: 'static> Transfer<T::AccountId> for Pallet<T, I> {
item: &Self::ItemId,
destination: &T::AccountId,
) -> DispatchResult {
Self::do_transfer(*collection, *item, destination.clone(), |_, _| Ok(()))
// The item's owner pays for the deposit of `AccountBalance` if the `dest` holds no items
// in `collection`. A malicious actor could have a deposit reserved from `dest` without
// them knowing about the transfer. The deposit amount can be accounted for in the off chain
// price of the NFT.
chungquantin marked this conversation as resolved.
Show resolved Hide resolved
Self::do_transfer(None, *collection, *item, destination.clone(), |_, _| Ok(()))
}

fn disable_transfer(collection: &Self::CollectionId, item: &Self::ItemId) -> DispatchResult {
Expand Down
Loading
Loading