1
0
Fork 0
mirror of https://github.com/sile/hls_m3u8.git synced 2024-06-26 16:30:34 +00:00
hls_m3u8/src/tags/media_segment/byte_range.rs

151 lines
3.9 KiB
Rust
Raw Normal View History

2019-09-06 10:55:00 +00:00
use std::fmt;
use std::str::FromStr;
2019-09-10 09:05:20 +00:00
2020-02-02 14:23:47 +00:00
use derive_more::{Deref, DerefMut};
2019-10-04 09:02:21 +00:00
use crate::types::{ByteRange, ProtocolVersion};
2019-09-13 14:06:52 +00:00
use crate::utils::tag;
2019-10-04 09:02:21 +00:00
use crate::{Error, RequiredVersion};
2019-09-10 09:05:20 +00:00
2019-09-22 16:00:38 +00:00
/// # [4.4.2.2. EXT-X-BYTERANGE]
2019-09-06 10:55:00 +00:00
///
2019-10-03 14:23:27 +00:00
/// The [`ExtXByteRange`] tag indicates that a [`Media Segment`] is a sub-range
2019-09-22 16:00:38 +00:00
/// of the resource identified by its `URI`.
///
2019-10-03 14:23:27 +00:00
/// [`Media Segment`]: crate::MediaSegment
2019-09-22 16:00:38 +00:00
/// [4.4.2.2. EXT-X-BYTERANGE]:
/// https://tools.ietf.org/html/draft-pantos-hls-rfc8216bis-04#section-4.4.2.2
2020-02-02 14:23:47 +00:00
#[derive(Deref, DerefMut, Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
2019-09-10 09:05:20 +00:00
pub struct ExtXByteRange(ByteRange);
2019-09-06 10:55:00 +00:00
impl ExtXByteRange {
pub(crate) const PREFIX: &'static str = "#EXT-X-BYTERANGE:";
2019-10-03 14:23:27 +00:00
/// Makes a new [`ExtXByteRange`] tag.
2019-09-22 08:57:28 +00:00
///
2019-09-10 09:05:20 +00:00
/// # Example
2020-02-02 12:38:11 +00:00
///
2019-09-10 09:05:20 +00:00
/// ```
2019-09-22 16:00:38 +00:00
/// # use hls_m3u8::tags::ExtXByteRange;
2019-09-10 09:05:20 +00:00
/// let byte_range = ExtXByteRange::new(20, Some(5));
/// ```
pub const fn new(length: usize, start: Option<usize>) -> Self {
Self(ByteRange::new(length, start))
2019-09-06 10:55:00 +00:00
}
2019-10-03 14:23:27 +00:00
/// Converts the [`ExtXByteRange`] to a [`ByteRange`].
2019-09-22 08:57:28 +00:00
///
2019-09-10 09:05:20 +00:00
/// # Example
2020-02-02 12:38:11 +00:00
///
2019-09-10 09:05:20 +00:00
/// ```
2019-09-22 16:00:38 +00:00
/// # use hls_m3u8::tags::ExtXByteRange;
2019-09-10 09:05:20 +00:00
/// use hls_m3u8::types::ByteRange;
///
/// let byte_range = ExtXByteRange::new(20, Some(5));
/// let range: ByteRange = byte_range.to_range();
/// ```
2019-10-03 15:01:15 +00:00
pub const fn to_range(&self) -> ByteRange { self.0 }
2019-09-22 08:57:28 +00:00
}
2019-09-06 10:55:00 +00:00
2020-02-02 12:38:11 +00:00
/// This tag requires [`ProtocolVersion::V4`].
2019-09-22 08:57:28 +00:00
impl RequiredVersion for ExtXByteRange {
2019-10-03 15:01:15 +00:00
fn required_version(&self) -> ProtocolVersion { ProtocolVersion::V4 }
2019-09-06 10:55:00 +00:00
}
impl fmt::Display for ExtXByteRange {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
2019-09-10 09:05:20 +00:00
write!(f, "{}", Self::PREFIX)?;
write!(f, "{}", self.0)?;
Ok(())
2019-09-06 10:55:00 +00:00
}
}
impl FromStr for ExtXByteRange {
type Err = Error;
2019-09-08 10:23:33 +00:00
2019-09-13 14:06:52 +00:00
fn from_str(input: &str) -> Result<Self, Self::Err> {
let input = tag(input, Self::PREFIX)?;
let tokens = input.splitn(2, '@').collect::<Vec<_>>();
2019-09-22 08:57:28 +00:00
2019-09-10 09:05:20 +00:00
if tokens.is_empty() {
2019-09-13 14:06:52 +00:00
return Err(Error::invalid_input());
2019-09-10 09:05:20 +00:00
}
2019-09-13 14:06:52 +00:00
let length = tokens[0].parse()?;
2019-09-10 09:05:20 +00:00
let start = {
if tokens.len() == 2 {
2019-09-22 18:33:40 +00:00
Some(tokens[1].parse()?)
} else {
None
2019-09-10 09:05:20 +00:00
}
};
2019-10-05 14:08:03 +00:00
Ok(Self::new(length, start))
2019-09-06 10:55:00 +00:00
}
}
#[cfg(test)]
mod test {
use super::*;
use pretty_assertions::assert_eq;
2019-09-06 10:55:00 +00:00
#[test]
2019-09-10 09:05:20 +00:00
fn test_display() {
let byte_range = ExtXByteRange::new(0, Some(5));
assert_eq!(byte_range.to_string(), "#EXT-X-BYTERANGE:0@5".to_string());
let byte_range = ExtXByteRange::new(99999, Some(2));
assert_eq!(
byte_range.to_string(),
"#EXT-X-BYTERANGE:99999@2".to_string()
);
let byte_range = ExtXByteRange::new(99999, None);
assert_eq!(byte_range.to_string(), "#EXT-X-BYTERANGE:99999".to_string());
}
#[test]
2019-09-21 09:53:34 +00:00
fn test_parser() {
2019-09-10 09:05:20 +00:00
let byte_range = ExtXByteRange::new(99999, Some(2));
assert_eq!(
byte_range,
"#EXT-X-BYTERANGE:99999@2".parse::<ExtXByteRange>().unwrap()
);
let byte_range = ExtXByteRange::new(99999, None);
assert_eq!(
byte_range,
"#EXT-X-BYTERANGE:99999".parse::<ExtXByteRange>().unwrap()
);
}
#[test]
fn test_deref() {
let byte_range = ExtXByteRange::new(0, Some(22));
2019-09-21 09:53:34 +00:00
assert_eq!(byte_range.length(), 0);
assert_eq!(byte_range.start(), Some(22));
2019-09-06 10:55:00 +00:00
}
2019-09-22 08:57:28 +00:00
2019-09-22 16:00:38 +00:00
#[test]
fn test_deref_mut() {
let mut byte_range = ExtXByteRange::new(0, Some(22));
byte_range.set_length(100);
byte_range.set_start(Some(50));
assert_eq!(byte_range.length(), 100);
assert_eq!(byte_range.start(), Some(50));
}
2019-09-22 08:57:28 +00:00
#[test]
fn test_required_version() {
assert_eq!(
ExtXByteRange::new(20, Some(5)).required_version(),
ProtocolVersion::V4
);
}
2019-09-06 10:55:00 +00:00
}