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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
use crate::{Channel, ERR_PREFIX};
use bstr::BStr;
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub enum PacketLine<'a> {
Data(&'a [u8]),
Flush,
Delimiter,
ResponseEnd,
}
impl<'a> PacketLine<'a> {
pub fn as_slice(&self) -> Option<&[u8]> {
match self {
PacketLine::Data(d) => Some(d),
PacketLine::Flush | PacketLine::Delimiter | PacketLine::ResponseEnd => None,
}
}
pub fn as_bstr(&self) -> Option<&BStr> {
self.as_slice().map(Into::into)
}
pub fn as_error(&self) -> Option<Error<'_>> {
self.as_slice().map(Error)
}
pub fn check_error(&self) -> Option<Error<'_>> {
self.as_slice().and_then(|data| {
if data.len() >= ERR_PREFIX.len() && &data[..ERR_PREFIX.len()] == ERR_PREFIX {
Some(Error(&data[ERR_PREFIX.len()..]))
} else {
None
}
})
}
pub fn as_text(&self) -> Option<Text<'_>> {
self.as_slice().map(Into::into)
}
pub fn as_band(&self, kind: Channel) -> Option<Band<'_>> {
self.as_slice().map(|d| match kind {
Channel::Data => Band::Data(d),
Channel::Progress => Band::Progress(d),
Channel::Error => Band::Error(d),
})
}
pub fn decode_band(&self) -> Result<Band<'_>, DecodeBandError> {
let d = self.as_slice().ok_or(DecodeBandError::NonDataLine)?;
Ok(match d[0] {
1 => Band::Data(&d[1..]),
2 => Band::Progress(&d[1..]),
3 => Band::Error(&d[1..]),
band => return Err(DecodeBandError::InvalidSideBand(band)),
})
}
}
use quick_error::quick_error;
quick_error! {
#[derive(Debug)]
#[allow(missing_docs)]
pub enum DecodeBandError {
InvalidSideBand(band: u8) {
display("attempt to decode a non-side channel line or input was malformed: {}", band)
}
NonDataLine {
display("attempt to decode a non-data line into a side-channel band")
}
}
}
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub struct Error<'a>(pub &'a [u8]);
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub struct Text<'a>(pub &'a [u8]);
impl<'a> From<&'a [u8]> for Text<'a> {
fn from(d: &'a [u8]) -> Self {
let d = if d[d.len() - 1] == b'\n' { &d[..d.len() - 1] } else { d };
Text(d)
}
}
impl<'a> Text<'a> {
pub fn as_slice(&self) -> &[u8] {
self.0
}
pub fn as_bstr(&self) -> &BStr {
self.0.into()
}
}
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub enum Band<'a> {
Data(&'a [u8]),
Progress(&'a [u8]),
Error(&'a [u8]),
}
#[cfg(all(not(feature = "blocking-io"), feature = "async-io"))]
mod async_io;
#[cfg(feature = "blocking-io")]
mod blocking_io;