2018-07-16 19:46:10 +00:00
|
|
|
// 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 serde::de::{Deserialize, Deserializer};
|
2018-07-22 14:55:24 +00:00
|
|
|
use serde::ser::{Serialize, SerializeStruct, Serializer};
|
2018-07-16 19:46:10 +00:00
|
|
|
|
|
|
|
use Buffer;
|
|
|
|
use BufferList;
|
|
|
|
use Caps;
|
|
|
|
use GenericFormattedValue;
|
|
|
|
use Sample;
|
|
|
|
use SampleRef;
|
|
|
|
use Segment;
|
|
|
|
use Structure;
|
|
|
|
|
|
|
|
impl<'a> Serialize for SampleRef {
|
|
|
|
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
|
|
let mut sample = serializer.serialize_struct("Sample", 5)?;
|
|
|
|
sample.serialize_field("buffer", &self.get_buffer())?;
|
|
|
|
sample.serialize_field("buffer_list", &self.get_buffer_list())?;
|
|
|
|
sample.serialize_field("caps", &self.get_caps())?;
|
|
|
|
sample.serialize_field("segment", &self.get_segment())?;
|
|
|
|
sample.serialize_field("info", &self.get_info())?;
|
|
|
|
sample.end()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Serialize for Sample {
|
|
|
|
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
|
|
self.as_ref().serialize(serializer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Deserialize)]
|
|
|
|
struct SampleDe {
|
|
|
|
buffer: Option<Buffer>,
|
|
|
|
buffer_list: Option<BufferList>,
|
|
|
|
caps: Option<Caps>,
|
|
|
|
segment: Option<Segment>,
|
|
|
|
info: Option<Structure>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<SampleDe> for Sample {
|
|
|
|
fn from(mut buf_de: SampleDe) -> Self {
|
|
|
|
if buf_de.buffer.is_some() {
|
|
|
|
Sample::new::<GenericFormattedValue>(
|
|
|
|
buf_de.buffer.as_ref(),
|
|
|
|
buf_de.caps.as_ref(),
|
|
|
|
buf_de.segment.as_ref(),
|
|
|
|
buf_de.info.take(),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
Sample::with_buffer_list::<GenericFormattedValue>(
|
|
|
|
buf_de.buffer_list.as_ref(),
|
|
|
|
buf_de.caps.as_ref(),
|
|
|
|
buf_de.segment.as_ref(),
|
|
|
|
buf_de.info.take(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'de> Deserialize<'de> for Sample {
|
|
|
|
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
|
2018-07-29 14:04:39 +00:00
|
|
|
SampleDe::deserialize(deserializer).map(|sample_de| sample_de.into())
|
2018-07-16 19:46:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
extern crate ron;
|
|
|
|
|
2018-07-29 16:18:24 +00:00
|
|
|
use Buffer;
|
|
|
|
use Caps;
|
|
|
|
use ClockTime;
|
|
|
|
use Format;
|
|
|
|
use GenericFormattedValue;
|
2018-07-16 19:46:10 +00:00
|
|
|
use Sample;
|
2018-07-29 16:18:24 +00:00
|
|
|
use Segment;
|
|
|
|
use SegmentFlags;
|
|
|
|
use Structure;
|
2018-07-16 19:46:10 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_serialize() {
|
|
|
|
::init().unwrap();
|
|
|
|
|
|
|
|
let mut pretty_config = ron::ser::PrettyConfig::default();
|
|
|
|
pretty_config.new_line = "".to_string();
|
|
|
|
|
|
|
|
let sample = {
|
|
|
|
let mut buffer = Buffer::from_slice(vec![1, 2, 3, 4]).unwrap();
|
|
|
|
{
|
|
|
|
let buffer = buffer.get_mut().unwrap();
|
|
|
|
buffer.set_pts(1.into());
|
|
|
|
buffer.set_offset(0);
|
|
|
|
buffer.set_offset_end(4);
|
|
|
|
buffer.set_duration(4.into());
|
|
|
|
}
|
|
|
|
|
|
|
|
let caps = Caps::builder("sample/caps")
|
|
|
|
.field("int", &12)
|
|
|
|
.field("bool", &true)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let mut segment = Segment::new();
|
|
|
|
segment.set_flags(SegmentFlags::RESET | SegmentFlags::SEGMENT);
|
|
|
|
segment.set_rate(1f64);
|
|
|
|
segment.set_applied_rate(0.9f64);
|
|
|
|
segment.set_format(Format::Time);
|
|
|
|
segment.set_base(GenericFormattedValue::Time(ClockTime::from_nseconds(123)));
|
|
|
|
segment.set_offset(GenericFormattedValue::Time(ClockTime::from_nseconds(42)));
|
|
|
|
segment.set_start(GenericFormattedValue::Time(ClockTime::from_nseconds(1024)));
|
|
|
|
segment.set_stop(GenericFormattedValue::Time(ClockTime::from_nseconds(2048)));
|
|
|
|
segment.set_time(GenericFormattedValue::Time(ClockTime::from_nseconds(1042)));
|
|
|
|
segment.set_position(GenericFormattedValue::Time(ClockTime::from_nseconds(256)));
|
|
|
|
segment.set_duration(GenericFormattedValue::Time(ClockTime::none()));
|
|
|
|
|
|
|
|
let info = Structure::builder("sample.info")
|
|
|
|
.field("f3", &123i32)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
Sample::new::<GenericFormattedValue>(
|
|
|
|
Some(&buffer),
|
|
|
|
Some(&caps),
|
|
|
|
Some(&segment),
|
|
|
|
Some(info),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
let res = ron::ser::to_string_pretty(&sample, pretty_config.clone());
|
|
|
|
assert_eq!(
|
2018-07-22 14:55:24 +00:00
|
|
|
Ok(concat!(
|
|
|
|
"(",
|
|
|
|
" buffer: Some((",
|
|
|
|
" pts: Some(1),",
|
|
|
|
" dts: None,",
|
|
|
|
" duration: Some(4),",
|
|
|
|
" offset: 0,",
|
|
|
|
" offset_end: 4,",
|
|
|
|
" flags: (",
|
|
|
|
" bits: 0,",
|
|
|
|
" ),",
|
|
|
|
" buffer: \"AQIDBA==\",",
|
|
|
|
" )),",
|
|
|
|
" buffer_list: None,",
|
2018-07-29 14:04:39 +00:00
|
|
|
" caps: Some(Some([",
|
2018-07-22 14:55:24 +00:00
|
|
|
" ((\"sample/caps\", [",
|
|
|
|
" (\"int\", \"i32\", 12),",
|
|
|
|
" (\"bool\", \"bool\", true),",
|
|
|
|
" ]), None),",
|
2018-07-29 14:04:39 +00:00
|
|
|
" ])),",
|
2018-07-22 14:55:24 +00:00
|
|
|
" segment: Some((",
|
|
|
|
" flags: (",
|
|
|
|
" bits: 9,",
|
|
|
|
" ),",
|
|
|
|
" rate: 1,",
|
|
|
|
" applied_rate: 0.9,",
|
|
|
|
" format: Time,",
|
|
|
|
" base: 123,",
|
|
|
|
" offset: 42,",
|
|
|
|
" start: 1024,",
|
|
|
|
" stop: 2048,",
|
|
|
|
" time: 1042,",
|
|
|
|
" position: 256,",
|
|
|
|
" duration: -1,",
|
|
|
|
" )),",
|
|
|
|
" info: Some((\"sample.info\", [",
|
|
|
|
" (\"f3\", \"i32\", 123),",
|
|
|
|
" ])),",
|
|
|
|
")"
|
|
|
|
).to_owned()),
|
2018-07-16 19:46:10 +00:00
|
|
|
res
|
|
|
|
);
|
|
|
|
|
|
|
|
let sample = {
|
|
|
|
let mut buffer = Buffer::from_slice(vec![1, 2, 3, 4]).unwrap();
|
|
|
|
{
|
|
|
|
let buffer = buffer.get_mut().unwrap();
|
|
|
|
buffer.set_pts(1.into());
|
|
|
|
buffer.set_offset(0);
|
|
|
|
buffer.set_offset_end(4);
|
|
|
|
buffer.set_duration(4.into());
|
|
|
|
}
|
|
|
|
Sample::new::<GenericFormattedValue>(Some(&buffer), None, None, None)
|
|
|
|
};
|
|
|
|
|
|
|
|
// `Sample`'s `Segment` is allocated in GStreamer 1.x, should be fixed in version 2.0
|
|
|
|
|
|
|
|
let res = ron::ser::to_string_pretty(&sample, pretty_config);
|
|
|
|
assert_eq!(
|
2018-07-22 14:55:24 +00:00
|
|
|
Ok(concat!(
|
|
|
|
"(",
|
|
|
|
" buffer: Some((",
|
|
|
|
" pts: Some(1),",
|
|
|
|
" dts: None,",
|
|
|
|
" duration: Some(4),",
|
|
|
|
" offset: 0,",
|
|
|
|
" offset_end: 4,",
|
|
|
|
" flags: (",
|
|
|
|
" bits: 0,",
|
|
|
|
" ),",
|
|
|
|
" buffer: \"AQIDBA==\",",
|
|
|
|
" )),",
|
|
|
|
" buffer_list: None,",
|
|
|
|
" caps: None,",
|
|
|
|
" segment: Some((",
|
|
|
|
" flags: (",
|
|
|
|
" bits: 0,",
|
|
|
|
" ),",
|
|
|
|
" rate: 1,",
|
|
|
|
" applied_rate: 1,",
|
|
|
|
" format: Time,",
|
|
|
|
" base: 0,",
|
|
|
|
" offset: 0,",
|
|
|
|
" start: 0,",
|
|
|
|
" stop: -1,",
|
|
|
|
" time: 0,",
|
|
|
|
" position: 0,",
|
|
|
|
" duration: -1,",
|
|
|
|
" )),",
|
|
|
|
" info: None,",
|
|
|
|
")"
|
|
|
|
).to_owned()),
|
2018-07-16 19:46:10 +00:00
|
|
|
res
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_deserialize() {
|
|
|
|
::init().unwrap();
|
|
|
|
|
|
|
|
let buffer_ron = r#"
|
|
|
|
(
|
|
|
|
buffer: Some((
|
|
|
|
pts: Some(1),
|
|
|
|
dts: None,
|
|
|
|
duration: Some(4),
|
|
|
|
offset: 0,
|
|
|
|
offset_end: 4,
|
|
|
|
flags: (
|
|
|
|
bits: 0,
|
|
|
|
),
|
|
|
|
buffer: "AQIDBA==",
|
|
|
|
)),
|
|
|
|
buffer_list: None,
|
2018-07-29 14:04:39 +00:00
|
|
|
caps: Some(Some([
|
2018-07-22 13:48:26 +00:00
|
|
|
(("sample/caps", [
|
2018-07-16 19:46:10 +00:00
|
|
|
("int", "i32", 12),
|
|
|
|
("bool", "bool", true),
|
2018-07-22 13:48:26 +00:00
|
|
|
]), None),
|
2018-07-29 14:04:39 +00:00
|
|
|
])),
|
2018-07-16 19:46:10 +00:00
|
|
|
segment: Some((
|
|
|
|
flags: (
|
|
|
|
bits: 0,
|
|
|
|
),
|
|
|
|
rate: 1,
|
|
|
|
applied_rate: 0.9,
|
|
|
|
format: Time,
|
|
|
|
base: 123,
|
|
|
|
offset: 42,
|
|
|
|
start: 1024,
|
|
|
|
stop: 2048,
|
|
|
|
time: 1042,
|
|
|
|
position: 256,
|
|
|
|
duration: -1,
|
|
|
|
)),
|
|
|
|
info: Some(("sample.info", [
|
|
|
|
("f3", "i32", 123),
|
|
|
|
])),
|
|
|
|
)"#;
|
|
|
|
let sample: Sample = ron::de::from_str(buffer_ron).unwrap();
|
|
|
|
let buffer = sample.get_buffer().unwrap();
|
|
|
|
assert_eq!(buffer.get_pts(), 1.into());
|
|
|
|
assert_eq!(buffer.get_offset_end(), 4);
|
|
|
|
{
|
|
|
|
let data = buffer.map_readable().unwrap();
|
|
|
|
assert_eq!(data.as_slice(), vec![1, 2, 3, 4].as_slice());
|
|
|
|
}
|
|
|
|
assert!(sample.get_buffer_list().is_none());
|
|
|
|
assert!(sample.get_caps().is_some());
|
|
|
|
assert!(sample.get_segment().is_some());
|
|
|
|
assert!(sample.get_info().is_some());
|
|
|
|
|
|
|
|
let buffer_ron = r#"
|
|
|
|
(
|
|
|
|
buffer: None,
|
|
|
|
buffer_list: Some([
|
|
|
|
(
|
|
|
|
pts: Some(1),
|
|
|
|
dts: None,
|
|
|
|
duration: Some(4),
|
|
|
|
offset: 0,
|
|
|
|
offset_end: 4,
|
|
|
|
flags: (
|
|
|
|
bits: 0,
|
|
|
|
),
|
|
|
|
buffer: "AQIDBA==",
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
caps: None,
|
|
|
|
segment: None,
|
|
|
|
info: None,
|
|
|
|
)"#;
|
|
|
|
let sample: Sample = ron::de::from_str(buffer_ron).unwrap();
|
|
|
|
assert!(sample.get_buffer().is_none());
|
|
|
|
assert!(sample.get_buffer_list().is_some());
|
|
|
|
assert!(sample.get_caps().is_none());
|
|
|
|
// Not true in GStreamer 1.x, should be fixed in version 2.0
|
|
|
|
//assert!(sample.get_segment().is_none());
|
|
|
|
assert!(sample.get_info().is_none());
|
|
|
|
}
|
2018-07-29 16:18:24 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_roundrip() {
|
|
|
|
::init().unwrap();
|
|
|
|
|
|
|
|
// Segment present
|
|
|
|
let sample = {
|
|
|
|
let mut buffer = Buffer::from_slice(vec![1, 2, 3, 4]).unwrap();
|
|
|
|
{
|
|
|
|
let buffer = buffer.get_mut().unwrap();
|
|
|
|
buffer.set_pts(1.into());
|
|
|
|
buffer.set_offset(0);
|
|
|
|
buffer.set_offset_end(4);
|
|
|
|
buffer.set_duration(4.into());
|
|
|
|
}
|
|
|
|
|
|
|
|
let caps = Caps::builder("sample/caps")
|
|
|
|
.field("int", &12)
|
|
|
|
.field("bool", &true)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let mut segment = Segment::new();
|
|
|
|
segment.set_flags(SegmentFlags::RESET | SegmentFlags::SEGMENT);
|
|
|
|
segment.set_rate(1f64);
|
|
|
|
segment.set_applied_rate(0.9f64);
|
|
|
|
segment.set_format(Format::Time);
|
|
|
|
segment.set_base(GenericFormattedValue::Time(ClockTime::from_nseconds(123)));
|
|
|
|
segment.set_offset(GenericFormattedValue::Time(ClockTime::from_nseconds(42)));
|
|
|
|
segment.set_start(GenericFormattedValue::Time(ClockTime::from_nseconds(1024)));
|
|
|
|
segment.set_stop(GenericFormattedValue::Time(ClockTime::from_nseconds(2048)));
|
|
|
|
segment.set_time(GenericFormattedValue::Time(ClockTime::from_nseconds(1042)));
|
|
|
|
segment.set_position(GenericFormattedValue::Time(ClockTime::from_nseconds(256)));
|
|
|
|
segment.set_duration(GenericFormattedValue::Time(ClockTime::none()));
|
|
|
|
|
|
|
|
let info = Structure::builder("sample.info")
|
|
|
|
.field("f3", &123i32)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
Sample::new::<GenericFormattedValue>(
|
|
|
|
Some(&buffer),
|
|
|
|
Some(&caps),
|
|
|
|
Some(&segment),
|
|
|
|
Some(info),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
let sample_ser = ron::ser::to_string(&sample).unwrap();
|
|
|
|
let sample_de: Sample = ron::de::from_str(sample_ser.as_str()).unwrap();
|
|
|
|
let buffer_de = sample_de.get_buffer().unwrap();
|
|
|
|
assert_eq!(buffer_de.get_pts(), 1.into());
|
|
|
|
assert_eq!(buffer_de.get_offset_end(), 4);
|
|
|
|
{
|
|
|
|
let data = buffer_de.map_readable().unwrap();
|
|
|
|
assert_eq!(data.as_slice(), vec![1, 2, 3, 4].as_slice());
|
|
|
|
}
|
|
|
|
assert!(sample_de.get_buffer_list().is_none());
|
|
|
|
assert!(sample_de.get_caps().is_some());
|
|
|
|
assert!(sample_de.get_segment().is_some());
|
|
|
|
assert!(sample_de.get_info().is_some());
|
|
|
|
}
|
2018-07-16 19:46:10 +00:00
|
|
|
}
|