mirror of
https://github.com/bitcoinresearchkit/brk.git
synced 2026-07-05 00:03:40 -07:00
159 lines
4.6 KiB
Rust
159 lines
4.6 KiB
Rust
use brk_traversable::Traversable;
|
|
use rayon::prelude::*;
|
|
use serde::Serialize;
|
|
|
|
use super::CohortName;
|
|
|
|
/// "At least X% loss" threshold names (9 thresholds).
|
|
pub const LOSS_NAMES: Loss<CohortName> = Loss {
|
|
all: CohortName::new("utxos_in_loss", "All", "In Loss"),
|
|
_10pct: CohortName::new("utxos_over_10pct_in_loss", ">=10%", "Over 10% in Loss"),
|
|
_20pct: CohortName::new("utxos_over_20pct_in_loss", ">=20%", "Over 20% in Loss"),
|
|
_30pct: CohortName::new("utxos_over_30pct_in_loss", ">=30%", "Over 30% in Loss"),
|
|
_40pct: CohortName::new("utxos_over_40pct_in_loss", ">=40%", "Over 40% in Loss"),
|
|
_50pct: CohortName::new("utxos_over_50pct_in_loss", ">=50%", "Over 50% in Loss"),
|
|
_60pct: CohortName::new("utxos_over_60pct_in_loss", ">=60%", "Over 60% in Loss"),
|
|
_70pct: CohortName::new("utxos_over_70pct_in_loss", ">=70%", "Over 70% in Loss"),
|
|
_80pct: CohortName::new("utxos_over_80pct_in_loss", ">=80%", "Over 80% in Loss"),
|
|
};
|
|
|
|
/// Number of loss thresholds.
|
|
pub const LOSS_COUNT: usize = 9;
|
|
|
|
impl Loss<CohortName> {
|
|
pub const fn names() -> &'static Self {
|
|
&LOSS_NAMES
|
|
}
|
|
}
|
|
|
|
/// 9 "at least X% loss" aggregate thresholds.
|
|
///
|
|
/// Each is a suffix sum over the profitability ranges, from most loss-making up.
|
|
#[derive(Default, Clone, Traversable, Serialize)]
|
|
pub struct Loss<T> {
|
|
pub all: T,
|
|
pub _10pct: T,
|
|
pub _20pct: T,
|
|
pub _30pct: T,
|
|
pub _40pct: T,
|
|
pub _50pct: T,
|
|
pub _60pct: T,
|
|
pub _70pct: T,
|
|
pub _80pct: T,
|
|
}
|
|
|
|
impl<T> Loss<T> {
|
|
pub fn new<F>(mut create: F) -> Self
|
|
where
|
|
F: FnMut(&'static str) -> T,
|
|
{
|
|
let n = &LOSS_NAMES;
|
|
Self {
|
|
all: create(n.all.id),
|
|
_10pct: create(n._10pct.id),
|
|
_20pct: create(n._20pct.id),
|
|
_30pct: create(n._30pct.id),
|
|
_40pct: create(n._40pct.id),
|
|
_50pct: create(n._50pct.id),
|
|
_60pct: create(n._60pct.id),
|
|
_70pct: create(n._70pct.id),
|
|
_80pct: create(n._80pct.id),
|
|
}
|
|
}
|
|
|
|
pub fn try_new<F, E>(mut create: F) -> Result<Self, E>
|
|
where
|
|
F: FnMut(&'static str) -> Result<T, E>,
|
|
{
|
|
let n = &LOSS_NAMES;
|
|
Ok(Self {
|
|
all: create(n.all.id)?,
|
|
_10pct: create(n._10pct.id)?,
|
|
_20pct: create(n._20pct.id)?,
|
|
_30pct: create(n._30pct.id)?,
|
|
_40pct: create(n._40pct.id)?,
|
|
_50pct: create(n._50pct.id)?,
|
|
_60pct: create(n._60pct.id)?,
|
|
_70pct: create(n._70pct.id)?,
|
|
_80pct: create(n._80pct.id)?,
|
|
})
|
|
}
|
|
|
|
pub fn iter(&self) -> impl Iterator<Item = &T> {
|
|
[
|
|
&self.all,
|
|
&self._10pct,
|
|
&self._20pct,
|
|
&self._30pct,
|
|
&self._40pct,
|
|
&self._50pct,
|
|
&self._60pct,
|
|
&self._70pct,
|
|
&self._80pct,
|
|
]
|
|
.into_iter()
|
|
}
|
|
|
|
pub fn iter_mut(&mut self) -> impl Iterator<Item = &mut T> {
|
|
[
|
|
&mut self.all,
|
|
&mut self._10pct,
|
|
&mut self._20pct,
|
|
&mut self._30pct,
|
|
&mut self._40pct,
|
|
&mut self._50pct,
|
|
&mut self._60pct,
|
|
&mut self._70pct,
|
|
&mut self._80pct,
|
|
]
|
|
.into_iter()
|
|
}
|
|
|
|
pub fn par_iter_mut(&mut self) -> impl ParallelIterator<Item = &mut T>
|
|
where
|
|
T: Send + Sync,
|
|
{
|
|
[
|
|
&mut self.all,
|
|
&mut self._10pct,
|
|
&mut self._20pct,
|
|
&mut self._30pct,
|
|
&mut self._40pct,
|
|
&mut self._50pct,
|
|
&mut self._60pct,
|
|
&mut self._70pct,
|
|
&mut self._80pct,
|
|
]
|
|
.into_par_iter()
|
|
}
|
|
|
|
/// Access as array for indexed accumulation.
|
|
pub fn as_array_mut(&mut self) -> [&mut T; LOSS_COUNT] {
|
|
[
|
|
&mut self.all,
|
|
&mut self._10pct,
|
|
&mut self._20pct,
|
|
&mut self._30pct,
|
|
&mut self._40pct,
|
|
&mut self._50pct,
|
|
&mut self._60pct,
|
|
&mut self._70pct,
|
|
&mut self._80pct,
|
|
]
|
|
}
|
|
|
|
/// Iterate from narrowest (_80pct) to broadest (all), yielding each threshold
|
|
/// with a growing suffix slice of `ranges` (1 range, 2 ranges, ..., LOSS_COUNT).
|
|
pub fn iter_mut_with_growing_suffix<'a, R>(
|
|
&'a mut self,
|
|
ranges: &'a [R],
|
|
) -> impl Iterator<Item = (&'a mut T, &'a [R])> {
|
|
let len = ranges.len();
|
|
self.as_array_mut()
|
|
.into_iter()
|
|
.rev()
|
|
.enumerate()
|
|
.map(move |(n, threshold)| (threshold, &ranges[len - 2 - n..]))
|
|
}
|
|
}
|