mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer-rs.git
synced 2025-01-11 09:45:26 +00:00
283 lines
10 KiB
Rust
283 lines
10 KiB
Rust
// Copyright (C) 2018 François Laignel <fengalin@free.fr>
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
use std::convert::{TryFrom, TryInto};
|
|
|
|
use glib::translate::{FromGlib, ToGlib};
|
|
use glib::value::{SetValue, SetValueOptional};
|
|
use glib::StaticType;
|
|
|
|
use serde::de::{Deserialize, Deserializer, Error};
|
|
use serde::ser;
|
|
use serde::ser::{Serialize, Serializer};
|
|
use DateTime;
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
enum DateTimeVariants {
|
|
Y(i32),
|
|
YM(i32, i32),
|
|
YMD(i32, i32, i32),
|
|
YMDhmTz(i32, i32, i32, i32, i32, f32),
|
|
YMDhmsTz(i32, i32, i32, i32, i32, f64, f32),
|
|
}
|
|
|
|
// Note: ser / de for `glib::Date` should be implemented in the `glib` crate
|
|
// However, there is no `ser_de` feature in `glib` right now. The limitation is that
|
|
// `Date` fields can only be ser / de when they are used in `Value`s (which implies
|
|
// `Array`s, `List`s, `Structure` fields and `Tag`s)
|
|
pub(crate) struct Date(glib::Date);
|
|
|
|
impl From<glib::Date> for Date {
|
|
fn from(glib_date: glib::Date) -> Self {
|
|
skip_assert_initialized!();
|
|
Date(glib_date)
|
|
}
|
|
}
|
|
|
|
impl SetValue for Date {
|
|
unsafe fn set_value(value: &mut glib::Value, this: &Self) {
|
|
glib::value::SetValue::set_value(value, &this.0);
|
|
}
|
|
}
|
|
|
|
impl SetValueOptional for Date {
|
|
unsafe fn set_value_optional(value: &mut glib::Value, this: Option<&Self>) {
|
|
glib::value::SetValueOptional::set_value_optional(value, this.map(|this| &this.0));
|
|
}
|
|
}
|
|
|
|
impl StaticType for Date {
|
|
fn static_type() -> glib::Type {
|
|
glib::Date::static_type()
|
|
}
|
|
}
|
|
|
|
impl<'a> Serialize for Date {
|
|
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
DateTimeVariants::YMD(
|
|
self.0.get_year() as i32,
|
|
self.0.get_month().to_glib() as i32,
|
|
self.0.get_day() as i32,
|
|
)
|
|
.serialize(serializer)
|
|
}
|
|
}
|
|
|
|
impl<'a> Serialize for DateTime {
|
|
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
let variant = if self.has_second() {
|
|
DateTimeVariants::YMDhmsTz(
|
|
self.get_year(),
|
|
self.get_month(),
|
|
self.get_day(),
|
|
self.get_hour(),
|
|
self.get_minute(),
|
|
f64::from(self.get_second()) + f64::from(self.get_microsecond()) / 1_000_000f64,
|
|
self.get_time_zone_offset(),
|
|
)
|
|
} else if self.has_time() {
|
|
DateTimeVariants::YMDhmTz(
|
|
self.get_year(),
|
|
self.get_month(),
|
|
self.get_day(),
|
|
self.get_hour(),
|
|
self.get_minute(),
|
|
self.get_time_zone_offset(),
|
|
)
|
|
} else if self.has_day() {
|
|
DateTimeVariants::YMD(self.get_year(), self.get_month(), self.get_day())
|
|
} else if self.has_month() {
|
|
DateTimeVariants::YM(self.get_year(), self.get_month())
|
|
} else if self.has_year() {
|
|
DateTimeVariants::Y(self.get_year())
|
|
} else {
|
|
return Err(ser::Error::custom(format!(
|
|
"no parts could be found in `DateTime` {}",
|
|
self,
|
|
)));
|
|
};
|
|
|
|
variant.serialize(serializer)
|
|
}
|
|
}
|
|
|
|
impl TryFrom<DateTimeVariants> for Date {
|
|
type Error = glib::BoolError;
|
|
|
|
fn try_from(dt_variant: DateTimeVariants) -> Result<Self, Self::Error> {
|
|
skip_assert_initialized!();
|
|
match dt_variant {
|
|
DateTimeVariants::YMD(y, m, d) => {
|
|
let month = glib::DateMonth::from_glib(m);
|
|
if let glib::DateMonth::__Unknown(_) = month {
|
|
return Err(glib_bool_error!("Out of range `month` for `Date`"));
|
|
}
|
|
|
|
Ok(Date(glib::Date::new_dmy(
|
|
d.try_into()
|
|
.map_err(|_| glib_bool_error!("Out of range `day` for `Date`"))?,
|
|
month,
|
|
y.try_into()
|
|
.map_err(|_| glib_bool_error!("Out of range `year` for `Date`"))?,
|
|
)))
|
|
}
|
|
_ => Err(glib_bool_error!(
|
|
"Incompatible variant for `Date` (expecting \"YMD\")"
|
|
)),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'de> Deserialize<'de> for Date {
|
|
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
|
skip_assert_initialized!();
|
|
DateTimeVariants::deserialize(deserializer)
|
|
.and_then(|dt_variant| dt_variant.try_into().map_err(D::Error::custom))
|
|
}
|
|
}
|
|
|
|
#[allow(clippy::many_single_char_names)]
|
|
impl TryFrom<DateTimeVariants> for DateTime {
|
|
type Error = glib::BoolError;
|
|
|
|
fn try_from(dt_variant: DateTimeVariants) -> Result<Self, Self::Error> {
|
|
skip_assert_initialized!();
|
|
match dt_variant {
|
|
DateTimeVariants::Y(y) => DateTime::new_y(y),
|
|
DateTimeVariants::YM(y, m) => DateTime::new_ym(y, m),
|
|
DateTimeVariants::YMD(y, m, d) => DateTime::new_ymd(y, m, d),
|
|
DateTimeVariants::YMDhmTz(y, m, d, h, mn, tz) => {
|
|
DateTime::new(tz, y, m, d, h, mn, -1f64)
|
|
}
|
|
DateTimeVariants::YMDhmsTz(y, m, d, h, mn, s, tz) => {
|
|
DateTime::new(tz, y, m, d, h, mn, s)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'de> Deserialize<'de> for DateTime {
|
|
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
|
skip_assert_initialized!();
|
|
DateTimeVariants::deserialize(deserializer)
|
|
.and_then(|dt_variant| dt_variant.try_into().map_err(D::Error::custom))
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
extern crate ron;
|
|
extern crate serde_json;
|
|
|
|
use DateTime;
|
|
|
|
#[test]
|
|
fn test_serialize() {
|
|
::init().unwrap();
|
|
|
|
let mut pretty_config = ron::ser::PrettyConfig::default();
|
|
pretty_config.new_line = "".to_string();
|
|
|
|
let datetime = DateTime::new(2f32, 2018, 5, 28, 16, 6, 42.123_456f64).unwrap();
|
|
let res = ron::ser::to_string_pretty(&datetime, pretty_config.clone());
|
|
assert_eq!(
|
|
Ok("YMDhmsTz(2018, 5, 28, 16, 6, 42.123456, 2)".to_owned()),
|
|
res,
|
|
);
|
|
|
|
let res = serde_json::to_string(&datetime).unwrap();
|
|
assert_eq!(
|
|
r#"{"YMDhmsTz":[2018,5,28,16,6,42.123456,2.0]}"#.to_owned(),
|
|
res
|
|
);
|
|
|
|
let datetime = DateTime::new(2f32, 2018, 5, 28, 16, 6, -1f64).unwrap();
|
|
let res = ron::ser::to_string_pretty(&datetime, pretty_config.clone());
|
|
assert_eq!(Ok("YMDhmTz(2018, 5, 28, 16, 6, 2)".to_owned()), res,);
|
|
|
|
let datetime = DateTime::new_ymd(2018, 5, 28).unwrap();
|
|
let res = ron::ser::to_string_pretty(&datetime, pretty_config.clone());
|
|
assert_eq!(Ok("YMD(2018, 5, 28)".to_owned()), res);
|
|
|
|
let datetime = DateTime::new_ym(2018, 5).unwrap();
|
|
let res = ron::ser::to_string_pretty(&datetime, pretty_config.clone());
|
|
assert_eq!(Ok("YM(2018, 5)".to_owned()), res);
|
|
|
|
let datetime = DateTime::new_y(2018).unwrap();
|
|
let res = ron::ser::to_string_pretty(&datetime, pretty_config);
|
|
assert_eq!(Ok("Y(2018)".to_owned()), res);
|
|
}
|
|
|
|
#[test]
|
|
fn test_deserialize() {
|
|
::init().unwrap();
|
|
|
|
let datetime_ron = "YMDhmsTz(2018, 5, 28, 16, 6, 42.123456, 2)";
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ron).unwrap();
|
|
assert_eq!(
|
|
datetime_de,
|
|
DateTime::new(2f32, 2018, 5, 28, 16, 6, 42.123_456f64).unwrap()
|
|
);
|
|
|
|
let datetime_json = r#"{"YMDhmsTz":[2018,5,28,16,6,42.123456,2.0]}"#;
|
|
let datetime_de: DateTime = serde_json::from_str(datetime_json).unwrap();
|
|
assert_eq!(
|
|
datetime_de,
|
|
DateTime::new(2f32, 2018, 5, 28, 16, 6, 42.123_456f64).unwrap()
|
|
);
|
|
|
|
let datetime_ron = "YMDhmTz(2018, 5, 28, 16, 6, 2)";
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ron).unwrap();
|
|
assert_eq!(
|
|
datetime_de,
|
|
DateTime::new(2f32, 2018, 5, 28, 16, 6, -1f64).unwrap()
|
|
);
|
|
|
|
let datetime_ron = "YMD(2018, 5, 28)";
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ron).unwrap();
|
|
assert_eq!(datetime_de, DateTime::new_ymd(2018, 5, 28).unwrap());
|
|
|
|
let datetime_ron = "YM(2018, 5)";
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ron).unwrap();
|
|
assert_eq!(datetime_de, DateTime::new_ym(2018, 5).unwrap());
|
|
|
|
let datetime_ron = "Y(2018)";
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ron).unwrap();
|
|
assert_eq!(datetime_de, DateTime::new_y(2018).unwrap());
|
|
}
|
|
|
|
#[test]
|
|
fn test_serde_roundtrip() {
|
|
::init().unwrap();
|
|
|
|
let datetime = DateTime::new(2f32, 2018, 5, 28, 16, 6, 42.123_456f64).unwrap();
|
|
let datetime_ser = ron::ser::to_string(&datetime).unwrap();
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ser.as_str()).unwrap();
|
|
assert_eq!(datetime_de, datetime);
|
|
|
|
let datetime = DateTime::new(2f32, 2018, 5, 28, 16, 6, -1f64).unwrap();
|
|
let datetime_ser = ron::ser::to_string(&datetime).unwrap();
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ser.as_str()).unwrap();
|
|
assert_eq!(datetime_de, datetime);
|
|
|
|
let datetime = DateTime::new_ymd(2018, 5, 28).unwrap();
|
|
let datetime_ser = ron::ser::to_string(&datetime).unwrap();
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ser.as_str()).unwrap();
|
|
assert_eq!(datetime_de, datetime);
|
|
|
|
let datetime = DateTime::new_ym(2018, 5).unwrap();
|
|
let datetime_ser = ron::ser::to_string(&datetime).unwrap();
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ser.as_str()).unwrap();
|
|
assert_eq!(datetime_de, datetime);
|
|
|
|
let datetime = DateTime::new_y(2018).unwrap();
|
|
let datetime_ser = ron::ser::to_string(&datetime).unwrap();
|
|
let datetime_de: DateTime = ron::de::from_str(datetime_ser.as_str()).unwrap();
|
|
assert_eq!(datetime_de, datetime);
|
|
}
|
|
}
|