use core::fmt;
use core::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign};
#[cfg(feature = "arbitrary")]
use arbitrary::{Arbitrary, Unstructured};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
pub const WITNESS_SCALE_FACTOR: usize = 4;
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[cfg_attr(feature = "serde", serde(transparent))]
pub struct Weight(u64);
impl Weight {
pub const ZERO: Weight = Weight(0);
pub const MIN: Weight = Weight(u64::MIN);
pub const MAX: Weight = Weight(u64::MAX);
pub const WITNESS_SCALE_FACTOR: u64 = WITNESS_SCALE_FACTOR as u64;
pub const MAX_BLOCK: Weight = Weight(4_000_000);
pub const MIN_TRANSACTION: Weight = Weight(Self::WITNESS_SCALE_FACTOR * 60);
pub const fn from_wu(wu: u64) -> Self { Weight(wu) }
pub const fn from_wu_usize(wu: usize) -> Self { Weight(wu as u64) }
pub fn from_kwu(wu: u64) -> Option<Self> { wu.checked_mul(1000).map(Weight) }
pub fn from_vb(vb: u64) -> Option<Self> {
vb.checked_mul(Self::WITNESS_SCALE_FACTOR).map(Weight::from_wu)
}
pub const fn from_vb_unwrap(vb: u64) -> Weight {
match vb.checked_mul(Self::WITNESS_SCALE_FACTOR) {
Some(weight) => Weight(weight),
None => panic!("checked_mul overflowed"),
}
}
pub const fn from_vb_unchecked(vb: u64) -> Self { Weight::from_wu(vb * 4) }
pub const fn from_witness_data_size(witness_size: u64) -> Self { Weight(witness_size) }
pub const fn from_non_witness_data_size(non_witness_size: u64) -> Self {
Weight(non_witness_size * Self::WITNESS_SCALE_FACTOR)
}
pub const fn to_wu(self) -> u64 { self.0 }
pub const fn to_kwu_floor(self) -> u64 { self.0 / 1000 }
pub const fn to_vbytes_floor(self) -> u64 { self.0 / Self::WITNESS_SCALE_FACTOR }
pub const fn to_vbytes_ceil(self) -> u64 {
(self.0 + Self::WITNESS_SCALE_FACTOR - 1) / Self::WITNESS_SCALE_FACTOR
}
pub fn checked_add(self, rhs: Self) -> Option<Self> { self.0.checked_add(rhs.0).map(Self) }
pub fn checked_sub(self, rhs: Self) -> Option<Self> { self.0.checked_sub(rhs.0).map(Self) }
pub fn checked_mul(self, rhs: u64) -> Option<Self> { self.0.checked_mul(rhs).map(Self) }
pub fn checked_div(self, rhs: u64) -> Option<Self> { self.0.checked_div(rhs).map(Self) }
}
impl fmt::Display for Weight {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if f.alternate() {
write!(f, "{} wu", self.0)
} else {
fmt::Display::fmt(&self.0, f)
}
}
}
impl From<Weight> for u64 {
fn from(value: Weight) -> Self { value.to_wu() }
}
impl Add for Weight {
type Output = Weight;
fn add(self, rhs: Weight) -> Self::Output { Weight(self.0 + rhs.0) }
}
impl AddAssign for Weight {
fn add_assign(&mut self, rhs: Self) { self.0 += rhs.0 }
}
impl Sub for Weight {
type Output = Weight;
fn sub(self, rhs: Weight) -> Self::Output { Weight(self.0 - rhs.0) }
}
impl SubAssign for Weight {
fn sub_assign(&mut self, rhs: Self) { self.0 -= rhs.0 }
}
impl Mul<u64> for Weight {
type Output = Weight;
fn mul(self, rhs: u64) -> Self::Output { Weight(self.0 * rhs) }
}
impl Mul<Weight> for u64 {
type Output = Weight;
fn mul(self, rhs: Weight) -> Self::Output { Weight(self * rhs.0) }
}
impl MulAssign<u64> for Weight {
fn mul_assign(&mut self, rhs: u64) { self.0 *= rhs }
}
impl Div<u64> for Weight {
type Output = Weight;
fn div(self, rhs: u64) -> Self::Output { Weight(self.0 / rhs) }
}
impl Div<Weight> for Weight {
type Output = u64;
fn div(self, rhs: Weight) -> Self::Output { self.to_wu() / rhs.to_wu() }
}
impl DivAssign<u64> for Weight {
fn div_assign(&mut self, rhs: u64) { self.0 /= rhs }
}
impl core::iter::Sum for Weight {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = Self>,
{
Weight(iter.map(Weight::to_wu).sum())
}
}
impl<'a> core::iter::Sum<&'a Weight> for Weight {
fn sum<I>(iter: I) -> Self
where
I: Iterator<Item = &'a Weight>,
{
iter.cloned().sum()
}
}
crate::impl_parse_str_from_int_infallible!(Weight, u64, from_wu);
#[cfg(feature = "arbitrary")]
impl<'a> Arbitrary<'a> for Weight {
fn arbitrary(u: &mut Unstructured<'a>) -> arbitrary::Result<Self> {
let w = u64::arbitrary(u)?;
Ok(Weight(w))
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn weight_constructor() {
assert_eq!(Weight::ZERO, Weight::from_wu(0));
assert_eq!(Weight::ZERO, Weight::from_wu_usize(0_usize));
}
#[test]
fn kilo_weight_constructor() {
assert_eq!(Weight(1_000), Weight::from_kwu(1).expect("expected weight unit"));
}
#[test]
#[should_panic]
fn kilo_weight_constructor_panic() {
Weight::from_kwu(u64::MAX).expect("expected weight unit");
}
#[test]
fn from_vb() {
let w = Weight::from_vb(1).expect("expected weight unit");
assert_eq!(Weight(4), w);
let w = Weight::from_vb(u64::MAX);
assert_eq!(None, w);
}
#[test]
fn from_vb_const() {
const WU: Weight = Weight::from_vb_unwrap(1);
assert_eq!(Weight(4), WU);
}
#[test]
fn from_vb_unchecked() {
let w = Weight::from_vb_unchecked(1);
assert_eq!(Weight(4), w);
}
#[test]
#[cfg(debug_assertions)]
#[should_panic]
fn from_vb_unchecked_panic() { Weight::from_vb_unchecked(u64::MAX); }
#[test]
fn from_witness_data_size() {
let witness_data_size = 1;
assert_eq!(Weight(witness_data_size), Weight::from_witness_data_size(witness_data_size));
}
#[test]
fn from_non_witness_data_size() {
assert_eq!(Weight(4), Weight::from_non_witness_data_size(1));
}
#[test]
fn to_kwu_floor() {
assert_eq!(1, Weight(1_000).to_kwu_floor());
}
#[test]
fn to_vb_floor() {
assert_eq!(1, Weight(4).to_vbytes_floor());
assert_eq!(1, Weight(5).to_vbytes_floor());
}
#[test]
fn to_vb_ceil() {
assert_eq!(1, Weight(4).to_vbytes_ceil());
assert_eq!(2, Weight(5).to_vbytes_ceil());
}
#[test]
fn checked_add() {
let result = Weight(1).checked_add(Weight(1)).expect("expected weight unit");
assert_eq!(Weight(2), result);
let result = Weight::MAX.checked_add(Weight(1));
assert_eq!(None, result);
}
#[test]
fn checked_sub() {
let result = Weight(1).checked_sub(Weight(1)).expect("expected weight unit");
assert_eq!(Weight::ZERO, result);
let result = Weight::MIN.checked_sub(Weight(1));
assert_eq!(None, result);
}
#[test]
fn checked_mul() {
let result = Weight(2).checked_mul(2).expect("expected weight unit");
assert_eq!(Weight(4), result);
let result = Weight::MAX.checked_mul(2);
assert_eq!(None, result);
}
#[test]
fn checked_div() {
let result = Weight(2).checked_div(2).expect("expected weight unit");
assert_eq!(Weight(1), result);
let result = Weight(2).checked_div(0);
assert_eq!(None, result);
}
}