1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
use pest::Parser;
use pest_derive::Parser;
#[derive(Parser)]
#[grammar = "./byte_range.pest"]
struct ByteRangeParser;
#[derive(Debug, Eq, PartialEq, Clone, Hash)]
pub enum ByteRange {
FromTo(u64),
FromToAll(u64, u64),
Last(u64),
}
impl ByteRange {
pub fn parse(header: &str) -> Vec<Self> {
let byte_range_spec_iter = match ByteRangeParser::parse(Rule::byte_ranges_specifier, header)
{
Err(_) => {
return vec![];
}
Ok(x) => x.peek().unwrap().into_inner(),
};
let mut result = Vec::new();
for spec in byte_range_spec_iter {
match spec.as_rule() {
Rule::from_to => {
let offset: u64 = spec.into_inner().peek().unwrap().as_str().parse().unwrap();
result.push(ByteRange::FromTo(offset));
}
Rule::from_to_all => {
let mut inner_pairs = spec.into_inner();
let begin: u64 = inner_pairs.next().unwrap().as_str().parse().unwrap();
let end: u64 = inner_pairs.next().unwrap().as_str().parse().unwrap();
if begin > end {
continue;
}
result.push(ByteRange::FromToAll(begin, end));
}
Rule::last => {
let length: u64 = spec.into_inner().peek().unwrap().as_str().parse().unwrap();
result.push(ByteRange::Last(length));
}
Rule::EOI => {}
_ => unreachable!(),
}
}
result
}
}