/* * Copyright Stalwart Labs Ltd. See the COPYING * file at the top-level directory of this distribution. * * Licensed under the Apache License, Version 2.0 or the MIT license * , at your * option. This file may not be copied, modified, or distributed * except according to those terms. */ use std::borrow::Cow; use crate::encoders::encode::rfc2047_encode; use super::Header; /// MIME Content-Type or Content-Disposition header #[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord)] pub struct ContentType<'x> { pub c_type: Cow<'x, str>, pub attributes: Vec<(Cow<'x, str>, Cow<'x, str>)>, } impl<'x> ContentType<'x> { /// Create a new Content-Type or Content-Disposition header pub fn new(c_type: impl Into>) -> Self { Self { c_type: c_type.into(), attributes: Vec::new(), } } /// Set a Content-Type / Content-Disposition attribute pub fn attribute( mut self, key: impl Into>, value: impl Into>, ) -> Self { self.attributes.push((key.into(), value.into())); self } /// Returns true when the part is text/* pub fn is_text(&self) -> bool { self.c_type.starts_with("text/") } /// Returns true when the part is an attachment pub fn is_attachment(&self) -> bool { self.c_type == "attachment" } } impl<'x> Header for ContentType<'x> { fn write_header( &self, mut output: impl std::io::Write, mut bytes_written: usize, ) -> std::io::Result { output.write_all(self.c_type.as_bytes())?; bytes_written += self.c_type.len(); if !self.attributes.is_empty() { output.write_all(b"; ")?; bytes_written += 2; for (pos, (key, value)) in self.attributes.iter().enumerate() { if bytes_written + key.len() + value.len() + 3 >= 76 { output.write_all(b"\r\n\t")?; bytes_written = 1; } output.write_all(key.as_bytes())?; output.write_all(b"=")?; bytes_written += rfc2047_encode(value, &mut output)? + key.len() + 1; if pos < self.attributes.len() - 1 { output.write_all(b"; ")?; bytes_written += 2; } } } output.write_all(b"\r\n")?; Ok(0) } }