Skip to content

Commit

Permalink
payout and customers fixes (#178)
Browse files Browse the repository at this point in the history
  • Loading branch information
aie0 authored Nov 30, 2023
1 parent 523a9d4 commit c20bff2
Show file tree
Hide file tree
Showing 7 changed files with 188 additions and 74 deletions.
68 changes: 26 additions & 42 deletions pallets/ddc-customers/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,7 @@ mod tests;
use codec::{Decode, Encode, HasCompact};

use ddc_primitives::{BucketId, ClusterId};
use ddc_traits::{
cluster::ClusterVisitor,
customer::{CustomerCharger, CustomerChargerError},
};
use ddc_traits::{cluster::ClusterVisitor, customer::CustomerCharger};
use frame_support::{
parameter_types,
traits::{Currency, DefensiveSaturating, ExistenceRequirement},
Expand All @@ -21,7 +18,7 @@ use frame_support::{
use scale_info::TypeInfo;
use sp_runtime::{
traits::{AccountIdConversion, AtLeast32BitUnsigned, CheckedAdd, CheckedSub, Saturating, Zero},
RuntimeDebug, SaturatedConversion,
DispatchError, RuntimeDebug, SaturatedConversion,
};
use sp_std::prelude::*;

Expand Down Expand Up @@ -129,17 +126,13 @@ impl<
.ok_or(Error::<T>::ArithmeticOverflow)?;
if temp <= value {
unlocking_balance = temp;
self.active =
self.active.checked_sub(&last.value).ok_or(Error::<T>::ArithmeticUnderflow)?;
self.unlocking.pop();
} else {
let diff =
value.checked_sub(&unlocking_balance).ok_or(Error::<T>::ArithmeticUnderflow)?;

unlocking_balance =
unlocking_balance.checked_add(&diff).ok_or(Error::<T>::ArithmeticOverflow)?;
self.active =
self.active.checked_sub(&diff).ok_or(Error::<T>::ArithmeticUnderflow)?;
last.value =
last.value.checked_sub(&diff).ok_or(Error::<T>::ArithmeticUnderflow)?;
}
Expand Down Expand Up @@ -429,7 +422,7 @@ pub mod pallet {
.map_err(|_| Error::<T>::NoMoreChunks)?;
};

Self::update_ledger(&owner, &ledger);
<Ledger<T>>::insert(&owner, &ledger);

Self::deposit_event(Event::<T>::InitialDepositUnlock(ledger.owner, value));
}
Expand Down Expand Up @@ -466,7 +459,7 @@ pub mod pallet {
log::debug!("Updating ledger");
// This was the consequence of a partial deposit unlock. just update the ledger and
// move on.
Self::update_ledger(&owner, &ledger);
<Ledger<T>>::insert(&owner, &ledger);
};

log::debug!("Current total: {:?}", ledger.total);
Expand Down Expand Up @@ -506,11 +499,9 @@ pub mod pallet {
owner: &T::AccountId,
ledger: &AccountsLedger<T::AccountId, BalanceOf<T>, T>,
) -> DispatchResult {
let account_id = Self::account_id();

<T as pallet::Config>::Currency::transfer(
owner,
&account_id,
&Self::account_id(),
ledger.total,
ExistenceRequirement::KeepAlive,
)?;
Expand All @@ -519,14 +510,6 @@ pub mod pallet {
Ok(())
}

/// Update the ledger for a owner.
fn update_ledger(
owner: &T::AccountId,
ledger: &AccountsLedger<T::AccountId, BalanceOf<T>, T>,
) {
<Ledger<T>>::insert(owner, ledger);
}

/// Remove all associated data of a owner account from the accounts system.
///
/// Assumes storage is upgraded before calling.
Expand All @@ -547,47 +530,48 @@ pub mod pallet {
content_owner: T::AccountId,
billing_vault: T::AccountId,
amount: u128,
) -> Result<u128, CustomerChargerError> {
) -> Result<u128, DispatchError> {
let actually_charged: BalanceOf<T>;
let mut ledger = Self::ledger(&content_owner).ok_or(CustomerChargerError::NotOwner)?;
let mut amount_to_deduct = amount.saturated_into::<BalanceOf<T>>();
let mut ledger = Self::ledger(&content_owner).ok_or(Error::<T>::NotOwner)?;
let amount_to_deduct = amount.saturated_into::<BalanceOf<T>>();

ensure!(ledger.total >= ledger.active, CustomerChargerError::ArithmeticUnderflow);
if ledger.active >= amount_to_deduct {
actually_charged = amount_to_deduct;
ledger.active = ledger
.active
.checked_sub(&amount_to_deduct)
.ok_or(CustomerChargerError::ArithmeticUnderflow)?;
.ok_or(Error::<T>::ArithmeticUnderflow)?;
ledger.total = ledger
.total
.checked_sub(&amount_to_deduct)
.ok_or(CustomerChargerError::ArithmeticUnderflow)?;
Self::update_ledger(&content_owner, &ledger);
.ok_or(Error::<T>::ArithmeticUnderflow)?;

<Ledger<T>>::insert(&content_owner, &ledger);
} else {
let diff = amount_to_deduct
.checked_sub(&ledger.active)
.ok_or(CustomerChargerError::ArithmeticUnderflow)?;
actually_charged = diff;
.ok_or(Error::<T>::ArithmeticUnderflow)?;

actually_charged = ledger.active;
ledger.total = ledger
.total
.checked_sub(&ledger.active)
.ok_or(CustomerChargerError::ArithmeticUnderflow)?;
amount_to_deduct = ledger.active;
.ok_or(Error::<T>::ArithmeticUnderflow)?;
ledger.active = BalanceOf::<T>::zero();
let (ledger, _charged) = ledger
.charge_unlocking(diff)
.map_err(|_| CustomerChargerError::UnlockFailed)?;
Self::update_ledger(&content_owner, &ledger);
};

let (ledger, charged) = ledger.charge_unlocking(diff)?;

actually_charged.checked_add(&charged).ok_or(Error::<T>::ArithmeticUnderflow)?;

<Ledger<T>>::insert(&content_owner, &ledger);
}

<T as pallet::Config>::Currency::transfer(
&Self::account_id(),
&billing_vault,
amount_to_deduct,
ExistenceRequirement::KeepAlive,
)
.map_err(|_| CustomerChargerError::TransferFailed)?;
actually_charged,
ExistenceRequirement::AllowDeath,
)?;
Self::deposit_event(Event::<T>::Charged(content_owner, amount_to_deduct));

Ok(actually_charged.saturated_into::<u128>())
Expand Down
6 changes: 4 additions & 2 deletions pallets/ddc-customers/src/mock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -163,8 +163,10 @@ impl ExtBuilder {
sp_tracing::try_init_simple();
let mut storage = frame_system::GenesisConfig::default().build_storage::<Test>().unwrap();

let _ = pallet_balances::GenesisConfig::<Test> { balances: vec![(1, 100), (2, 100)] }
.assimilate_storage(&mut storage);
let _ = pallet_balances::GenesisConfig::<Test> {
balances: vec![(1, 100), (2, 100), (3, 1000)],
}
.assimilate_storage(&mut storage);

TestExternalities::new(storage)
}
Expand Down
114 changes: 102 additions & 12 deletions pallets/ddc-customers/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -79,26 +79,27 @@ fn deposit_and_deposit_extra_works() {
BalancesError::<Test, _>::KeepAlive
);

let amount1 = 10_u128;
// Deposited
assert_ok!(DdcCustomers::deposit(RuntimeOrigin::signed(account_1), 10_u128));
assert_ok!(DdcCustomers::deposit(RuntimeOrigin::signed(account_1), amount1));

// Check storage
assert_eq!(
DdcCustomers::ledger(&account_1),
Some(AccountsLedger {
owner: 1,
total: 10_u128,
active: 10_u128,
owner: account_1,
total: amount1,
active: amount1,
unlocking: Default::default(),
})
);

// Checking that event was emitted
System::assert_last_event(Event::Deposited(account_1, 10).into());
System::assert_last_event(Event::Deposited(account_1, amount1).into());

// Deposit should fail when called the second time
assert_noop!(
DdcCustomers::deposit(RuntimeOrigin::signed(account_1), 10_u128),
DdcCustomers::deposit(RuntimeOrigin::signed(account_1), amount1),
Error::<Test>::AlreadyPaired
);

Expand All @@ -109,21 +110,110 @@ fn deposit_and_deposit_extra_works() {
);

// Deposited extra
assert_ok!(DdcCustomers::deposit_extra(RuntimeOrigin::signed(account_1), 20_u128));
let amount2 = 20_u128;
assert_ok!(DdcCustomers::deposit_extra(RuntimeOrigin::signed(account_1), amount2));

// Check storage
assert_eq!(
DdcCustomers::ledger(&account_1),
Some(AccountsLedger {
owner: 1,
total: 30_u128,
active: 30_u128,
owner: account_1,
total: amount1 + amount2,
active: amount1 + amount2,
unlocking: Default::default(),
})
);

// Checking that event was emitted
System::assert_last_event(Event::Deposited(account_1, 20).into());
System::assert_last_event(Event::Deposited(account_1, amount2).into());
})
}

#[test]
fn charge_content_owner_works() {
ExtBuilder.build_and_execute(|| {
System::set_block_number(1);

let account_3 = 3;
let vault = 4;
let deposit = 100_u128;

let balance_before_deposit = Balances::free_balance(account_3);
// Deposited
assert_ok!(DdcCustomers::deposit(RuntimeOrigin::signed(account_3), deposit));
let balance_after_deposit = Balances::free_balance(account_3);
assert_eq!(balance_before_deposit - deposit, balance_after_deposit);

let pallet_balance = Balances::free_balance(DdcCustomers::account_id());
assert_eq!(deposit, pallet_balance);

// Check storage
assert_eq!(
DdcCustomers::ledger(&account_3),
Some(AccountsLedger {
owner: account_3,
total: deposit,
active: deposit,
unlocking: Default::default(),
})
);

// Checking that event was emitted
System::assert_last_event(Event::Deposited(account_3, deposit).into());

// successful transfer
let charge1 = 10;
let charged = DdcCustomers::charge_content_owner(account_3, vault, charge1).unwrap();
assert_eq!(charge1, charged);

let vault_balance = Balances::free_balance(vault);
assert_eq!(charged, vault_balance);

let account_balance = Balances::free_balance(account_3);
assert_eq!(balance_after_deposit, account_balance);

let pallet_balance_after_charge = Balances::free_balance(DdcCustomers::account_id());
assert_eq!(pallet_balance - charged, pallet_balance_after_charge);

// Check storage
assert_eq!(
DdcCustomers::ledger(&account_3),
Some(AccountsLedger {
owner: account_3,
total: deposit - charge1,
active: deposit - charge1,
unlocking: Default::default(),
})
);

// failed transfer
let charge2 = 100u128;
let charge_result = DdcCustomers::charge_content_owner(account_3, vault, charge2).unwrap();
assert_eq!(
DdcCustomers::ledger(&account_3),
Some(AccountsLedger {
owner: account_3,
total: 0,
active: 0,
unlocking: Default::default(),
})
);

assert_eq!(0, Balances::free_balance(DdcCustomers::account_id()));
assert_eq!(charge_result, deposit - charge1);

assert_ok!(DdcCustomers::deposit_extra(RuntimeOrigin::signed(account_3), deposit));
assert_eq!(
DdcCustomers::ledger(&account_3),
Some(AccountsLedger {
owner: account_3,
total: deposit,
active: deposit,
unlocking: Default::default(),
})
);

assert_eq!(deposit, Balances::free_balance(DdcCustomers::account_id()));
})
}

Expand Down Expand Up @@ -188,7 +278,7 @@ fn unlock_and_withdraw_deposit_works() {
})
);

// Unlock remaining chuncks & withdraw
// Unlock remaining chunks & withdraw
assert_ok!(DdcCustomers::unlock_deposit(RuntimeOrigin::signed(account_1), 3_u128));
System::set_block_number(52);
assert_ok!(DdcCustomers::withdraw_unlocked_deposit(RuntimeOrigin::signed(account_1)));
Expand Down
Loading

0 comments on commit c20bff2

Please sign in to comment.