facet-core 0.46.0

Core reflection traits and types for the facet ecosystem - provides the Facet trait, Shape metadata, and type-erased pointers
Documentation
#![cfg(feature = "time")]

use alloc::string::String;
use time::{OffsetDateTime, UtcDateTime};

use crate::{
    Def, Facet, OxPtrConst, OxPtrUninit, ParseError, PtrConst, Shape, ShapeBuilder, TryFromOutcome,
    Type, UserType, VTableIndirect,
};

// UtcDateTime implementation

unsafe fn utc_display(
    source: OxPtrConst,
    f: &mut core::fmt::Formatter<'_>,
) -> Option<core::fmt::Result> {
    unsafe {
        let udt = source.get::<UtcDateTime>();
        Some(
            match udt.format(&time::format_description::well_known::Rfc3339) {
                Ok(s) => write!(f, "{s}"),
                Err(_) => write!(f, "<invalid UtcDateTime>"),
            },
        )
    }
}

unsafe fn utc_try_from(
    target: OxPtrUninit,
    src_shape: &'static Shape,
    src: PtrConst,
) -> TryFromOutcome {
    unsafe {
        if src_shape.id == <String as Facet>::SHAPE.id {
            let source_str = src.read::<String>();
            match UtcDateTime::parse(&source_str, &time::format_description::well_known::Rfc3339) {
                Ok(val) => {
                    target.put(val);
                    TryFromOutcome::Converted
                }
                Err(_) => TryFromOutcome::Failed("could not parse date".into()),
            }
        } else {
            TryFromOutcome::Unsupported
        }
    }
}

unsafe fn utc_parse(s: &str, target: OxPtrUninit) -> Option<Result<(), ParseError>> {
    unsafe {
        let parsed = UtcDateTime::parse(s, &time::format_description::well_known::Rfc3339)
            .map_err(|_| ParseError::from_str("could not parse date"));
        Some(match parsed {
            Ok(val) => {
                target.put(val);
                Ok(())
            }
            Err(e) => Err(e),
        })
    }
}

unsafe fn utc_partial_eq(a: OxPtrConst, b: OxPtrConst) -> Option<bool> {
    unsafe {
        let a = a.get::<UtcDateTime>();
        let b = b.get::<UtcDateTime>();
        Some(a == b)
    }
}

const UTC_VTABLE: VTableIndirect = VTableIndirect {
    display: Some(utc_display),
    try_from: Some(utc_try_from),
    parse: Some(utc_parse),
    partial_eq: Some(utc_partial_eq),
    ..VTableIndirect::EMPTY
};

unsafe impl Facet<'_> for UtcDateTime {
    const SHAPE: &'static Shape = &const {
        ShapeBuilder::for_sized::<UtcDateTime>("UtcDateTime")
            .module_path("time")
            .ty(Type::User(UserType::Opaque))
            .def(Def::Scalar)
            .vtable_indirect(&UTC_VTABLE)
            .build()
    };
}

// OffsetDateTime implementation

unsafe fn offset_display(
    source: OxPtrConst,
    f: &mut core::fmt::Formatter<'_>,
) -> Option<core::fmt::Result> {
    unsafe {
        let odt = source.get::<OffsetDateTime>();
        Some(
            match odt.format(&time::format_description::well_known::Rfc3339) {
                Ok(s) => write!(f, "{s}"),
                Err(_) => write!(f, "<invalid OffsetDateTime>"),
            },
        )
    }
}

unsafe fn offset_try_from(
    target: OxPtrUninit,
    src_shape: &'static Shape,
    src: PtrConst,
) -> TryFromOutcome {
    unsafe {
        if src_shape.id == <String as Facet>::SHAPE.id {
            let source_str = src.read::<String>();
            match OffsetDateTime::parse(&source_str, &time::format_description::well_known::Rfc3339)
            {
                Ok(val) => {
                    target.put(val);
                    TryFromOutcome::Converted
                }
                Err(_) => TryFromOutcome::Failed("could not parse date".into()),
            }
        } else {
            TryFromOutcome::Unsupported
        }
    }
}

unsafe fn offset_parse(s: &str, target: OxPtrUninit) -> Option<Result<(), ParseError>> {
    unsafe {
        let parsed = OffsetDateTime::parse(s, &time::format_description::well_known::Rfc3339)
            .map_err(|_| ParseError::from_str("could not parse date"));
        Some(match parsed {
            Ok(val) => {
                target.put(val);
                Ok(())
            }
            Err(e) => Err(e),
        })
    }
}

unsafe fn offset_partial_eq(a: OxPtrConst, b: OxPtrConst) -> Option<bool> {
    unsafe {
        let a = a.get::<OffsetDateTime>();
        let b = b.get::<OffsetDateTime>();
        Some(a == b)
    }
}

const OFFSET_VTABLE: VTableIndirect = VTableIndirect {
    display: Some(offset_display),
    try_from: Some(offset_try_from),
    parse: Some(offset_parse),
    partial_eq: Some(offset_partial_eq),
    ..VTableIndirect::EMPTY
};

unsafe impl Facet<'_> for OffsetDateTime {
    const SHAPE: &'static Shape = &const {
        ShapeBuilder::for_sized::<OffsetDateTime>("OffsetDateTime")
            .module_path("time")
            .ty(Type::User(UserType::Opaque))
            .def(Def::Scalar)
            .vtable_indirect(&OFFSET_VTABLE)
            .build()
    };
}

#[cfg(test)]
mod tests {
    use core::fmt;

    use time::OffsetDateTime;

    use crate::{Facet, PtrConst};

    #[test]
    fn parse_offset_date_time() {
        facet_testhelpers::setup();

        let target = OffsetDateTime::SHAPE.allocate().unwrap();
        unsafe {
            OffsetDateTime::SHAPE
                .call_parse("2023-03-14T15:09:26Z", target)
                .unwrap()
                .unwrap();
        }
        let odt: OffsetDateTime = unsafe { target.assume_init().read() };
        assert_eq!(
            odt,
            OffsetDateTime::parse(
                "2023-03-14T15:09:26Z",
                &time::format_description::well_known::Rfc3339
            )
            .unwrap()
        );

        {
            struct DisplayWrapper(PtrConst);

            impl fmt::Display for DisplayWrapper {
                fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
                    unsafe { OffsetDateTime::SHAPE.call_display(self.0, f).unwrap() }
                }
            }

            let s = format!(
                "{}",
                DisplayWrapper(unsafe { target.assume_init().as_const() })
            );
            assert_eq!(s, "2023-03-14T15:09:26Z");
        }

        // Deallocate the heap allocation to avoid memory leaks under Miri
        unsafe {
            OffsetDateTime::SHAPE.deallocate_uninit(target).unwrap();
        }
    }
}