1
0
Fork 0
mirror of https://github.com/sile/hls_m3u8.git synced 2025-01-10 20:25:25 +00:00

improve ExtXDateRange

This commit is contained in:
Luro02 2020-03-25 11:21:20 +01:00
parent b8fd4c15d5
commit d1fdb7fec1
No known key found for this signature in database
GPG key ID: B66FD4F74501A9CF

View file

@ -19,105 +19,159 @@ use crate::{Error, RequiredVersion};
#[builder(setter(into))] #[builder(setter(into))]
#[shorthand(enable(must_use, into))] #[shorthand(enable(must_use, into))]
pub struct ExtXDateRange { pub struct ExtXDateRange {
/// A string that uniquely identifies an [`ExtXDateRange`] in the Playlist. /// A string that uniquely identifies an [`ExtXDateRange`] in the playlist.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is required. /// This field is required.
id: String, id: String,
/// A client-defined string that specifies some set of attributes and their /// A client-defined string that specifies some set of attributes and their
/// associated value semantics. All [`ExtXDateRange`]s with the same class /// associated value semantics. All [`ExtXDateRange`]s with the same class
/// attribute value must adhere to these semantics. /// attribute value must adhere to these semantics.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is optional. /// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
class: Option<String>, class: Option<String>,
/// The date at which the [`ExtXDateRange`] begins. /// The date at which the [`ExtXDateRange`] begins.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is required. /// This field is required.
#[cfg(feature = "chrono")] #[cfg(feature = "chrono")]
#[shorthand(enable(copy), disable(into))]
start_date: DateTime<FixedOffset>, start_date: DateTime<FixedOffset>,
/// The date at which the [`ExtXDateRange`] begins.
///
/// ## Note
///
/// This field is required.
#[cfg(not(feature = "chrono"))]
start_date: String,
/// The date at which the [`ExtXDateRange`] ends. It must be equal to or /// The date at which the [`ExtXDateRange`] ends. It must be equal to or
/// later than the value of the [`start-date`] attribute. /// later than the value of the [`start-date`] attribute.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is optional. /// This field is optional.
/// ///
/// [`start-date`]: #method.start_date /// [`start-date`]: #method.start_date
#[cfg(feature = "chrono")] #[cfg(feature = "chrono")]
#[shorthand(enable(copy), disable(into))]
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
end_date: Option<DateTime<FixedOffset>>, end_date: Option<DateTime<FixedOffset>>,
/// The date at which the [`ExtXDateRange`] ends. It must be equal to or
/// later than the value of the start-date field.
///
/// ## Note
///
/// This field is optional.
///
/// [`start-date`]: #method.start_date
#[cfg(not(feature = "chrono"))] #[cfg(not(feature = "chrono"))]
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
end_date: Option<String>, end_date: Option<String>,
/// The duration of the [`ExtXDateRange`]. A single instant in time (e.g., /// The duration of the [`ExtXDateRange`]. A single instant in time (e.g.,
/// crossing a finish line) should be represented with a duration of 0. /// crossing a finish line) should be represented with a duration of 0.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is optional. /// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
duration: Option<Duration>, #[shorthand(enable(skip))]
/// The expected duration of the [`ExtXDateRange`]. pub duration: Option<Duration>,
/// This attribute should be used to indicate the expected duration of a /// This field indicates the expected duration of an [`ExtXDateRange`],
/// [`ExtXDateRange`] whose actual duration is not yet known. /// whose actual duration is not yet known.
/// ///
/// # Note /// ## Note
/// ///
/// This attribute is optional. /// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
planned_duration: Option<Duration>, #[shorthand(enable(skip))]
/// You can read about this attribute here pub planned_duration: Option<Duration>,
/// <https://tools.ietf.org/html/rfc8216#section-4.3.2.7.1> /// SCTE-35 (ANSI/SCTE 35 2013) is a joint ANSI/Society of Cable and
/// Telecommunications Engineers standard that describes the inline
/// insertion of cue tones in mpeg-ts streams.
/// ///
/// # Note /// SCTE-35 was originally used in the US to signal a local ad insertion
/// opportunity in the transport streams, and in Europe to insert local TV
/// programs (e.g. local news transmissions). It is now used to signal all
/// kinds of program and ad events in linear transport streams and in newer
/// ABR delivery formats such as HLS and DASH.
/// ///
/// This attribute is optional. /// <https://en.wikipedia.org/wiki/SCTE-35>
///
/// ## Note
///
/// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
scte35_cmd: Option<String>, scte35_cmd: Option<String>,
/// You can read about this attribute here /// SCTE-35 (ANSI/SCTE 35 2013) is a joint ANSI/Society of Cable and
/// <https://tools.ietf.org/html/rfc8216#section-4.3.2.7.1> /// Telecommunications Engineers standard that describes the inline
/// insertion of cue tones in mpeg-ts streams.
/// ///
/// # Note /// SCTE-35 was originally used in the US to signal a local ad insertion
/// opportunity in the transport streams, and in Europe to insert local TV
/// programs (e.g. local news transmissions). It is now used to signal all
/// kinds of program and ad events in linear transport streams and in newer
/// ABR delivery formats such as HLS and DASH.
/// ///
/// This attribute is optional. /// <https://en.wikipedia.org/wiki/SCTE-35>
///
/// ## Note
///
/// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
scte35_out: Option<String>, scte35_out: Option<String>,
/// You can read about this attribute here /// SCTE-35 (ANSI/SCTE 35 2013) is a joint ANSI/Society of Cable and
/// <https://tools.ietf.org/html/rfc8216#section-4.3.2.7.1> /// Telecommunications Engineers standard that describes the inline
/// insertion of cue tones in mpeg-ts streams.
/// ///
/// # Note /// SCTE-35 was originally used in the US to signal a local ad insertion
/// opportunity in the transport streams, and in Europe to insert local TV
/// programs (e.g. local news transmissions). It is now used to signal all
/// kinds of program and ad events in linear transport streams and in newer
/// ABR delivery formats such as HLS and DASH.
/// ///
/// This attribute is optional. /// <https://en.wikipedia.org/wiki/SCTE-35>
///
/// ## Note
///
/// This field is optional.
#[builder(setter(strip_option), default)] #[builder(setter(strip_option), default)]
scte35_in: Option<String>, scte35_in: Option<String>,
/// This attribute indicates that the end of the range containing it is /// This field indicates that the [`ExtXDateRange::end_date`] is equal to
/// equal to the [`start-date`] of its following range. The following range /// the [`ExtXDateRange::start_date`] of the following range.
/// is the [`ExtXDateRange`] of the same class, that has the earliest
/// [`start-date`] after the [`start-date`] of the range in question.
/// ///
/// # Note /// The following range is the [`ExtXDateRange`] with the same class, that
/// has the earliest start date after the start date of the range in
/// question.
/// ///
/// This attribute is optional. /// ## Note
///
/// This field is optional.
#[builder(default)] #[builder(default)]
end_on_next: bool, #[shorthand(enable(skip))]
pub end_on_next: bool,
/// The `"X-"` prefix defines a namespace reserved for client-defined /// The `"X-"` prefix defines a namespace reserved for client-defined
/// attributes. The client-attribute must be a uppercase characters. /// attributes.
/// Clients should use a reverse-DNS syntax when defining their own
/// attribute names to avoid collisions. An example of a client-defined
/// attribute is `X-COM-EXAMPLE-AD-ID="XYZ123"`.
/// ///
/// # Note /// A client-attribute can only consist of uppercase characters (A-Z),
/// numbers (0-9) and `-`.
/// ///
/// This attribute is optional. /// Clients should use a reverse-dns naming scheme, when defining
/// their own attribute names to avoid collisions.
///
/// An example of a client-defined attribute is
/// `X-COM-EXAMPLE-AD-ID="XYZ123"`.
///
/// ## Note
///
/// This field is optional.
#[builder(default)] #[builder(default)]
#[shorthand(enable(collection_magic, get_mut))] #[shorthand(enable(collection_magic), disable(set, get))]
client_attributes: BTreeMap<String, Value>, pub client_attributes: BTreeMap<String, Value>,
} }
impl ExtXDateRangeBuilder { impl ExtXDateRangeBuilder {
@ -170,7 +224,6 @@ let date_range = ExtXDateRange::new(
doc = r#" doc = r#"
``` ```
# use hls_m3u8::tags::ExtXDateRange; # use hls_m3u8::tags::ExtXDateRange;
let date_range = ExtXDateRange::new("id", "2010-02-19T14:54:23.031+08:00"); let date_range = ExtXDateRange::new("id", "2010-02-19T14:54:23.031+08:00");
``` ```
"# "#
@ -200,7 +253,61 @@ let date_range = ExtXDateRange::new("id", "2010-02-19T14:54:23.031+08:00");
} }
/// Returns a builder for [`ExtXDateRange`]. /// Returns a builder for [`ExtXDateRange`].
///
/// # Example
#[cfg_attr(
feature = "chrono",
doc = r#"
```
# use hls_m3u8::tags::ExtXDateRange;
use std::time::Duration;
use chrono::{FixedOffset, TimeZone};
use hls_m3u8::types::Float;
let date_range = ExtXDateRange::builder()
.id("test_id")
.class("test_class")
.start_date(FixedOffset::east(0).ymd(2014, 3, 5).and_hms(11, 15, 0))
.end_date(FixedOffset::east(0).ymd(2014, 3, 5).and_hms(11, 16, 0))
.duration(Duration::from_secs_f64(60.1))
.planned_duration(Duration::from_secs_f64(59.993))
.insert_client_attribute("X-CUSTOM", Float::new(45.3))
.scte35_cmd("0xFC002F0000000000FF2")
.scte35_out("0xFC002F0000000000FF0")
.scte35_in("0xFC002F0000000000FF1")
.end_on_next(true)
.build()?;
# Ok::<(), String>(())
```
"#
)]
#[cfg_attr(
not(feature = "chrono"),
doc = r#"
```
# use hls_m3u8::tags::ExtXDateRange;
use std::time::Duration;
use hls_m3u8::types::Float;
let date_range = ExtXDateRange::builder()
.id("test_id")
.class("test_class")
.start_date("2014-03-05T11:15:00Z")
.end_date("2014-03-05T11:16:00Z")
.duration(Duration::from_secs_f64(60.1))
.planned_duration(Duration::from_secs_f64(59.993))
.insert_client_attribute("X-CUSTOM", Float::new(45.3))
.scte35_cmd("0xFC002F0000000000FF2")
.scte35_out("0xFC002F0000000000FF0")
.scte35_in("0xFC002F0000000000FF1")
.end_on_next(true)
.build()?;
# Ok::<(), String>(())
```
"#
)]
#[must_use] #[must_use]
#[inline]
pub fn builder() -> ExtXDateRangeBuilder { ExtXDateRangeBuilder::default() } pub fn builder() -> ExtXDateRangeBuilder { ExtXDateRangeBuilder::default() }
} }
@ -267,13 +374,23 @@ impl FromStr for ExtXDateRange {
"SCTE35-IN" => scte35_in = Some(unquote(value)), "SCTE35-IN" => scte35_in = Some(unquote(value)),
"END-ON-NEXT" => { "END-ON-NEXT" => {
if value != "YES" { if value != "YES" {
return Err(Error::custom("The value of `END-ON-NEXT` has to be `YES`!")); return Err(Error::custom("`END-ON-NEXT` must be `YES`"));
} }
end_on_next = true; end_on_next = true;
} }
_ => { _ => {
if key.starts_with("X-") { if key.starts_with("X-") {
client_attributes.insert(key.to_ascii_uppercase(), value.parse()?); if key.chars().any(|c| {
c.is_ascii_lowercase()
|| !c.is_ascii()
|| !(c.is_alphanumeric() || c == '-')
}) {
return Err(Error::custom(
"a client attribute can only consist of uppercase ascii characters, numbers or `-`",
));
}
client_attributes.insert(key.to_string(), value.parse()?);
} else { } else {
// [6.3.1. General Client Responsibilities] // [6.3.1. General Client Responsibilities]
// > ignore any attribute/value pair with an // > ignore any attribute/value pair with an
@ -287,8 +404,28 @@ impl FromStr for ExtXDateRange {
let start_date = start_date.ok_or_else(|| Error::missing_value("START-DATE"))?; let start_date = start_date.ok_or_else(|| Error::missing_value("START-DATE"))?;
if end_on_next && class.is_none() { if end_on_next && class.is_none() {
return Err(Error::invalid_input()); return Err(Error::missing_attribute("CLASS"));
} else if end_on_next && duration.is_some() {
return Err(Error::unexpected_attribute("DURATION"));
} else if end_on_next && end_date.is_some() {
return Err(Error::unexpected_attribute("END-DATE"));
} }
// TODO: verify this without chrono?
// https://tools.ietf.org/html/rfc8216#section-4.3.2.7
#[cfg(feature = "chrono")]
{
if let (Some(Ok(duration)), Some(end_date)) =
(duration.map(chrono::Duration::from_std), &end_date)
{
if start_date + duration != *end_date {
return Err(Error::custom(
"end_date must be equal to start_date + duration",
));
}
}
}
Ok(Self { Ok(Self {
id, id,
class, class,
@ -470,11 +607,11 @@ mod test {
.end_date({ .end_date({
#[cfg(feature = "chrono")] #[cfg(feature = "chrono")]
{ {
FixedOffset::east(0).ymd(2014, 3, 5).and_hms(11, 16, 0) FixedOffset::east(0).ymd(2014, 3, 5).and_hms_milli(11, 16, 0, 100)
} }
#[cfg(not(feature = "chrono"))] #[cfg(not(feature = "chrono"))]
{ {
"2014-03-05T11:16:00Z" "2014-03-05T11:16:00.100Z"
} }
}) })
.duration(Duration::from_secs_f64(60.1)) .duration(Duration::from_secs_f64(60.1))
@ -483,7 +620,6 @@ mod test {
.scte35_cmd("0xFC002F0000000000FF2") .scte35_cmd("0xFC002F0000000000FF2")
.scte35_out("0xFC002F0000000000FF0") .scte35_out("0xFC002F0000000000FF0")
.scte35_in("0xFC002F0000000000FF1") .scte35_in("0xFC002F0000000000FF1")
.end_on_next(true)
.build() .build()
.unwrap(), .unwrap(),
concat!( concat!(
@ -491,14 +627,13 @@ mod test {
"ID=\"test_id\",", "ID=\"test_id\",",
"CLASS=\"test_class\",", "CLASS=\"test_class\",",
"START-DATE=\"2014-03-05T11:15:00Z\",", "START-DATE=\"2014-03-05T11:15:00Z\",",
"END-DATE=\"2014-03-05T11:16:00Z\",", "END-DATE=\"2014-03-05T11:16:00.100Z\",",
"DURATION=60.1,", "DURATION=60.1,",
"PLANNED-DURATION=59.993,", "PLANNED-DURATION=59.993,",
"SCTE35-CMD=0xFC002F0000000000FF2,", "SCTE35-CMD=0xFC002F0000000000FF2,",
"SCTE35-OUT=0xFC002F0000000000FF0,", "SCTE35-OUT=0xFC002F0000000000FF0,",
"SCTE35-IN=0xFC002F0000000000FF1,", "SCTE35-IN=0xFC002F0000000000FF1,",
"X-CUSTOM=45.3,", "X-CUSTOM=45.3",
"END-ON-NEXT=YES"
) )
}, },
} }