types: added fract sats

This commit is contained in:
nym21
2026-01-26 10:43:26 +01:00
parent 371fb2cb17
commit d9dabb4a96
2 changed files with 193 additions and 0 deletions

View File

@@ -127,6 +127,7 @@ mod rawlocktime;
mod recommendedfees;
mod rewardstats;
mod sats;
mod satsfract;
mod semesterindex;
mod stored_bool;
mod stored_f32;
@@ -298,6 +299,7 @@ pub use rawlocktime::*;
pub use recommendedfees::*;
pub use rewardstats::*;
pub use sats::*;
pub use satsfract::*;
pub use semesterindex::*;
pub use stored_bool::*;
pub use stored_f32::*;

View File

@@ -0,0 +1,191 @@
use std::{
cmp::Ordering,
f64,
iter::Sum,
ops::{Add, AddAssign, Div, Mul, Sub},
};
use derive_more::Deref;
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use vecdb::{CheckedSub, Formattable, Pco};
/// Fractional satoshis (f64) - for representing USD prices in sats
///
/// Formula: `sats_fract = usd_value * 100_000_000 / btc_price`
///
/// When BTC is $100,000:
/// - $1 = 1,000 sats
/// - $0.001 = 1 sat
/// - $0.0001 = 0.1 sats (fractional)
#[derive(Debug, Deref, Default, Clone, Copy, Serialize, Deserialize, Pco, JsonSchema)]
pub struct SatsFract(f64);
impl SatsFract {
pub const ZERO: Self = Self(0.0);
pub const NAN: Self = Self(f64::NAN);
pub const SATS_PER_BTC: f64 = 100_000_000.0;
pub const fn new(sats: f64) -> Self {
Self(sats)
}
pub fn is_nan(&self) -> bool {
self.0.is_nan()
}
}
impl From<f64> for SatsFract {
#[inline]
fn from(value: f64) -> Self {
Self(value)
}
}
impl From<f32> for SatsFract {
#[inline]
fn from(value: f32) -> Self {
Self(value as f64)
}
}
impl From<usize> for SatsFract {
#[inline]
fn from(value: usize) -> Self {
Self(value as f64)
}
}
impl From<SatsFract> for f64 {
#[inline]
fn from(value: SatsFract) -> Self {
value.0
}
}
impl From<SatsFract> for f32 {
#[inline]
fn from(value: SatsFract) -> Self {
value.0 as f32
}
}
impl Add for SatsFract {
type Output = Self;
fn add(self, rhs: Self) -> Self::Output {
Self(self.0 + rhs.0)
}
}
impl AddAssign for SatsFract {
fn add_assign(&mut self, rhs: Self) {
*self = *self + rhs
}
}
impl Sub for SatsFract {
type Output = Self;
fn sub(self, rhs: Self) -> Self::Output {
Self(self.0 - rhs.0)
}
}
impl Mul for SatsFract {
type Output = Self;
fn mul(self, rhs: Self) -> Self::Output {
Self(self.0 * rhs.0)
}
}
impl Mul<usize> for SatsFract {
type Output = Self;
fn mul(self, rhs: usize) -> Self::Output {
Self(self.0 * rhs as f64)
}
}
impl Div<usize> for SatsFract {
type Output = Self;
fn div(self, rhs: usize) -> Self::Output {
if rhs == 0 {
Self::NAN
} else {
Self(self.0 / rhs as f64)
}
}
}
impl Div for SatsFract {
type Output = Self;
fn div(self, rhs: Self) -> Self::Output {
if rhs.0 == 0.0 {
Self::NAN
} else {
Self(self.0 / rhs.0)
}
}
}
impl CheckedSub for SatsFract {
fn checked_sub(self, rhs: Self) -> Option<Self> {
Some(Self(self.0 - rhs.0))
}
}
impl CheckedSub<usize> for SatsFract {
fn checked_sub(self, rhs: usize) -> Option<Self> {
Some(Self(self.0 - rhs as f64))
}
}
impl PartialEq for SatsFract {
fn eq(&self, other: &Self) -> bool {
match (self.0.is_nan(), other.0.is_nan()) {
(true, true) => true,
(true, false) | (false, true) => false,
(false, false) => self.0 == other.0,
}
}
}
impl Eq for SatsFract {}
#[allow(clippy::derive_ord_xor_partial_ord)]
impl PartialOrd for SatsFract {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
#[allow(clippy::derive_ord_xor_partial_ord)]
impl Ord for SatsFract {
fn cmp(&self, other: &Self) -> Ordering {
match (self.0.is_nan(), other.0.is_nan()) {
(true, true) => Ordering::Equal,
(true, false) => Ordering::Less,
(false, true) => Ordering::Greater,
(false, false) => self.0.partial_cmp(&other.0).unwrap(),
}
}
}
impl Sum for SatsFract {
fn sum<I: Iterator<Item = Self>>(iter: I) -> Self {
Self(iter.map(|v| v.0).sum::<f64>())
}
}
impl std::fmt::Display for SatsFract {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut buf = ryu::Buffer::new();
let str = buf.format(self.0);
f.write_str(str)
}
}
impl Formattable for SatsFract {
#[inline(always)]
fn may_need_escaping() -> bool {
false
}
}