2023-03-14 02:25:00 +00:00
|
|
|
use chrono::format::StrftimeItems;
|
2024-08-22 18:18:15 +00:00
|
|
|
use chrono::DateTime;
|
|
|
|
use chrono::FixedOffset;
|
|
|
|
use chrono::LocalResult;
|
|
|
|
use chrono::TimeZone;
|
|
|
|
use chrono::Utc;
|
2023-06-28 14:12:40 +00:00
|
|
|
use jj_lib::backend::Timestamp;
|
2023-03-14 02:25:00 +00:00
|
|
|
use once_cell::sync::Lazy;
|
2024-02-28 05:23:52 +00:00
|
|
|
use thiserror::Error;
|
2023-01-14 13:40:04 +00:00
|
|
|
|
2023-03-14 13:28:22 +00:00
|
|
|
/// Parsed formatting items which should never contain an error.
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
|
|
pub struct FormattingItems<'a> {
|
|
|
|
items: Vec<chrono::format::Item<'a>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> FormattingItems<'a> {
|
|
|
|
/// Parses strftime-like format string.
|
|
|
|
pub fn parse(format: &'a str) -> Option<Self> {
|
|
|
|
// 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::<Option<_>>()?;
|
|
|
|
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 }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-28 05:23:52 +00:00
|
|
|
#[derive(Debug, Error)]
|
|
|
|
#[error("Out-of-range date")]
|
|
|
|
pub struct TimestampOutOfRange;
|
|
|
|
|
|
|
|
fn datetime_from_timestamp(
|
|
|
|
context: &Timestamp,
|
|
|
|
) -> Result<DateTime<FixedOffset>, TimestampOutOfRange> {
|
2023-01-14 13:40:04 +00:00
|
|
|
let utc = match Utc.timestamp_opt(
|
|
|
|
context.timestamp.0.div_euclid(1000),
|
|
|
|
(context.timestamp.0.rem_euclid(1000)) as u32 * 1000000,
|
|
|
|
) {
|
|
|
|
LocalResult::None => {
|
2024-02-28 05:23:52 +00:00
|
|
|
return Err(TimestampOutOfRange);
|
2023-01-14 13:40:04 +00:00
|
|
|
}
|
|
|
|
LocalResult::Single(x) => x,
|
|
|
|
LocalResult::Ambiguous(y, _z) => y,
|
|
|
|
};
|
|
|
|
|
2024-02-28 05:23:52 +00:00
|
|
|
Ok(utc.with_timezone(
|
|
|
|
&FixedOffset::east_opt(context.tz_offset * 60)
|
|
|
|
.unwrap_or_else(|| FixedOffset::east_opt(0).unwrap()),
|
|
|
|
))
|
2023-01-14 13:40:04 +00:00
|
|
|
}
|
|
|
|
|
2024-02-28 05:23:52 +00:00
|
|
|
pub fn format_absolute_timestamp(timestamp: &Timestamp) -> Result<String, TimestampOutOfRange> {
|
2023-03-14 13:28:22 +00:00
|
|
|
static DEFAULT_FORMAT: Lazy<FormattingItems> =
|
|
|
|
Lazy::new(|| FormattingItems::parse("%Y-%m-%d %H:%M:%S.%3f %:z").unwrap());
|
|
|
|
format_absolute_timestamp_with(timestamp, &DEFAULT_FORMAT)
|
|
|
|
}
|
|
|
|
|
2024-02-28 05:23:52 +00:00
|
|
|
pub fn format_absolute_timestamp_with(
|
|
|
|
timestamp: &Timestamp,
|
|
|
|
format: &FormattingItems,
|
|
|
|
) -> Result<String, TimestampOutOfRange> {
|
|
|
|
let datetime = datetime_from_timestamp(timestamp)?;
|
|
|
|
Ok(datetime.format_with_items(format.items.iter()).to_string())
|
2023-01-14 13:40:04 +00:00
|
|
|
}
|
|
|
|
|
2024-02-28 05:23:52 +00:00
|
|
|
pub fn format_duration(
|
|
|
|
from: &Timestamp,
|
|
|
|
to: &Timestamp,
|
|
|
|
format: &timeago::Formatter,
|
|
|
|
) -> Result<String, TimestampOutOfRange> {
|
|
|
|
let duration = datetime_from_timestamp(to)?
|
|
|
|
.signed_duration_since(datetime_from_timestamp(from)?)
|
|
|
|
.to_std()
|
|
|
|
.map_err(|_: chrono::OutOfRangeError| TimestampOutOfRange)?;
|
|
|
|
Ok(format.convert(duration))
|
2023-01-14 13:40:04 +00:00
|
|
|
}
|