mirror of
https://github.com/actix/actix-web.git
synced 2024-11-25 19:11:10 +00:00
338 lines
9.1 KiB
Rust
338 lines
9.1 KiB
Rust
use std::fmt;
|
|
|
|
use derive_more::Error;
|
|
|
|
/// Copy of `http_range::HttpRangeParseError`.
|
|
#[derive(Debug, Clone)]
|
|
enum HttpRangeParseError {
|
|
InvalidRange,
|
|
NoOverlap,
|
|
}
|
|
|
|
impl From<http_range::HttpRangeParseError> for HttpRangeParseError {
|
|
fn from(err: http_range::HttpRangeParseError) -> Self {
|
|
match err {
|
|
http_range::HttpRangeParseError::InvalidRange => Self::InvalidRange,
|
|
http_range::HttpRangeParseError::NoOverlap => Self::NoOverlap,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, Error)]
|
|
#[non_exhaustive]
|
|
pub struct ParseRangeErr(#[error(not(source))] HttpRangeParseError);
|
|
|
|
impl fmt::Display for ParseRangeErr {
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
f.write_str("invalid Range header: ")?;
|
|
f.write_str(match self.0 {
|
|
HttpRangeParseError::InvalidRange => "invalid syntax",
|
|
HttpRangeParseError::NoOverlap => "range starts after end of content",
|
|
})
|
|
}
|
|
}
|
|
|
|
/// HTTP Range header representation.
|
|
#[derive(Debug, Clone, Copy)]
|
|
pub struct HttpRange {
|
|
/// Start of range.
|
|
pub start: u64,
|
|
|
|
/// Length of range.
|
|
pub length: u64,
|
|
}
|
|
|
|
impl HttpRange {
|
|
/// Parses Range HTTP header string as per RFC 2616.
|
|
///
|
|
/// `header` is HTTP Range header (e.g. `bytes=bytes=0-9`).
|
|
/// `size` is full size of response (file).
|
|
pub fn parse(header: &str, size: u64) -> Result<Vec<HttpRange>, ParseRangeErr> {
|
|
let ranges =
|
|
http_range::HttpRange::parse(header, size).map_err(|err| ParseRangeErr(err.into()))?;
|
|
|
|
Ok(ranges
|
|
.iter()
|
|
.map(|range| HttpRange {
|
|
start: range.start,
|
|
length: range.length,
|
|
})
|
|
.collect())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
struct T(&'static str, u64, Vec<HttpRange>);
|
|
|
|
#[test]
|
|
fn test_parse() {
|
|
let tests = vec![
|
|
T("", 0, vec![]),
|
|
T("", 1000, vec![]),
|
|
T("foo", 0, vec![]),
|
|
T("bytes=", 0, vec![]),
|
|
T("bytes=7", 10, vec![]),
|
|
T("bytes= 7 ", 10, vec![]),
|
|
T("bytes=1-", 0, vec![]),
|
|
T("bytes=5-4", 10, vec![]),
|
|
T("bytes=0-2,5-4", 10, vec![]),
|
|
T("bytes=2-5,4-3", 10, vec![]),
|
|
T("bytes=--5,4--3", 10, vec![]),
|
|
T("bytes=A-", 10, vec![]),
|
|
T("bytes=A- ", 10, vec![]),
|
|
T("bytes=A-Z", 10, vec![]),
|
|
T("bytes= -Z", 10, vec![]),
|
|
T("bytes=5-Z", 10, vec![]),
|
|
T("bytes=Ran-dom, garbage", 10, vec![]),
|
|
T("bytes=0x01-0x02", 10, vec![]),
|
|
T("bytes= ", 10, vec![]),
|
|
T("bytes= , , , ", 10, vec![]),
|
|
T(
|
|
"bytes=0-9",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 10,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=0-",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 10,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=5-",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 5,
|
|
length: 5,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=0-20",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 10,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=15-,0-5",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 6,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=1-2,5-",
|
|
10,
|
|
vec![
|
|
HttpRange {
|
|
start: 1,
|
|
length: 2,
|
|
},
|
|
HttpRange {
|
|
start: 5,
|
|
length: 5,
|
|
},
|
|
],
|
|
),
|
|
T(
|
|
"bytes=-2 , 7-",
|
|
11,
|
|
vec![
|
|
HttpRange {
|
|
start: 9,
|
|
length: 2,
|
|
},
|
|
HttpRange {
|
|
start: 7,
|
|
length: 4,
|
|
},
|
|
],
|
|
),
|
|
T(
|
|
"bytes=0-0 ,2-2, 7-",
|
|
11,
|
|
vec![
|
|
HttpRange {
|
|
start: 0,
|
|
length: 1,
|
|
},
|
|
HttpRange {
|
|
start: 2,
|
|
length: 1,
|
|
},
|
|
HttpRange {
|
|
start: 7,
|
|
length: 4,
|
|
},
|
|
],
|
|
),
|
|
T(
|
|
"bytes=-5",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 5,
|
|
length: 5,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=-15",
|
|
10,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 10,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=0-499",
|
|
10000,
|
|
vec![HttpRange {
|
|
start: 0,
|
|
length: 500,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=500-999",
|
|
10000,
|
|
vec![HttpRange {
|
|
start: 500,
|
|
length: 500,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=-500",
|
|
10000,
|
|
vec![HttpRange {
|
|
start: 9500,
|
|
length: 500,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=9500-",
|
|
10000,
|
|
vec![HttpRange {
|
|
start: 9500,
|
|
length: 500,
|
|
}],
|
|
),
|
|
T(
|
|
"bytes=0-0,-1",
|
|
10000,
|
|
vec![
|
|
HttpRange {
|
|
start: 0,
|
|
length: 1,
|
|
},
|
|
HttpRange {
|
|
start: 9999,
|
|
length: 1,
|
|
},
|
|
],
|
|
),
|
|
T(
|
|
"bytes=500-600,601-999",
|
|
10000,
|
|
vec![
|
|
HttpRange {
|
|
start: 500,
|
|
length: 101,
|
|
},
|
|
HttpRange {
|
|
start: 601,
|
|
length: 399,
|
|
},
|
|
],
|
|
),
|
|
T(
|
|
"bytes=500-700,601-999",
|
|
10000,
|
|
vec![
|
|
HttpRange {
|
|
start: 500,
|
|
length: 201,
|
|
},
|
|
HttpRange {
|
|
start: 601,
|
|
length: 399,
|
|
},
|
|
],
|
|
),
|
|
// Match Apache laxity:
|
|
T(
|
|
"bytes= 1 -2 , 4- 5, 7 - 8 , ,,",
|
|
11,
|
|
vec![
|
|
HttpRange {
|
|
start: 1,
|
|
length: 2,
|
|
},
|
|
HttpRange {
|
|
start: 4,
|
|
length: 2,
|
|
},
|
|
HttpRange {
|
|
start: 7,
|
|
length: 2,
|
|
},
|
|
],
|
|
),
|
|
];
|
|
|
|
for t in tests {
|
|
let header = t.0;
|
|
let size = t.1;
|
|
let expected = t.2;
|
|
|
|
let res = HttpRange::parse(header, size);
|
|
|
|
if res.is_err() {
|
|
if expected.is_empty() {
|
|
continue;
|
|
} else {
|
|
panic!(
|
|
"parse({}, {}) returned error {:?}",
|
|
header,
|
|
size,
|
|
res.unwrap_err()
|
|
);
|
|
}
|
|
}
|
|
|
|
let got = res.unwrap();
|
|
|
|
if got.len() != expected.len() {
|
|
panic!(
|
|
"len(parseRange({}, {})) = {}, want {}",
|
|
header,
|
|
size,
|
|
got.len(),
|
|
expected.len()
|
|
);
|
|
}
|
|
|
|
for i in 0..expected.len() {
|
|
if got[i].start != expected[i].start {
|
|
panic!(
|
|
"parseRange({}, {})[{}].start = {}, want {}",
|
|
header, size, i, got[i].start, expected[i].start
|
|
)
|
|
}
|
|
if got[i].length != expected[i].length {
|
|
panic!(
|
|
"parseRange({}, {})[{}].length = {}, want {}",
|
|
header, size, i, got[i].length, expected[i].length
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|