blob: 639ed58faad77c1dd9612fa6c6f49f7b4b4b3284 [file] [log] [blame]
use time::{format_description::FormatItem, macros::format_description};
use crate::{time::Format, Time};
/// E.g. `2018-12-24`
pub const SHORT: &[FormatItem<'_>] = format_description!("[year]-[month]-[day]");
/// E.g. `Thu, 18 Aug 2022 12:45:06 +0800`
pub const RFC2822: &[FormatItem<'_>] = format_description!(
"[weekday repr:short], [day] [month repr:short] [year] [hour]:[minute]:[second] [offset_hour sign:mandatory][offset_minute]"
);
/// E.g. `Thu, 8 Aug 2022 12:45:06 +0800`. This is output by `git log --pretty=%aD`.
pub const GIT_RFC2822: &[FormatItem<'_>] = format_description!(
"[weekday repr:short], \
[day padding:none] \
[month repr:short] \
[year] \
[hour]:[minute]:[second] \
[offset_hour sign:mandatory][offset_minute]"
);
/// E.g. `2022-08-17 22:04:58 +0200`
pub const ISO8601: &[FormatItem<'_>] =
format_description!("[year]-[month]-[day] [hour]:[minute]:[second] [offset_hour sign:mandatory][offset_minute]");
/// E.g. `2022-08-17T21:43:13+08:00`
pub const ISO8601_STRICT: &[FormatItem<'_>] =
format_description!("[year]-[month]-[day]T[hour]:[minute]:[second][offset_hour sign:mandatory]:[offset_minute]");
/// E.g. `123456789`
pub const UNIX: Format<'static> = Format::Unix;
/// E.g. `1660874655 +0800`
pub const RAW: Format<'static> = Format::Raw;
/// E.g. `Thu Sep 04 2022 10:45:06 -0400`, like the git `DEFAULT`, but with the year and time fields swapped.
pub const GITOXIDE: &[FormatItem<'_>] = format_description!(
"[weekday repr:short] [month repr:short] [day] [year] [hour]:[minute]:[second] [offset_hour sign:mandatory][offset_minute]"
);
/// E.g. `Thu Sep 4 10:45:06 2022 -0400`. This is output by `git log --pretty=%ad`.
pub const DEFAULT: &[FormatItem<'_>] = format_description!(
"[weekday repr:short] \
[month repr:short] \
[day padding:none] \
[hour]:[minute]:[second] \
[year] \
[offset_hour sign:mandatory][offset_minute]"
);
mod format_impls {
use time::format_description::FormatItem;
use crate::time::Format;
impl<'a> From<&'a [FormatItem<'a>]> for Format<'a> {
fn from(f: &'a [FormatItem<'a>]) -> Self {
Format::Custom(f)
}
}
}
/// Formatting
impl Time {
/// Format this instance according to the given `format`.
///
/// Use the [`format_description`](https://time-rs.github.io/book/api/format-description.html) macro to create and
/// validate formats at compile time, courtesy of the [`time`] crate.
pub fn format<'a>(&self, format: impl Into<Format<'a>>) -> String {
self.format_inner(format.into())
}
fn format_inner(&self, format: Format<'_>) -> String {
match format {
Format::Custom(format) => self
.to_time()
.format(&format)
.expect("well-known format into memory never fails"),
Format::Unix => self.seconds.to_string(),
Format::Raw => self.to_bstring().to_string(),
}
}
}
impl Time {
fn to_time(self) -> time::OffsetDateTime {
time::OffsetDateTime::from_unix_timestamp(self.seconds)
.expect("always valid unix time")
.to_offset(time::UtcOffset::from_whole_seconds(self.offset).expect("valid offset"))
}
}