mirror of
https://github.com/bitcoinresearchkit/brk.git
synced 2026-06-14 16:53:35 -07:00
248 lines
6.1 KiB
Rust
248 lines
6.1 KiB
Rust
use std::ops::{Add, Rem};
|
|
|
|
use allocative::Allocative;
|
|
use brk_error::Error;
|
|
use jiff::Span;
|
|
use serde::Serialize;
|
|
use vecdb::{CheckedSub, FromCoarserIndex, PrintableIndex, StoredCompressed};
|
|
use zerocopy_derive::{FromBytes, Immutable, IntoBytes, KnownLayout};
|
|
|
|
use crate::{DecadeIndex, MonthIndex, QuarterIndex, SemesterIndex, WeekIndex, YearIndex};
|
|
|
|
use super::Date;
|
|
|
|
#[derive(
|
|
Debug,
|
|
Default,
|
|
Clone,
|
|
Copy,
|
|
PartialEq,
|
|
Eq,
|
|
PartialOrd,
|
|
Ord,
|
|
FromBytes,
|
|
Immutable,
|
|
IntoBytes,
|
|
KnownLayout,
|
|
Serialize,
|
|
StoredCompressed,
|
|
Allocative,
|
|
)]
|
|
pub struct DateIndex(u16);
|
|
|
|
impl DateIndex {
|
|
pub const BYTES: usize = size_of::<Self>();
|
|
}
|
|
|
|
impl From<DateIndex> for usize {
|
|
fn from(value: DateIndex) -> Self {
|
|
value.0 as usize
|
|
}
|
|
}
|
|
|
|
impl From<DateIndex> for u64 {
|
|
fn from(value: DateIndex) -> Self {
|
|
value.0 as u64
|
|
}
|
|
}
|
|
|
|
impl From<usize> for DateIndex {
|
|
fn from(value: usize) -> Self {
|
|
Self(value as u16)
|
|
}
|
|
}
|
|
|
|
impl From<DateIndex> for i64 {
|
|
fn from(value: DateIndex) -> Self {
|
|
value.0 as i64
|
|
}
|
|
}
|
|
|
|
impl Add<usize> for DateIndex {
|
|
type Output = Self;
|
|
fn add(self, rhs: usize) -> Self::Output {
|
|
Self(self.0 + rhs as u16)
|
|
}
|
|
}
|
|
|
|
impl TryFrom<Date> for DateIndex {
|
|
type Error = Error;
|
|
fn try_from(value: Date) -> Result<Self, Self::Error> {
|
|
let value_ = jiff::civil::Date::from(value);
|
|
if value_ < Date::INDEX_ZERO_ {
|
|
Err(Error::UnindexableDate)
|
|
} else if value == Date::INDEX_ZERO {
|
|
Ok(Self(0))
|
|
} else if value_ < Date::INDEX_ONE_ {
|
|
Err(Error::UnindexableDate)
|
|
} else if value == Date::INDEX_ONE {
|
|
Ok(Self(1))
|
|
} else {
|
|
Ok(Self(Date::INDEX_ONE_.until(value_)?.get_days() as u16 + 1))
|
|
}
|
|
}
|
|
}
|
|
|
|
impl CheckedSub for DateIndex {
|
|
fn checked_sub(self, rhs: Self) -> Option<Self> {
|
|
self.0.checked_sub(rhs.0).map(Self)
|
|
}
|
|
}
|
|
|
|
impl Rem<usize> for DateIndex {
|
|
type Output = Self;
|
|
fn rem(self, rhs: usize) -> Self::Output {
|
|
Self(self.0 % rhs as u16)
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for DateIndex {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let mut buf = itoa::Buffer::new();
|
|
let str = buf.format(self.0);
|
|
f.write_str(str)
|
|
}
|
|
}
|
|
|
|
impl PrintableIndex for DateIndex {
|
|
fn to_string() -> &'static str {
|
|
"dateindex"
|
|
}
|
|
|
|
fn to_possible_strings() -> &'static [&'static str] {
|
|
&["d", "date", "dateindex"]
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<WeekIndex> for DateIndex {
|
|
fn min_from(coarser: WeekIndex) -> usize {
|
|
let coarser = usize::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else if coarser == 1 {
|
|
1
|
|
} else {
|
|
4 + (coarser - 2) * 7
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: WeekIndex) -> usize {
|
|
let coarser = usize::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else if coarser == 1 {
|
|
3
|
|
} else {
|
|
3 + (coarser - 1) * 7
|
|
}
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<MonthIndex> for DateIndex {
|
|
fn min_from(coarser: MonthIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else {
|
|
let d = Date::new(2009, 1, 1)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(coarser))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: MonthIndex) -> usize {
|
|
let d = Date::new(2009, 1, 31)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(u16::from(coarser)))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<QuarterIndex> for DateIndex {
|
|
fn min_from(coarser: QuarterIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else {
|
|
let d = Date::new(2009, 1, 1)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(3 * coarser))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: QuarterIndex) -> usize {
|
|
let d = Date::new(2009, 3, 31)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(3 * u16::from(coarser)))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<SemesterIndex> for DateIndex {
|
|
fn min_from(coarser: SemesterIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else {
|
|
let d = Date::new(2009, 1, 1)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(6 * coarser))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: SemesterIndex) -> usize {
|
|
let d = Date::new(2009, 5, 31)
|
|
.into_jiff()
|
|
.checked_add(Span::new().months(1 + 6 * u16::from(coarser)))
|
|
.unwrap();
|
|
DateIndex::try_from(Date::from(d)).unwrap().into()
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<YearIndex> for DateIndex {
|
|
fn min_from(coarser: YearIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else {
|
|
Self::try_from(Date::new(2009 + coarser, 1, 1))
|
|
.unwrap()
|
|
.into()
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: YearIndex) -> usize {
|
|
Self::try_from(Date::new(2009 + u16::from(coarser), 12, 31))
|
|
.unwrap()
|
|
.into()
|
|
}
|
|
}
|
|
|
|
impl FromCoarserIndex<DecadeIndex> for DateIndex {
|
|
fn min_from(coarser: DecadeIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
if coarser == 0 {
|
|
0
|
|
} else {
|
|
Self::try_from(Date::new(2000 + 10 * coarser, 1, 1))
|
|
.unwrap()
|
|
.into()
|
|
}
|
|
}
|
|
|
|
fn max_from_(coarser: DecadeIndex) -> usize {
|
|
let coarser = u16::from(coarser);
|
|
Self::try_from(Date::new(2009 + (10 * coarser), 12, 31))
|
|
.unwrap()
|
|
.into()
|
|
}
|
|
}
|