582 lines
21 KiB
Rust
582 lines
21 KiB
Rust
use std::borrow::Cow;
|
|
use std::collections::HashSet;
|
|
use std::num::NonZeroU32;
|
|
|
|
use anyhow::{anyhow, bail, Result};
|
|
|
|
use imap_codec::imap_types::body::{
|
|
BasicFields, Body as FetchBody, BodyStructure, MultiPartExtensionData, SinglePartExtensionData,
|
|
SpecificFields,
|
|
};
|
|
use imap_codec::imap_types::core::{AString, IString, NString, Vec1};
|
|
use imap_codec::imap_types::fetch::{Part as FetchPart, Section as FetchSection};
|
|
|
|
use eml_codec::{
|
|
header, mime, mime::r#type::Deductible, part::composite, part::discrete, part::AnyPart,
|
|
};
|
|
|
|
use crate::imap::imf_view::ImfView;
|
|
|
|
pub enum BodySection<'a> {
|
|
Full(Cow<'a, [u8]>),
|
|
Slice {
|
|
body: Cow<'a, [u8]>,
|
|
origin_octet: u32,
|
|
},
|
|
}
|
|
|
|
/// Logic for BODY[<section>]<<partial>>
|
|
/// Works in 3 times:
|
|
/// 1. Find the section (RootMime::subset)
|
|
/// 2. Apply the extraction logic (SelectedMime::extract), like TEXT, HEADERS, etc.
|
|
/// 3. Keep only the given subset provided by partial
|
|
///
|
|
/// Example of message sections:
|
|
///
|
|
/// ```
|
|
/// HEADER ([RFC-2822] header of the message)
|
|
/// TEXT ([RFC-2822] text body of the message) MULTIPART/MIXED
|
|
/// 1 TEXT/PLAIN
|
|
/// 2 APPLICATION/OCTET-STREAM
|
|
/// 3 MESSAGE/RFC822
|
|
/// 3.HEADER ([RFC-2822] header of the message)
|
|
/// 3.TEXT ([RFC-2822] text body of the message) MULTIPART/MIXED
|
|
/// 3.1 TEXT/PLAIN
|
|
/// 3.2 APPLICATION/OCTET-STREAM
|
|
/// 4 MULTIPART/MIXED
|
|
/// 4.1 IMAGE/GIF
|
|
/// 4.1.MIME ([MIME-IMB] header for the IMAGE/GIF)
|
|
/// 4.2 MESSAGE/RFC822
|
|
/// 4.2.HEADER ([RFC-2822] header of the message)
|
|
/// 4.2.TEXT ([RFC-2822] text body of the message) MULTIPART/MIXED
|
|
/// 4.2.1 TEXT/PLAIN
|
|
/// 4.2.2 MULTIPART/ALTERNATIVE
|
|
/// 4.2.2.1 TEXT/PLAIN
|
|
/// 4.2.2.2 TEXT/RICHTEXT
|
|
/// ```
|
|
pub fn body_ext<'a>(
|
|
part: &'a AnyPart<'a>,
|
|
section: &'a Option<FetchSection<'a>>,
|
|
partial: &'a Option<(u32, NonZeroU32)>,
|
|
) -> Result<BodySection<'a>> {
|
|
let root_mime = NodeMime(part);
|
|
let (extractor, path) = SubsettedSection::from(section);
|
|
let selected_mime = root_mime.subset(path)?;
|
|
let extracted_full = selected_mime.extract(&extractor)?;
|
|
Ok(extracted_full.to_body_section(partial))
|
|
}
|
|
|
|
/// Logic for BODY and BODYSTRUCTURE
|
|
///
|
|
/// ```raw
|
|
/// b fetch 29878:29879 (BODY)
|
|
/// * 29878 FETCH (BODY (("text" "plain" ("charset" "utf-8") NIL NIL "quoted-printable" 3264 82)("text" "html" ("charset" "utf-8") NIL NIL "quoted-printable" 31834 643) "alternative"))
|
|
/// * 29879 FETCH (BODY ("text" "html" ("charset" "us-ascii") NIL NIL "7bit" 4107 131))
|
|
/// ^^^^^^^^^^^^^^^^^^^^^^ ^^^ ^^^ ^^^^^^ ^^^^ ^^^
|
|
/// | | | | | | number of lines
|
|
/// | | | | | size
|
|
/// | | | | content transfer encoding
|
|
/// | | | description
|
|
/// | | id
|
|
/// | parameter list
|
|
/// b OK Fetch completed (0.001 + 0.000 secs).
|
|
/// ```
|
|
pub fn bodystructure(part: &AnyPart, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
NodeMime(part).structure(is_ext)
|
|
}
|
|
|
|
/// NodeMime
|
|
///
|
|
/// Used for recursive logic on MIME.
|
|
/// See SelectedMime for inspection.
|
|
struct NodeMime<'a>(&'a AnyPart<'a>);
|
|
impl<'a> NodeMime<'a> {
|
|
/// A MIME object is a tree of elements.
|
|
/// The path indicates which element must be picked.
|
|
/// This function returns the picked element as the new view
|
|
fn subset(self, path: Option<&'a FetchPart>) -> Result<SelectedMime<'a>> {
|
|
match path {
|
|
None => Ok(SelectedMime(self.0)),
|
|
Some(v) => self.rec_subset(v.0.as_ref()),
|
|
}
|
|
}
|
|
|
|
fn rec_subset(self, path: &'a [NonZeroU32]) -> Result<SelectedMime> {
|
|
if path.is_empty() {
|
|
Ok(SelectedMime(self.0))
|
|
} else {
|
|
match self.0 {
|
|
AnyPart::Mult(x) => {
|
|
let next = Self(x.children
|
|
.get(path[0].get() as usize - 1)
|
|
.ok_or(anyhow!("Unable to resolve subpath {:?}, current multipart has only {} elements", path, x.children.len()))?);
|
|
next.rec_subset(&path[1..])
|
|
},
|
|
AnyPart::Msg(x) => {
|
|
let next = Self(x.child.as_ref());
|
|
next.rec_subset(path)
|
|
},
|
|
_ => bail!("You tried to access a subpart on an atomic part (text or binary). Unresolved subpath {:?}", path),
|
|
}
|
|
}
|
|
}
|
|
|
|
fn structure(&self, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
match self.0 {
|
|
AnyPart::Txt(x) => NodeTxt(self, x).structure(is_ext),
|
|
AnyPart::Bin(x) => NodeBin(self, x).structure(is_ext),
|
|
AnyPart::Mult(x) => NodeMult(self, x).structure(is_ext),
|
|
AnyPart::Msg(x) => NodeMsg(self, x).structure(is_ext),
|
|
}
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
|
|
/// A FetchSection must be handled in 2 times:
|
|
/// - First we must extract the MIME part
|
|
/// - Then we must process it as desired
|
|
/// The given struct mixes both work, so
|
|
/// we separate this work here.
|
|
enum SubsettedSection<'a> {
|
|
Part,
|
|
Header,
|
|
HeaderFields(&'a Vec1<AString<'a>>),
|
|
HeaderFieldsNot(&'a Vec1<AString<'a>>),
|
|
Text,
|
|
Mime,
|
|
}
|
|
impl<'a> SubsettedSection<'a> {
|
|
fn from(section: &'a Option<FetchSection>) -> (Self, Option<&'a FetchPart>) {
|
|
match section {
|
|
Some(FetchSection::Text(maybe_part)) => (Self::Text, maybe_part.as_ref()),
|
|
Some(FetchSection::Header(maybe_part)) => (Self::Header, maybe_part.as_ref()),
|
|
Some(FetchSection::HeaderFields(maybe_part, fields)) => {
|
|
(Self::HeaderFields(fields), maybe_part.as_ref())
|
|
}
|
|
Some(FetchSection::HeaderFieldsNot(maybe_part, fields)) => {
|
|
(Self::HeaderFieldsNot(fields), maybe_part.as_ref())
|
|
}
|
|
Some(FetchSection::Mime(part)) => (Self::Mime, Some(part)),
|
|
Some(FetchSection::Part(part)) => (Self::Part, Some(part)),
|
|
None => (Self::Part, None),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Used for current MIME inspection
|
|
///
|
|
/// See NodeMime for recursive logic
|
|
pub struct SelectedMime<'a>(pub &'a AnyPart<'a>);
|
|
impl<'a> SelectedMime<'a> {
|
|
pub fn header_value(&'a self, to_match_ext: &[u8]) -> Option<&'a [u8]> {
|
|
let to_match = to_match_ext.to_ascii_lowercase();
|
|
|
|
self.eml_mime()
|
|
.kv
|
|
.iter()
|
|
.filter_map(|field| match field {
|
|
header::Field::Good(header::Kv2(k, v)) => Some((k, v)),
|
|
_ => None,
|
|
})
|
|
.find(|(k, _)| k.to_ascii_lowercase() == to_match)
|
|
.map(|(_, v)| v)
|
|
.copied()
|
|
}
|
|
|
|
/// The subsetted fetch section basically tells us the
|
|
/// extraction logic to apply on our selected MIME.
|
|
/// This function acts as a router for these logic.
|
|
fn extract(&self, extractor: &SubsettedSection<'a>) -> Result<ExtractedFull<'a>> {
|
|
match extractor {
|
|
SubsettedSection::Text => self.text(),
|
|
SubsettedSection::Header => self.header(),
|
|
SubsettedSection::HeaderFields(fields) => self.header_fields(fields, false),
|
|
SubsettedSection::HeaderFieldsNot(fields) => self.header_fields(fields, true),
|
|
SubsettedSection::Part => self.part(),
|
|
SubsettedSection::Mime => self.mime(),
|
|
}
|
|
}
|
|
|
|
fn mime(&self) -> Result<ExtractedFull<'a>> {
|
|
let bytes = match &self.0 {
|
|
AnyPart::Txt(p) => p.mime.fields.raw,
|
|
AnyPart::Bin(p) => p.mime.fields.raw,
|
|
AnyPart::Msg(p) => p.child.mime().raw,
|
|
AnyPart::Mult(p) => p.mime.fields.raw,
|
|
};
|
|
Ok(ExtractedFull(bytes.into()))
|
|
}
|
|
|
|
fn part(&self) -> Result<ExtractedFull<'a>> {
|
|
let bytes = match &self.0 {
|
|
AnyPart::Txt(p) => p.body,
|
|
AnyPart::Bin(p) => p.body,
|
|
AnyPart::Msg(p) => p.raw_part,
|
|
AnyPart::Mult(_) => bail!("Multipart part has no body"),
|
|
};
|
|
Ok(ExtractedFull(bytes.to_vec().into()))
|
|
}
|
|
|
|
fn eml_mime(&self) -> &eml_codec::mime::NaiveMIME<'_> {
|
|
match &self.0 {
|
|
AnyPart::Msg(msg) => msg.child.mime(),
|
|
other => other.mime(),
|
|
}
|
|
}
|
|
|
|
/// The [...] HEADER.FIELDS, and HEADER.FIELDS.NOT part
|
|
/// specifiers refer to the [RFC-2822] header of the message or of
|
|
/// an encapsulated [MIME-IMT] MESSAGE/RFC822 message.
|
|
/// HEADER.FIELDS and HEADER.FIELDS.NOT are followed by a list of
|
|
/// field-name (as defined in [RFC-2822]) names, and return a
|
|
/// subset of the header. The subset returned by HEADER.FIELDS
|
|
/// contains only those header fields with a field-name that
|
|
/// matches one of the names in the list; similarly, the subset
|
|
/// returned by HEADER.FIELDS.NOT contains only the header fields
|
|
/// with a non-matching field-name. The field-matching is
|
|
/// case-insensitive but otherwise exact.
|
|
fn header_fields(
|
|
&self,
|
|
fields: &'a Vec1<AString<'a>>,
|
|
invert: bool,
|
|
) -> Result<ExtractedFull<'a>> {
|
|
// Build a lowercase ascii hashset with the fields to fetch
|
|
let index = fields
|
|
.as_ref()
|
|
.iter()
|
|
.map(|x| {
|
|
match x {
|
|
AString::Atom(a) => a.inner().as_bytes(),
|
|
AString::String(IString::Literal(l)) => l.as_ref(),
|
|
AString::String(IString::Quoted(q)) => q.inner().as_bytes(),
|
|
}
|
|
.to_ascii_lowercase()
|
|
})
|
|
.collect::<HashSet<_>>();
|
|
|
|
// Extract MIME headers
|
|
let mime = self.eml_mime();
|
|
|
|
// Filter our MIME headers based on the field index
|
|
// 1. Keep only the correctly formatted headers
|
|
// 2. Keep only based on the index presence or absence
|
|
// 3. Reduce as a byte vector
|
|
let buffer = mime
|
|
.kv
|
|
.iter()
|
|
.filter_map(|field| match field {
|
|
header::Field::Good(header::Kv2(k, v)) => Some((k, v)),
|
|
_ => None,
|
|
})
|
|
.filter(|(k, _)| index.contains(&k.to_ascii_lowercase()) ^ invert)
|
|
.fold(vec![], |mut acc, (k, v)| {
|
|
acc.extend(*k);
|
|
acc.extend(b": ");
|
|
acc.extend(*v);
|
|
acc.extend(b"\r\n");
|
|
acc
|
|
});
|
|
|
|
Ok(ExtractedFull(buffer.into()))
|
|
}
|
|
|
|
/// The HEADER [...] part specifiers refer to the [RFC-2822] header of the message or of
|
|
/// an encapsulated [MIME-IMT] MESSAGE/RFC822 message.
|
|
/// ```raw
|
|
/// HEADER ([RFC-2822] header of the message)
|
|
/// ```
|
|
fn header(&self) -> Result<ExtractedFull<'a>> {
|
|
let msg = self
|
|
.0
|
|
.as_message()
|
|
.ok_or(anyhow!("Selected part must be a message/rfc822"))?;
|
|
Ok(ExtractedFull(msg.raw_headers.into()))
|
|
}
|
|
|
|
/// The TEXT part specifier refers to the text body of the message, omitting the [RFC-2822] header.
|
|
fn text(&self) -> Result<ExtractedFull<'a>> {
|
|
let msg = self
|
|
.0
|
|
.as_message()
|
|
.ok_or(anyhow!("Selected part must be a message/rfc822"))?;
|
|
Ok(ExtractedFull(msg.raw_body.into()))
|
|
}
|
|
|
|
// ------------
|
|
|
|
/// Basic field of a MIME part that is
|
|
/// common to all parts
|
|
fn basic_fields(&self) -> Result<BasicFields<'static>> {
|
|
let sz = match self.0 {
|
|
AnyPart::Txt(x) => x.body.len(),
|
|
AnyPart::Bin(x) => x.body.len(),
|
|
AnyPart::Msg(x) => x.raw_part.len(),
|
|
AnyPart::Mult(_) => 0,
|
|
};
|
|
let m = self.0.mime();
|
|
let parameter_list = m
|
|
.ctype
|
|
.as_ref()
|
|
.map(|x| {
|
|
x.params
|
|
.iter()
|
|
.map(|p| {
|
|
(
|
|
IString::try_from(String::from_utf8_lossy(p.name).to_string()),
|
|
IString::try_from(p.value.to_string()),
|
|
)
|
|
})
|
|
.filter(|(k, v)| k.is_ok() && v.is_ok())
|
|
.map(|(k, v)| (k.unwrap(), v.unwrap()))
|
|
.collect()
|
|
})
|
|
.unwrap_or(vec![]);
|
|
|
|
Ok(BasicFields {
|
|
parameter_list,
|
|
id: NString(
|
|
m.id.as_ref()
|
|
.and_then(|ci| IString::try_from(ci.to_string()).ok()),
|
|
),
|
|
description: NString(
|
|
m.description
|
|
.as_ref()
|
|
.and_then(|cd| IString::try_from(cd.to_string()).ok()),
|
|
),
|
|
content_transfer_encoding: match m.transfer_encoding {
|
|
mime::mechanism::Mechanism::_8Bit => unchecked_istring("8bit"),
|
|
mime::mechanism::Mechanism::Binary => unchecked_istring("binary"),
|
|
mime::mechanism::Mechanism::QuotedPrintable => {
|
|
unchecked_istring("quoted-printable")
|
|
}
|
|
mime::mechanism::Mechanism::Base64 => unchecked_istring("base64"),
|
|
_ => unchecked_istring("7bit"),
|
|
},
|
|
// @FIXME we can't compute the size of the message currently...
|
|
size: u32::try_from(sz)?,
|
|
})
|
|
}
|
|
}
|
|
|
|
// ---------------------------
|
|
struct NodeMsg<'a>(&'a NodeMime<'a>, &'a composite::Message<'a>);
|
|
impl<'a> NodeMsg<'a> {
|
|
fn structure(&self, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
let basic = SelectedMime(self.0 .0).basic_fields()?;
|
|
|
|
Ok(BodyStructure::Single {
|
|
body: FetchBody {
|
|
basic,
|
|
specific: SpecificFields::Message {
|
|
envelope: Box::new(ImfView(&self.1.imf).message_envelope()),
|
|
body_structure: Box::new(NodeMime(&self.1.child).structure(is_ext)?),
|
|
number_of_lines: nol(self.1.raw_part),
|
|
},
|
|
},
|
|
extension_data: match is_ext {
|
|
true => Some(SinglePartExtensionData {
|
|
md5: NString(None),
|
|
tail: None,
|
|
}),
|
|
_ => None,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
#[allow(dead_code)]
|
|
struct NodeMult<'a>(&'a NodeMime<'a>, &'a composite::Multipart<'a>);
|
|
impl<'a> NodeMult<'a> {
|
|
fn structure(&self, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
let itype = &self.1.mime.interpreted_type;
|
|
let subtype = IString::try_from(itype.subtype.to_string())
|
|
.unwrap_or(unchecked_istring("alternative"));
|
|
|
|
let inner_bodies = self
|
|
.1
|
|
.children
|
|
.iter()
|
|
.filter_map(|inner| NodeMime(&inner).structure(is_ext).ok())
|
|
.collect::<Vec<_>>();
|
|
|
|
Vec1::validate(&inner_bodies)?;
|
|
let bodies = Vec1::unvalidated(inner_bodies);
|
|
|
|
Ok(BodyStructure::Multi {
|
|
bodies,
|
|
subtype,
|
|
extension_data: match is_ext {
|
|
true => Some(MultiPartExtensionData {
|
|
parameter_list: vec![(
|
|
IString::try_from("boundary").unwrap(),
|
|
IString::try_from(self.1.mime.interpreted_type.boundary.to_string())?,
|
|
)],
|
|
tail: None,
|
|
}),
|
|
_ => None,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
struct NodeTxt<'a>(&'a NodeMime<'a>, &'a discrete::Text<'a>);
|
|
impl<'a> NodeTxt<'a> {
|
|
fn structure(&self, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
let mut basic = SelectedMime(self.0 .0).basic_fields()?;
|
|
|
|
// Get the interpreted content type, set it
|
|
let itype = match &self.1.mime.interpreted_type {
|
|
Deductible::Inferred(v) | Deductible::Explicit(v) => v,
|
|
};
|
|
let subtype =
|
|
IString::try_from(itype.subtype.to_string()).unwrap_or(unchecked_istring("plain"));
|
|
|
|
// Add charset to the list of parameters if we know it has been inferred as it will be
|
|
// missing from the parsed content.
|
|
if let Deductible::Inferred(charset) = &itype.charset {
|
|
basic.parameter_list.push((
|
|
unchecked_istring("charset"),
|
|
IString::try_from(charset.to_string()).unwrap_or(unchecked_istring("us-ascii")),
|
|
));
|
|
}
|
|
|
|
Ok(BodyStructure::Single {
|
|
body: FetchBody {
|
|
basic,
|
|
specific: SpecificFields::Text {
|
|
subtype,
|
|
number_of_lines: nol(self.1.body),
|
|
},
|
|
},
|
|
extension_data: match is_ext {
|
|
true => Some(SinglePartExtensionData {
|
|
md5: NString(None),
|
|
tail: None,
|
|
}),
|
|
_ => None,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
struct NodeBin<'a>(&'a NodeMime<'a>, &'a discrete::Binary<'a>);
|
|
impl<'a> NodeBin<'a> {
|
|
fn structure(&self, is_ext: bool) -> Result<BodyStructure<'static>> {
|
|
let basic = SelectedMime(self.0 .0).basic_fields()?;
|
|
|
|
let default = mime::r#type::NaiveType {
|
|
main: &b"application"[..],
|
|
sub: &b"octet-stream"[..],
|
|
params: vec![],
|
|
};
|
|
let ct = self.1.mime.fields.ctype.as_ref().unwrap_or(&default);
|
|
|
|
let r#type = IString::try_from(String::from_utf8_lossy(ct.main).to_string()).or(Err(
|
|
anyhow!("Unable to build IString from given Content-Type type given"),
|
|
))?;
|
|
|
|
let subtype = IString::try_from(String::from_utf8_lossy(ct.sub).to_string()).or(Err(
|
|
anyhow!("Unable to build IString from given Content-Type subtype given"),
|
|
))?;
|
|
|
|
Ok(BodyStructure::Single {
|
|
body: FetchBody {
|
|
basic,
|
|
specific: SpecificFields::Basic { r#type, subtype },
|
|
},
|
|
extension_data: match is_ext {
|
|
true => Some(SinglePartExtensionData {
|
|
md5: NString(None),
|
|
tail: None,
|
|
}),
|
|
_ => None,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
// ---------------------------
|
|
|
|
struct ExtractedFull<'a>(Cow<'a, [u8]>);
|
|
impl<'a> ExtractedFull<'a> {
|
|
/// It is possible to fetch a substring of the designated text.
|
|
/// This is done by appending an open angle bracket ("<"), the
|
|
/// octet position of the first desired octet, a period, the
|
|
/// maximum number of octets desired, and a close angle bracket
|
|
/// (">") to the part specifier. If the starting octet is beyond
|
|
/// the end of the text, an empty string is returned.
|
|
///
|
|
/// Any partial fetch that attempts to read beyond the end of the
|
|
/// text is truncated as appropriate. A partial fetch that starts
|
|
/// at octet 0 is returned as a partial fetch, even if this
|
|
/// truncation happened.
|
|
///
|
|
/// Note: This means that BODY[]<0.2048> of a 1500-octet message
|
|
/// will return BODY[]<0> with a literal of size 1500, not
|
|
/// BODY[].
|
|
///
|
|
/// Note: A substring fetch of a HEADER.FIELDS or
|
|
/// HEADER.FIELDS.NOT part specifier is calculated after
|
|
/// subsetting the header.
|
|
fn to_body_section(self, partial: &'_ Option<(u32, NonZeroU32)>) -> BodySection<'a> {
|
|
match partial {
|
|
Some((begin, len)) => self.partialize(*begin, *len),
|
|
None => BodySection::Full(self.0),
|
|
}
|
|
}
|
|
|
|
fn partialize(self, begin: u32, len: NonZeroU32) -> BodySection<'a> {
|
|
// Asked range is starting after the end of the content,
|
|
// returning an empty buffer
|
|
if begin as usize > self.0.len() {
|
|
return BodySection::Slice {
|
|
body: Cow::Borrowed(&[][..]),
|
|
origin_octet: begin,
|
|
};
|
|
}
|
|
|
|
// Asked range is ending after the end of the content,
|
|
// slice only the beginning of the buffer
|
|
if (begin + len.get()) as usize >= self.0.len() {
|
|
return BodySection::Slice {
|
|
body: match self.0 {
|
|
Cow::Borrowed(body) => Cow::Borrowed(&body[begin as usize..]),
|
|
Cow::Owned(body) => Cow::Owned(body[begin as usize..].to_vec()),
|
|
},
|
|
origin_octet: begin,
|
|
};
|
|
}
|
|
|
|
// Range is included inside the considered content,
|
|
// this is the "happy case"
|
|
BodySection::Slice {
|
|
body: match self.0 {
|
|
Cow::Borrowed(body) => {
|
|
Cow::Borrowed(&body[begin as usize..(begin + len.get()) as usize])
|
|
}
|
|
Cow::Owned(body) => {
|
|
Cow::Owned(body[begin as usize..(begin + len.get()) as usize].to_vec())
|
|
}
|
|
},
|
|
origin_octet: begin,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// ---- LEGACY
|
|
|
|
/// s is set to static to ensure that only compile time values
|
|
/// checked by developpers are passed.
|
|
fn unchecked_istring(s: &'static str) -> IString {
|
|
IString::try_from(s).expect("this value is expected to be a valid imap-codec::IString")
|
|
}
|
|
|
|
// Number Of Lines
|
|
fn nol(input: &[u8]) -> u32 {
|
|
input
|
|
.iter()
|
|
.filter(|x| **x == b'\n')
|
|
.count()
|
|
.try_into()
|
|
.unwrap_or(0)
|
|
}
|