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
88
89
90
91
92
93
94
95
96
97
98
use std::fmt;
use std::str::FromStr;
use nom::{
character::complete::{char, digit1},
combinator::all_consuming,
sequence::preceded,
IResult,
};
use serde::{Deserialize, Serialize};
use crate::error::{map_add_intent, Error};
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
pub struct PaneId(pub String);
impl FromStr for PaneId {
type Err = Error;
fn from_str(input: &str) -> Result<Self, Self::Err> {
let desc = "PaneId";
let intent = "#{pane_id}";
let (_, pane_id) =
all_consuming(parse::pane_id)(input).map_err(|e| map_add_intent(desc, intent, e))?;
Ok(pane_id)
}
}
impl From<&u16> for PaneId {
fn from(value: &u16) -> Self {
Self(format!("%{value}"))
}
}
impl PaneId {
#[must_use]
pub fn as_str(&self) -> &str {
&self.0
}
}
impl fmt::Display for PaneId {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.0)
}
}
pub(crate) mod parse {
use super::{char, digit1, preceded, IResult, PaneId};
pub(crate) fn pane_id(input: &str) -> IResult<&str, PaneId> {
let (input, digit) = preceded(char('%'), digit1)(input)?;
let id = format!("%{digit}");
Ok((input, PaneId(id)))
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_parse_pane_id_fn() {
let actual = parse::pane_id("%43");
let expected = Ok(("", PaneId("%43".into())));
assert_eq!(actual, expected);
let actual = parse::pane_id("%4");
let expected = Ok(("", PaneId("%4".into())));
assert_eq!(actual, expected);
}
#[test]
fn test_parse_pane_id_struct() {
let actual = PaneId::from_str("%43");
assert!(actual.is_ok());
assert_eq!(actual.unwrap(), PaneId("%43".into()));
let actual = PaneId::from_str("4:38");
assert!(matches!(
actual,
Err(Error::ParseError {
desc: "PaneId",
intent: "#{pane_id}",
err: _
})
));
}
}