use chrono::format::StrftimeItems; use chrono::DateTime; use chrono::FixedOffset; use chrono::LocalResult; use chrono::TimeZone; use chrono::Utc; use jj_lib::backend::Timestamp; use once_cell::sync::Lazy; use thiserror::Error; /// Parsed formatting items which should never contain an error. #[derive(Clone, Debug, Eq, PartialEq)] pub struct FormattingItems<'a> { items: Vec>, } impl<'a> FormattingItems<'a> { /// Parses strftime-like format string. pub fn parse(format: &'a str) -> Option { // If the parsed format contained an error, format().to_string() would panic. let items = StrftimeItems::new(format) .map(|item| match item { chrono::format::Item::Error => None, _ => Some(item), }) .collect::>()?; Some(FormattingItems { items }) } pub fn into_owned(self) -> FormattingItems<'static> { use chrono::format::Item; let items = self .items .into_iter() .map(|item| match item { Item::Literal(s) => Item::OwnedLiteral(s.into()), Item::OwnedLiteral(s) => Item::OwnedLiteral(s), Item::Space(s) => Item::OwnedSpace(s.into()), Item::OwnedSpace(s) => Item::OwnedSpace(s), Item::Numeric(spec, pad) => Item::Numeric(spec, pad), Item::Fixed(spec) => Item::Fixed(spec), Item::Error => Item::Error, // shouldn't exist, but just copy }) .collect(); FormattingItems { items } } } #[derive(Debug, Error)] #[error("Out-of-range date")] pub struct TimestampOutOfRange; fn datetime_from_timestamp( context: &Timestamp, ) -> Result, TimestampOutOfRange> { let utc = match Utc.timestamp_opt( context.timestamp.0.div_euclid(1000), (context.timestamp.0.rem_euclid(1000)) as u32 * 1000000, ) { LocalResult::None => { return Err(TimestampOutOfRange); } LocalResult::Single(x) => x, LocalResult::Ambiguous(y, _z) => y, }; Ok(utc.with_timezone( &FixedOffset::east_opt(context.tz_offset * 60) .unwrap_or_else(|| FixedOffset::east_opt(0).unwrap()), )) } pub fn format_absolute_timestamp(timestamp: &Timestamp) -> Result { static DEFAULT_FORMAT: Lazy = Lazy::new(|| FormattingItems::parse("%Y-%m-%d %H:%M:%S.%3f %:z").unwrap()); format_absolute_timestamp_with(timestamp, &DEFAULT_FORMAT) } pub fn format_absolute_timestamp_with( timestamp: &Timestamp, format: &FormattingItems, ) -> Result { let datetime = datetime_from_timestamp(timestamp)?; Ok(datetime.format_with_items(format.items.iter()).to_string()) } pub fn format_duration( from: &Timestamp, to: &Timestamp, format: &timeago::Formatter, ) -> Result { let duration = datetime_from_timestamp(to)? .signed_duration_since(datetime_from_timestamp(from)?) .to_std() .map_err(|_: chrono::OutOfRangeError| TimestampOutOfRange)?; Ok(format.convert(duration)) }