use {
crate::{table::TomlTable, text::CowSpan},
std::ops::{Deref, DerefMut},
};
#[derive(Debug, PartialEq)]
pub struct TomlArray<'a> {
pub(crate) values: Vec<TomlValue<'a>>,
pub(crate) is_array_of_tables: bool,
}
impl<'a> Deref for TomlArray<'a> {
type Target = [TomlValue<'a>];
fn deref(&self) -> &Self::Target {
&self.values
}
}
impl<'a> DerefMut for TomlArray<'a> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.values
}
}
#[derive(Debug, PartialEq)]
pub enum TomlValue<'a> {
String(CowSpan<'a>),
Integer(i64),
Float(f64),
Boolean(bool),
Time(TomlTime),
Date(TomlDate),
DateTime(TomlDateTime),
OffsetDateTime(OffsetTomlDateTime),
Array(TomlArray<'a>),
Table(TomlTable<'a>),
}
impl<'a> TomlValue<'a> {
pub fn ty(&self) -> TomlValueType {
match *self {
Self::String(_) => TomlValueType::String,
Self::Integer(_) => TomlValueType::Integer,
Self::Float(_) => TomlValueType::Float,
Self::Boolean(_) => TomlValueType::Boolean,
Self::Time(_) => TomlValueType::Time,
Self::Date(_) => TomlValueType::Date,
Self::DateTime(_) => TomlValueType::DateTime,
Self::OffsetDateTime(_) => TomlValueType::OffsetDateTime,
Self::Array(_) => TomlValueType::Array,
Self::Table(_) => TomlValueType::Table,
}
}
pub fn as_string(&self) -> Option<&str> {
match self {
Self::String(string) => Some(string.as_str()),
_ => None,
}
}
pub fn as_integer(&self) -> Option<i64> {
match self {
Self::Integer(num) => Some(*num),
_ => None,
}
}
pub fn as_float(&self) -> Option<f64> {
match self {
Self::Float(num) => Some(*num),
_ => None,
}
}
pub fn as_bool(&self) -> Option<bool> {
match self {
Self::Boolean(bool) => Some(*bool),
_ => None,
}
}
pub fn as_array(&self) -> Option<&TomlArray<'a>> {
match self {
Self::Array(array) => Some(array),
_ => None,
}
}
pub fn as_table(&self) -> Option<&TomlTable<'a>> {
match self {
Self::Table(table) => Some(table),
_ => None,
}
}
pub fn as_date(&self) -> Option<TomlDate> {
match self {
Self::Date(date) => Some(*date),
_ => None,
}
}
pub fn as_time(&self) -> Option<TomlTime> {
match self {
Self::Time(time) => Some(*time),
_ => None,
}
}
pub fn as_datetime(&self) -> Option<TomlDateTime> {
match self {
Self::DateTime(datetime) => Some(*datetime),
_ => None,
}
}
pub fn as_offset_datetime(&self) -> Option<OffsetTomlDateTime> {
match self {
Self::OffsetDateTime(offset_datetime) => Some(*offset_datetime),
_ => None,
}
}
pub fn coerce_bool(&self) -> Option<bool> {
match self {
Self::Boolean(bool) => Some(*bool),
Self::String(str) => {
let str = str.as_str();
match str {
"true" | "True" => Some(true),
"false" | "False" => Some(false),
_ => None,
}
}
Self::Integer(int) => {
if *int == 0 {
Some(false)
} else if *int == 1 {
Some(true)
} else {
None
}
}
Self::Float(float) => {
if *float == 0.0 {
Some(false)
} else if *float == 1.0 {
Some(true)
} else {
None
}
}
_ => None,
}
}
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
#[allow(missing_docs)]
pub enum TomlValueType {
String,
Integer,
Float,
Boolean,
Time,
Date,
DateTime,
OffsetDateTime,
Array,
Table,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct TomlOffset {
pub hour: i8,
pub minute: u8,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct TomlDate {
pub year: u16,
pub month: u8,
pub month_day: u8,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct TomlTime {
pub hour: u8,
pub minute: u8,
pub second: u8,
pub nanosecond: u32,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct TomlDateTime {
pub date: TomlDate,
pub time: TomlTime,
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub struct OffsetTomlDateTime {
pub offset: TomlOffset,
pub date: TomlDate,
pub time: TomlTime,
}
#[cfg(any(test, feature = "chrono"))]
mod chrono_into_from {
use {
super::*,
chrono::{DateTime, FixedOffset, NaiveDate, NaiveDateTime, NaiveTime},
};
impl TryInto<NaiveDate> for TomlDate {
type Error = ();
fn try_into(self) -> Result<NaiveDate, Self::Error> {
NaiveDate::from_ymd_opt(self.year.into(), self.month.into(), self.month_day.into())
.ok_or(())
}
}
impl TryInto<NaiveTime> for TomlTime {
type Error = ();
fn try_into(self) -> Result<NaiveTime, Self::Error> {
NaiveTime::from_hms_nano_opt(
self.hour.into(),
self.minute.into(),
self.second.into(),
self.nanosecond,
)
.ok_or(())
}
}
impl TryInto<FixedOffset> for TomlOffset {
type Error = ();
fn try_into(self) -> Result<FixedOffset, Self::Error> {
let hour: i32 = self.hour.into();
let mut minute: i32 = self.minute.into();
minute *= hour.signum();
FixedOffset::east_opt(
hour.checked_mul(60).ok_or(())?.checked_mul(60).ok_or(())?
+ minute.checked_mul(60).ok_or(())?,
)
.ok_or(())
}
}
impl TryInto<NaiveDateTime> for TomlDateTime {
type Error = ();
fn try_into(self) -> Result<NaiveDateTime, Self::Error> {
let date: NaiveDate = self.date.try_into()?;
Ok(date.and_time(self.time.try_into()?))
}
}
impl TryInto<DateTime<FixedOffset>> for OffsetTomlDateTime {
type Error = ();
fn try_into(self) -> Result<DateTime<FixedOffset>, Self::Error> {
let offset: FixedOffset = self.offset.try_into()?;
let date: NaiveDate = self.date.try_into()?;
let datetime = date.and_time(self.time.try_into()?);
datetime.and_local_timezone(offset).single().ok_or(())
}
}
}