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
85
86
87
use crate::header::headers::OneOrMore;
use crate::header::name::Name;
use crate::header::{ConstNamed, ExtendValues, HeaderParse};
use crate::method::Method;
use crate::parse::{whitespace, ParseCtx};
use crate::print::PrintCtx;
use anyhow::Result;
use nom::bytes::complete::take_while;
use nom::character::complete::digit1;
use nom::combinator::map_res;
use nom::sequence::separated_pair;
use nom::Finish;
use std::num::ParseIntError;
use std::str::FromStr;
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct CSeq {
pub cseq: u32,
pub method: Method,
}
impl CSeq {
#[inline]
pub const fn new(cseq: u32, method: Method) -> CSeq {
CSeq { cseq, method }
}
}
impl ConstNamed for CSeq {
const NAME: Name = Name::CSEQ;
}
impl HeaderParse for CSeq {
fn parse<'i>(ctx: ParseCtx, i: &'i str) -> Result<(&'i str, Self)> {
let (rem, cseq) = map_res(
separated_pair(
map_res(digit1, FromStr::from_str),
take_while(whitespace),
Method::parse(ctx),
),
|(cseq, method)| -> Result<_, ParseIntError> { Ok(CSeq { cseq, method }) },
)(i)
.finish()?;
Ok((rem, cseq))
}
}
impl ExtendValues for CSeq {
fn extend_values(&self, ctx: PrintCtx<'_>, values: &mut OneOrMore) {
*values = self.create_values(ctx)
}
fn create_values(&self, _: PrintCtx<'_>) -> OneOrMore {
OneOrMore::One(format!("{} {}", self.cseq, self.method).into())
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::Headers;
const CSEQ: CSeq = CSeq {
cseq: 123,
method: Method::INVITE,
};
#[test]
fn print_cseq() {
let mut headers = Headers::new();
headers.insert_named(&CSEQ);
let headers = headers.to_string();
assert_eq!(headers, "CSeq: 123 INVITE\r\n");
}
#[test]
fn parse_cseq() {
let mut headers = Headers::new();
headers.insert(Name::CSEQ, "123 INVITE");
let cseq: CSeq = headers.get_named().unwrap();
assert_eq!(cseq, CSEQ);
}
}