Files
himalaya/src/imap/message/get.rs
T
Clément DOUIN 662bd26eb1 refactor: remove composer and reader
Composers and readers did not work as expected. It is just not possible for
himalaya to spawn a command that spawns $EDITOR, piping and redirection cannot
satisfy all the needs. Either the $EDITOR does not spawn (hangs over), either
himalaya does not collect any output from edition. The simplest way is to use an
intermediate temp file, or use process substitution. For eg., using mml:

  mml compose >(himalaya message send)

You can also write into a file then feed himalaya with it.
2026-06-01 16:27:54 +02:00

349 lines
11 KiB
Rust

// This file is part of Himalaya, a CLI to manage emails.
//
// Copyright (C) 2022-2026 soywod <pimalaya.org@posteo.net>
//
// This program is free software: you can redistribute it and/or modify it under
// the terms of the GNU Affero General Public License as published by the Free
// Software Foundation, either version 3 of the License, or (at your option) any
// later version.
//
// This program is distributed in the hope that it will be useful, but WITHOUT
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
// FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
// details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
use std::fmt;
use anyhow::{Result, bail};
use clap::Parser;
use comfy_table::{Cell, ContentArrangement, Row, Table, presets};
use io_imap::types::fetch::{MacroOrMessageDataItemNames, MessageDataItem, MessageDataItemName};
use mail_parser::{Addr, Address, ContentType, Message, MessageParser, MimeHeaders};
use pimalaya_cli::printer::Printer;
use serde::Serialize;
use crate::imap::{
client::ImapClient,
mailbox::arg::{MailboxNameOptionalFlag, MailboxNoSelectFlag},
};
/// Get a message and display its structure.
///
/// This command fetches a message and displays its headers along with
/// the body structure tree showing all MIME parts.
#[derive(Debug, Parser)]
pub struct ImapMessageGetCommand {
#[command(flatten)]
pub mailbox_name: MailboxNameOptionalFlag,
#[command(flatten)]
pub mailbox_no_select: MailboxNoSelectFlag,
/// The message UID (or sequence number with --seq).
pub id: u32,
/// Use sequence numbers instead of UIDs.
#[arg(long)]
pub seq: bool,
}
impl ImapMessageGetCommand {
pub fn execute(self, printer: &mut impl Printer, client: &mut ImapClient) -> Result<()> {
let mailbox = self.mailbox_name.inner.try_into()?;
if self.id == 0 {
bail!("ID must be non-zero");
}
if !self.mailbox_no_select.inner {
client.select(mailbox)?;
}
let item_names =
MacroOrMessageDataItemNames::MessageDataItemNames(vec![MessageDataItemName::BodyExt {
section: None,
partial: None,
peek: true,
}]);
let sequence_set = self.id.to_string().parse()?;
let mut data = client.fetch(sequence_set, item_names, !self.seq)?;
let Some((_, items)) = data.pop_first() else {
bail!("Get message `{}` error: no message data returned", self.id);
};
let mut raw_message: Option<Vec<u8>> = None;
for item in items.into_iter() {
if let MessageDataItem::BodyExt { data, .. } = item {
if let Some(data) = data.0 {
raw_message = Some(data.as_ref().to_vec());
}
}
}
let Some(raw) = raw_message else {
bail!("Get message `{}` error: no message data returned", self.id);
};
let Some(message) = MessageParser::new().parse(&raw) else {
bail!(
"Get message `{}` error: failed to parse MIME message",
self.id
);
};
let structure = MessageStructure::from_parsed(&message);
printer.out(structure)
}
}
#[derive(Clone, Debug, Serialize)]
pub struct MessageHeaders {
pub date: Option<String>,
pub subject: Option<String>,
pub message_id: Option<String>,
pub in_reply_to: Option<String>,
pub from: Vec<String>,
pub to: Vec<String>,
pub cc: Vec<String>,
pub bcc: Vec<String>,
pub reply_to: Vec<String>,
}
#[derive(Clone, Debug, Serialize)]
pub struct BodyPart {
pub content_type: String,
pub name: Option<String>,
pub size: Option<usize>,
#[serde(skip_serializing_if = "Vec::is_empty")]
pub parts: Vec<BodyPart>,
}
#[derive(Clone, Debug, Serialize)]
pub struct MessageStructure {
pub headers: MessageHeaders,
pub body: Option<BodyPart>,
}
impl MessageStructure {
pub fn from_parsed(message: &Message<'_>) -> Self {
// Extract headers
let headers = MessageHeaders {
date: message.date().map(|d| d.to_rfc3339()),
subject: message.subject().map(|s| s.to_string()),
message_id: message.message_id().map(|s| s.to_string()),
in_reply_to: message.in_reply_to().as_text().map(|s| s.to_string()),
from: extract_addresses(message.from()),
to: extract_addresses(message.to()),
cc: extract_addresses(message.cc()),
bcc: extract_addresses(message.bcc()),
reply_to: extract_addresses(message.reply_to()),
};
// Build body structure tree
let body = build_body_tree(message);
Self { headers, body }
}
}
fn extract_addresses(addr: Option<&Address<'_>>) -> Vec<String> {
match addr {
Some(Address::List(list)) => list.iter().map(format_addr).collect(),
Some(Address::Group(groups)) => groups
.iter()
.flat_map(|g| g.addresses.iter().map(format_addr))
.collect(),
None => Vec::new(),
}
}
fn format_addr(addr: &Addr<'_>) -> String {
match (&addr.name, &addr.address) {
(Some(name), Some(email)) => format!("{} <{}>", name, email),
(None, Some(email)) => email.to_string(),
(Some(name), None) => name.to_string(),
(None, None) => String::new(),
}
}
fn format_content_type(ct: &ContentType<'_>) -> String {
match &ct.c_subtype {
Some(sub) => format!("{}/{}", ct.c_type, sub),
None => ct.c_type.to_string(),
}
}
fn build_body_tree(message: &mail_parser::Message<'_>) -> Option<BodyPart> {
let content_type = message
.root_part()
.content_type()
.map(format_content_type)
.unwrap_or_else(|| "text/plain".to_string());
let is_multipart = content_type.starts_with("multipart/");
if is_multipart {
// Multipart message - build tree from parts
let parts: Vec<BodyPart> = message
.parts
.iter()
.skip(1) // Skip the root part (it's the multipart container)
.filter_map(|part| build_part_tree(part))
.collect();
Some(BodyPart {
content_type,
name: None,
size: None,
parts,
})
} else {
// Single part message
let size = message.raw_message.len();
Some(BodyPart {
content_type,
name: None,
size: Some(size),
parts: Vec::new(),
})
}
}
fn build_part_tree(part: &mail_parser::MessagePart<'_>) -> Option<BodyPart> {
let content_type = part
.content_type()
.map(format_content_type)
.unwrap_or_else(|| "application/octet-stream".to_string());
// Skip multipart container parts (they're represented by their children)
if content_type.starts_with("multipart/") {
return None;
}
let name = part.attachment_name().map(|s| s.to_string());
let size = part.len();
Some(BodyPart {
content_type,
name,
size: Some(size),
parts: Vec::new(),
})
}
fn format_size(bytes: usize) -> String {
if bytes >= 1024 * 1024 {
format!("{:.1} MB", bytes as f64 / (1024.0 * 1024.0))
} else if bytes >= 1024 {
format!("{:.1} KB", bytes as f64 / 1024.0)
} else {
format!("{} B", bytes)
}
}
impl fmt::Display for MessageStructure {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
// Display headers as a table
let mut table = Table::new();
table
.load_preset(presets::ASCII_MARKDOWN)
.set_content_arrangement(ContentArrangement::DynamicFullWidth)
.set_header(Row::from([Cell::new("HEADER"), Cell::new("VALUE")]));
if let Some(date) = &self.headers.date {
table.add_row(Row::from([Cell::new("Date"), Cell::new(date)]));
}
if let Some(subject) = &self.headers.subject {
table.add_row(Row::from([Cell::new("Subject"), Cell::new(subject)]));
}
if let Some(message_id) = &self.headers.message_id {
table.add_row(Row::from([Cell::new("Message-ID"), Cell::new(message_id)]));
}
if !self.headers.from.is_empty() {
table.add_row(Row::from([
Cell::new("From"),
Cell::new(self.headers.from.join(", ")),
]));
}
if !self.headers.to.is_empty() {
table.add_row(Row::from([
Cell::new("To"),
Cell::new(self.headers.to.join(", ")),
]));
}
if !self.headers.cc.is_empty() {
table.add_row(Row::from([
Cell::new("Cc"),
Cell::new(self.headers.cc.join(", ")),
]));
}
if !self.headers.bcc.is_empty() {
table.add_row(Row::from([
Cell::new("Bcc"),
Cell::new(self.headers.bcc.join(", ")),
]));
}
if !self.headers.reply_to.is_empty() {
table.add_row(Row::from([
Cell::new("Reply-To"),
Cell::new(self.headers.reply_to.join(", ")),
]));
}
if let Some(in_reply_to) = &self.headers.in_reply_to {
table.add_row(Row::from([
Cell::new("In-Reply-To"),
Cell::new(in_reply_to),
]));
}
writeln!(f)?;
write!(f, "{table}")?;
writeln!(f)?;
// Display body structure
if let Some(body) = &self.body {
writeln!(f, "\nBody structure:")?;
write_body_tree(f, body, "", true)?;
}
writeln!(f)?;
Ok(())
}
}
fn write_body_tree(
f: &mut fmt::Formatter<'_>,
part: &BodyPart,
prefix: &str,
is_last: bool,
) -> fmt::Result {
let connector = if is_last { "└─ " } else { "├─ " };
// Build the part description
let mut desc = part.content_type.clone();
if let Some(name) = &part.name {
desc.push_str(&format!(" \"{}\"", name));
}
if let Some(size) = part.size {
desc.push_str(&format!(" ({})", format_size(size)));
}
writeln!(f, "{}{}{}", prefix, connector, desc)?;
// Handle children
let child_prefix = if is_last {
format!("{} ", prefix)
} else {
format!("{}", prefix)
};
for (i, child) in part.parts.iter().enumerate() {
let is_last_child = i == part.parts.len() - 1;
write_body_tree(f, child, &child_prefix, is_last_child)?;
}
Ok(())
}