hifitime 4.3.0

Ultra-precise date and time handling in Rust for scientific applications with leap second support
Documentation
/*
* Hifitime
* Copyright (C) 2017-onward Christopher Rabotin <christopher.rabotin@gmail.com> et al. (cf. https://github.com/nyx-space/hifitime/graphs/contributors)
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*
* Documentation: https://nyxspace.com/
*/

use core::fmt;

use crate::{parser::Token, Duration, Epoch, TimeScale};

use super::format::Format;

#[cfg(not(feature = "std"))]
#[allow(unused_imports)] // Import is indeed used.
use num_traits::Float;
#[cfg_attr(kani, derive(kani::Arbitrary))]
#[derive(Copy, Clone, Default, Debug, PartialEq)]
pub(crate) struct Item {
    pub(crate) token: Token,
    pub(crate) sep_char: Option<char>,
    pub(crate) second_sep_char: Option<char>,
    /// If set to true, then only a non-zero value is printed.
    pub(crate) optional: bool,
}

impl Item {
    pub(crate) fn new(token: Token, sep_char: Option<char>, second_sep_char: Option<char>) -> Self {
        let mut me = Self {
            token,
            sep_char,
            second_sep_char,
            optional: false,
        };

        // Maybe the user provided the question mark first
        if let Some(char) = me.sep_char {
            if char == '?' {
                me.sep_char = None;
                me.optional = true;
            }
        }

        if let Some(char) = me.second_sep_char {
            if char == '?' {
                me.second_sep_char = None;
                me.optional = true;
            }
        }

        // Finally rearrange if needed.
        if me.sep_char.is_none() && me.second_sep_char.is_some() {
            core::mem::swap(&mut me.sep_char, &mut me.second_sep_char);
        }

        me
    }

    pub(crate) fn sep_char_is(&self, c_in: char) -> bool {
        self.sep_char.is_some() && self.sep_char.unwrap() == c_in
    }

    pub(crate) fn sep_char_is_not(&self, c_in: char) -> bool {
        self.sep_char.is_some() && self.sep_char.unwrap() != c_in
    }

    pub(crate) fn second_sep_char_is(&self, c_in: char) -> bool {
        self.second_sep_char.is_some() && self.second_sep_char.unwrap() == c_in
    }

    pub(crate) fn second_sep_char_is_not(&self, c_in: char) -> bool {
        self.second_sep_char.is_some() && self.second_sep_char.unwrap() != c_in
    }
}

#[cfg_attr(kani, derive(kani::Arbitrary))]
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct Formatter {
    epoch: Epoch,
    offset: Duration,
    format: Format,
}

impl Formatter {
    pub fn new(epoch: Epoch, format: Format) -> Self {
        Self {
            epoch,
            offset: Duration::ZERO,
            format,
        }
    }

    pub fn with_timezone(epoch: Epoch, offset: Duration, format: Format) -> Self {
        Self {
            epoch: epoch + offset,
            offset,
            format,
        }
    }

    pub fn to_time_scale(epoch: Epoch, format: Format, time_scale: TimeScale) -> Self {
        Self::new(epoch.to_time_scale(time_scale), format)
    }

    pub fn set_timezone(&mut self, offset: Duration) {
        self.offset = offset;
    }
}

impl fmt::Display for Formatter {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        // We make sure to only call this as needed.
        let write_sep = |f: &mut fmt::Formatter, i: usize, format: &Format| -> fmt::Result {
            if i > 0 {
                // Print the first separation character of the previous item
                if let Some(sep) = format.items[i - 1].unwrap().sep_char {
                    write!(f, "{sep}")?;
                }
                // Print the second separation character
                if let Some(sep) = format.items[i - 1].unwrap().second_sep_char {
                    write!(f, "{sep}")?;
                }
            }
            Ok(())
        };

        if self.format.need_gregorian() {
            // This is a specific branch so we don't recompute the gregorian information for each token.
            let (y, mm, dd, hh, min, s, nanos) =
                Epoch::compute_gregorian(self.epoch.duration, self.epoch.time_scale);
            // And format.
            for (i, maybe_item) in self
                .format
                .items
                .iter()
                .enumerate()
                .take(self.format.num_items)
            {
                let item = maybe_item.as_ref().unwrap();

                match item.token {
                    Token::Year => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{y:04}")?
                    }
                    Token::YearShort => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{y:02}")?
                    }
                    Token::Month => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{mm:02}")?
                    }
                    Token::Day => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{dd:02}")?
                    }
                    Token::Hour => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{hh:02}")?
                    }
                    Token::Minute => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{min:02}")?
                    }
                    Token::Second => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{s:02}")?
                    }
                    Token::Subsecond => {
                        if !item.optional || nanos > 0 {
                            write_sep(f, i, &self.format)?;
                            write!(f, "{nanos:09}")?
                        }
                    }
                    Token::OffsetHours => {
                        write_sep(f, i, &self.format)?;
                        let (sign, days, mut hours, minutes, seconds, _, _, _) =
                            self.offset.decompose();

                        if days > 0 {
                            hours += 24 * days;
                        }

                        write!(
                            f,
                            "{}{:02}:{:02}",
                            if sign >= 0 { '+' } else { '-' },
                            hours,
                            minutes
                        )?;

                        if seconds > 0 {
                            write!(f, "{seconds:02}")?;
                        }
                    }
                    Token::OffsetMinutes => {
                        // To print the offset, someone should use OffsetHours, so return an error here.
                        return Err(fmt::Error);
                    }
                    Token::Timescale => {
                        if !item.optional || self.epoch.time_scale != TimeScale::UTC {
                            write_sep(f, i, &self.format)?;
                            write!(f, "{}", self.epoch.time_scale)?;
                        }
                    }
                    Token::DayOfYearInteger => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{:03}", self.epoch.day_of_year().floor() as u16)?
                    }
                    Token::DayOfYear => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.day_of_year())?
                    }
                    Token::Weekday => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.weekday())?
                    }
                    Token::WeekdayShort => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{:x}", self.epoch.weekday())?
                    }
                    Token::WeekdayDecimal => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.weekday().to_c89_weekday())?
                    }
                    Token::MonthName => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.month_name())?
                    }
                    Token::MonthNameShort => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{:x}", self.epoch.month_name())?
                    }
                };
            }
        } else {
            for (i, maybe_item) in self
                .format
                .items
                .iter()
                .enumerate()
                .take(self.format.num_items)
            {
                let item = maybe_item.as_ref().unwrap();
                match item.token {
                    Token::OffsetHours => {
                        write_sep(f, i, &self.format)?;
                        let (sign, days, mut hours, minutes, seconds, _, _, _) =
                            self.offset.decompose();

                        if days > 0 {
                            hours += 24 * days;
                        }

                        write!(
                            f,
                            "{}{:02}:{:02}",
                            if sign >= 0 { '+' } else { '-' },
                            hours,
                            minutes
                        )?;

                        if seconds > 0 {
                            write!(f, "{seconds:02}")?;
                        }
                    }
                    Token::OffsetMinutes => {
                        // To print the offset, someone should use OffsetHours, so return an error here.
                        return Err(fmt::Error);
                    }
                    Token::Timescale => {
                        if !item.optional || self.epoch.time_scale != TimeScale::UTC {
                            write_sep(f, i, &self.format)?;
                            write!(f, "{}", self.epoch.time_scale)?;
                        }
                    }
                    Token::DayOfYearInteger => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{:03}", self.epoch.day_of_year().floor() as u16)?
                    }
                    Token::DayOfYear => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.day_of_year())?
                    }
                    Token::WeekdayDecimal => {
                        write_sep(f, i, &self.format)?;
                        write!(f, "{}", self.epoch.weekday().to_c89_weekday())?
                    }
                    _ => unreachable!(),
                };

                if let Some(sep) = item.sep_char {
                    write!(f, "{sep}")?;
                } else if let Some(sep) = item.second_sep_char {
                    write!(f, "{sep}")?;
                }
            }
        }
        Ok(())
    }
}