// This file is part of Bizinikiwi. // Copyright (C) Parity Technologies (UK) Ltd. and Dijital Kurdistan Tech Institute // SPDX-License-Identifier: Apache-2.0 // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use super::*; use pezframe_support::{ pezpallet_prelude::*, traits::{ fungible::Balanced, tokens::{Fortitude::Polite, Precision::Exact, Preservation::Expendable}, OnUnbalanced, }, }; use pezsp_arithmetic::{ traits::{SaturatedConversion, Saturating}, FixedPointNumber, FixedU64, }; use pezsp_runtime::traits::{AccountIdConversion, BlockNumberProvider}; impl Pezpallet { pub fn current_timeslice() -> Timeslice { let latest = RCBlockNumberProviderOf::::current_block_number(); let timeslice_period = T::TimeslicePeriod::get(); (latest / timeslice_period).saturated_into() } pub fn latest_timeslice_ready_to_commit(config: &ConfigRecordOf) -> Timeslice { let latest = RCBlockNumberProviderOf::::current_block_number(); let advanced = latest.saturating_add(config.advance_notice); let timeslice_period = T::TimeslicePeriod::get(); (advanced / timeslice_period).saturated_into() } pub fn next_timeslice_to_commit( config: &ConfigRecordOf, status: &StatusRecord, ) -> Option { if status.last_committed_timeslice < Self::latest_timeslice_ready_to_commit(config) { Some(status.last_committed_timeslice + 1) } else { None } } pub fn account_id() -> T::AccountId { T::PalletId::get().into_account_truncating() } pub fn sale_price(sale: &SaleInfoRecordOf, now: RelayBlockNumberOf) -> BalanceOf { let num = now.saturating_sub(sale.sale_start).min(sale.leadin_length).saturated_into(); let through = FixedU64::from_rational(num, sale.leadin_length.saturated_into()); T::PriceAdapter::leadin_factor_at(through).saturating_mul_int(sale.end_price) } pub(crate) fn charge(who: &T::AccountId, amount: BalanceOf) -> DispatchResult { let credit = T::Currency::withdraw(&who, amount, Exact, Expendable, Polite)?; T::OnRevenue::on_unbalanced(credit); Ok(()) } /// Buy a core at the specified price (price is to be determined by the caller). /// /// Note: It is the responsibility of the caller to write back the changed `SaleInfoRecordOf` to /// storage. pub(crate) fn purchase_core( who: &T::AccountId, price: BalanceOf, sale: &mut SaleInfoRecordOf, ) -> Result { Self::charge(who, price)?; log::debug!("Purchased core at: {:?}", price); let core = sale.first_core.saturating_add(sale.cores_sold); sale.cores_sold.saturating_inc(); if sale.cores_sold <= sale.ideal_cores_sold || sale.sellout_price.is_none() { sale.sellout_price = Some(price); } Ok(core) } pub fn issue( core: CoreIndex, begin: Timeslice, mask: CoreMask, end: Timeslice, owner: Option, paid: Option>, ) -> RegionId { let id = RegionId { begin, core, mask }; let record = RegionRecord { end, owner, paid }; Regions::::insert(&id, &record); id } pub(crate) fn utilize( mut region_id: RegionId, maybe_check_owner: Option, finality: Finality, ) -> Result)>, Error> { let status = Status::::get().ok_or(Error::::Uninitialized)?; let region = Regions::::get(®ion_id).ok_or(Error::::UnknownRegion)?; if let Some(check_owner) = maybe_check_owner { ensure!(Some(check_owner) == region.owner, Error::::NotOwner); } Regions::::remove(®ion_id); let last_committed_timeslice = status.last_committed_timeslice; if region_id.begin <= last_committed_timeslice { let duration = region.end.saturating_sub(region_id.begin); region_id.begin = last_committed_timeslice + 1; if region_id.begin >= region.end { Self::deposit_event(Event::RegionDropped { region_id, duration }); return Ok(None); } } else { Workplan::::mutate_extant((region_id.begin, region_id.core), |p| { p.retain(|i| (i.mask & region_id.mask).is_void()) }); } if finality == Finality::Provisional { Regions::::insert(®ion_id, ®ion); } Ok(Some((region_id, region))) } // Remove a region from on-demand pool contributions. Useful in cases where it was pooled // provisionally and it is being redispatched (partition/interlace/assign). // // Takes both the region_id and (a reference to) the region as arguments to avoid another DB // read. No-op for regions which have not been pooled. pub(crate) fn force_unpool_region( region_id: RegionId, region: &RegionRecordOf, status: &StatusRecord, ) { // We don't care if this fails or not, just that it is removed if present. This is to // account for the case where a region is pooled provisionally and redispatched. if InstaPoolContribution::::take(region_id).is_some() { // `InstaPoolHistory` is calculated from the `InstaPoolIo` one timeslice in advance. // Therefore we need to schedule this for the timeslice after that. let end_timeslice = status.last_committed_timeslice + 1; // InstaPoolIo has already accounted for regions that have already ended. Regions ending // this timeslice would have region.end == unpooled_at below. if region.end <= end_timeslice { return; } // Account for the change in `InstaPoolIo` either from the start of the region or from // the current timeslice if we are already part-way through the region. let size = region_id.mask.count_ones() as i32; let unpooled_at = end_timeslice.max(region_id.begin); InstaPoolIo::::mutate(unpooled_at, |a| a.private.saturating_reduce(size)); InstaPoolIo::::mutate(region.end, |a| a.private.saturating_accrue(size)); Self::deposit_event(Event::::RegionUnpooled { region_id, when: unpooled_at }); }; } }