nt_time/serde_with/unix_time/nanoseconds.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
// SPDX-FileCopyrightText: 2024 Shun Sakai
//
// SPDX-License-Identifier: Apache-2.0 OR MIT
//! Use [Unix time] in nanoseconds when serializing and deserializing a
//! [`FileTime`].
//!
//! Use this module in combination with Serde's [`with`] attribute.
//!
//! # Examples
//!
//! ```
//! use nt_time::{
//! serde::{Deserialize, Serialize},
//! serde_with::unix_time,
//! FileTime,
//! };
//!
//! #[derive(Deserialize, Serialize)]
//! struct Time {
//! #[serde(with = "unix_time::nanoseconds")]
//! time: FileTime,
//! }
//!
//! let ft = Time {
//! time: FileTime::NT_TIME_EPOCH,
//! };
//! let json = serde_json::to_string(&ft).unwrap();
//! assert_eq!(json, r#"{"time":-11644473600000000000}"#);
//!
//! let ft: Time = serde_json::from_str(&json).unwrap();
//! assert_eq!(ft.time, FileTime::NT_TIME_EPOCH);
//! ```
//!
//! [Unix time]: https://en.wikipedia.org/wiki/Unix_time
//! [`with`]: https://serde.rs/field-attrs.html#with
pub mod option;
use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer};
use crate::FileTime;
#[allow(clippy::missing_errors_doc)]
/// Serializes a [`FileTime`] into the given Serde serializer.
///
/// This serializes using [Unix time] in nanoseconds.
///
/// [Unix time]: https://en.wikipedia.org/wiki/Unix_time
#[inline]
pub fn serialize<S: Serializer>(ft: &FileTime, serializer: S) -> Result<S::Ok, S::Error> {
ft.to_unix_time_nanos().serialize(serializer)
}
#[allow(clippy::missing_errors_doc)]
/// Deserializes a [`FileTime`] from the given Serde deserializer.
///
/// This deserializes from its [Unix time] in nanoseconds.
///
/// [Unix time]: https://en.wikipedia.org/wiki/Unix_time
#[inline]
pub fn deserialize<'de, D: Deserializer<'de>>(deserializer: D) -> Result<FileTime, D::Error> {
FileTime::from_unix_time_nanos(<_>::deserialize(deserializer)?).map_err(D::Error::custom)
}
#[cfg(test)]
mod tests {
use super::*;
#[derive(Debug, Deserialize, Eq, PartialEq, Serialize)]
struct Test {
#[serde(with = "crate::serde_with::unix_time::nanoseconds")]
time: FileTime,
}
#[test]
fn serialize_json() {
assert_eq!(
serde_json::to_string(&Test {
time: FileTime::NT_TIME_EPOCH
})
.unwrap(),
r#"{"time":-11644473600000000000}"#
);
assert_eq!(
serde_json::to_string(&Test {
time: FileTime::UNIX_EPOCH
})
.unwrap(),
r#"{"time":0}"#
);
assert_eq!(
serde_json::to_string(&Test {
time: FileTime::MAX
})
.unwrap(),
r#"{"time":1833029933770955161500}"#
);
}
#[test]
fn deserialize_json() {
assert_eq!(
serde_json::from_str::<Test>(r#"{"time":-11644473600000000000}"#).unwrap(),
Test {
time: FileTime::NT_TIME_EPOCH
}
);
assert_eq!(
serde_json::from_str::<Test>(r#"{"time":0}"#).unwrap(),
Test {
time: FileTime::UNIX_EPOCH
}
);
assert_eq!(
serde_json::from_str::<Test>(r#"{"time":1833029933770955161500}"#).unwrap(),
Test {
time: FileTime::MAX
}
);
}
}