NeoMutt  2024-12-12-29-gecf7a5
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
attachments.c
Go to the documentation of this file.
1
30#include "config.h"
31#include <stdbool.h>
32#include <stdint.h>
33#include <stdio.h>
34#include <string.h>
35#include "mutt/lib.h"
36#include "config/lib.h"
37#include "email/lib.h"
38#include "core/lib.h"
39#include "gui/lib.h"
40#include "attachments.h"
41#include "ncrypt/lib.h"
42#include "parse/lib.h"
43#include "mview.h"
44
49{
50 const char *major;
52 const char *minor;
53 regex_t minor_regex;
54};
55
60static struct Notify *AttachmentsNotify = NULL;
61
69static void attachmatch_free(struct AttachMatch **ptr)
70{
71 if (!ptr || !*ptr)
72 return;
73
74 struct AttachMatch *am = *ptr;
75 regfree(&am->minor_regex);
76 FREE(&am->major);
77 FREE(ptr);
78}
79
84static struct AttachMatch *attachmatch_new(void)
85{
86 return MUTT_MEM_CALLOC(1, struct AttachMatch);
87}
88
93{
95
96 /* Lists of AttachMatch */
101}
102
106void attach_init(void)
107{
109 return;
110
113}
114
122static bool count_body_parts_check(struct ListHead *checklist, struct Body *b, bool dflt)
123{
124 /* If list is null, use default behavior. */
125 if (!checklist || STAILQ_EMPTY(checklist))
126 {
127 return false;
128 }
129
130 struct AttachMatch *a = NULL;
131 struct ListNode *np = NULL;
132 STAILQ_FOREACH(np, checklist, entries)
133 {
134 a = (struct AttachMatch *) np->data;
135 mutt_debug(LL_DEBUG3, "%s %d/%s ?? %s/%s [%d]... ", dflt ? "[OK] " : "[EXCL] ",
136 b->type, b->subtype ? b->subtype : "*", a->major, a->minor, a->major_int);
137 if (((a->major_int == TYPE_ANY) || (a->major_int == b->type)) &&
138 (!b->subtype || (regexec(&a->minor_regex, b->subtype, 0, NULL, 0) == 0)))
139 {
140 mutt_debug(LL_DEBUG3, "yes\n");
141 return true;
142 }
143 else
144 {
145 mutt_debug(LL_DEBUG3, "no\n");
146 }
147 }
148
149 return false;
150}
151
157static int count_body_parts(struct Body *b)
158{
159 if (!b)
160 return 0;
161
162 int count = 0;
163
164 for (struct Body *bp = b; bp; bp = bp->next)
165 {
166 /* Initial disposition is to count and not to recurse this part. */
167 bool shallcount = true; /* default */
168 bool shallrecurse = false;
169
170 mutt_debug(LL_DEBUG5, "desc=\"%s\"; fn=\"%s\", type=\"%d/%s\"\n",
171 bp->description ? bp->description : ("none"),
172 bp->filename ? bp->filename :
173 bp->d_filename ? bp->d_filename :
174 "(none)",
175 bp->type, bp->subtype ? bp->subtype : "*");
176
177 if (bp->type == TYPE_MESSAGE)
178 {
179 shallrecurse = true;
180
181 /* If it's an external body pointer, don't recurse it. */
182 if (mutt_istr_equal(bp->subtype, "external-body"))
183 shallrecurse = false;
184 }
185 else if (bp->type == TYPE_MULTIPART)
186 {
187 /* Always recurse multiparts, except multipart/alternative. */
188 shallrecurse = true;
189 if (mutt_istr_equal(bp->subtype, "alternative"))
190 {
191 const bool c_count_alternatives = cs_subset_bool(NeoMutt->sub, "count_alternatives");
192 shallrecurse = c_count_alternatives;
193 }
194 }
195
196 if ((bp->disposition == DISP_INLINE) && (bp->type != TYPE_MULTIPART) &&
197 (bp->type != TYPE_MESSAGE) && (bp == b))
198 {
199 shallcount = false; /* ignore fundamental inlines */
200 }
201
202 /* If this body isn't scheduled for enumeration already, don't bother
203 * profiling it further. */
204 if (shallcount)
205 {
206 /* Turn off shallcount if message type is not in ok list,
207 * or if it is in except list. Check is done separately for
208 * inlines vs. attachments. */
209
210 if (bp->disposition == DISP_ATTACH)
211 {
212 if (!count_body_parts_check(&AttachAllow, bp, true))
213 shallcount = false; /* attach not allowed */
214 if (count_body_parts_check(&AttachExclude, bp, false))
215 shallcount = false; /* attach excluded */
216 }
217 else
218 {
219 if (!count_body_parts_check(&InlineAllow, bp, true))
220 shallcount = false; /* inline not allowed */
221 if (count_body_parts_check(&InlineExclude, bp, false))
222 shallcount = false; /* excluded */
223 }
224 }
225
226 if (shallcount)
227 count++;
228 bp->attach_qualifies = shallcount;
229
230 mutt_debug(LL_DEBUG3, "%p shallcount = %d\n", (void *) bp, shallcount);
231
232 if (shallrecurse)
233 {
234 mutt_debug(LL_DEBUG3, "%p pre count = %d\n", (void *) bp, count);
235 bp->attach_count = count_body_parts(bp->parts);
236 count += bp->attach_count;
237 mutt_debug(LL_DEBUG3, "%p post count = %d\n", (void *) bp, count);
238 }
239 }
240
241 mutt_debug(LL_DEBUG3, "return %d\n", (count < 0) ? 0 : count);
242 return (count < 0) ? 0 : count;
243}
244
251int mutt_count_body_parts(struct Email *e, FILE *fp)
252{
253 if (!e)
254 return 0;
255
256 bool keep_parts = false;
257
258 if (e->attach_valid)
259 return e->attach_total;
260
261 if (e->body->parts)
262 keep_parts = true;
263 else
265
268 {
270 }
271 else
272 {
273 e->attach_total = 0;
274 }
275
276 e->attach_valid = true;
277
278 if (!keep_parts)
280
281 return e->attach_total;
282}
283
289{
290 if (!mv || !mv->mailbox)
291 return;
292
293 struct Mailbox *m = mv->mailbox;
294
295 for (int i = 0; i < m->msg_count; i++)
296 {
297 struct Email *e = m->emails[i];
298 if (!e)
299 break;
300 e->attach_valid = false;
301 e->attach_total = 0;
302 }
303}
304
313static enum CommandResult parse_attach_list(struct Buffer *buf, struct Buffer *s,
314 struct ListHead *head, struct Buffer *err)
315{
316 struct AttachMatch *a = NULL;
317 char *p = NULL;
318 char *tmpminor = NULL;
319 size_t len;
320 int rc;
321
322 do
323 {
325
326 if (!buf->data || (*buf->data == '\0'))
327 continue;
328
329 a = attachmatch_new();
330
331 /* some cheap hacks that I expect to remove */
332 if (mutt_istr_equal(buf->data, "any"))
333 a->major = mutt_str_dup("*/.*");
334 else if (mutt_istr_equal(buf->data, "none"))
335 a->major = mutt_str_dup("cheap_hack/this_should_never_match");
336 else
337 a->major = mutt_str_dup(buf->data);
338
339 p = strchr(a->major, '/');
340 if (p)
341 {
342 *p = '\0';
343 p++;
344 a->minor = p;
345 }
346 else
347 {
348 a->minor = "unknown";
349 }
350
351 len = strlen(a->minor);
352 tmpminor = MUTT_MEM_MALLOC(len + 3, char);
353 strcpy(&tmpminor[1], a->minor);
354 tmpminor[0] = '^';
355 tmpminor[len + 1] = '$';
356 tmpminor[len + 2] = '\0';
357
359 rc = REG_COMP(&a->minor_regex, tmpminor, REG_ICASE);
360
361 FREE(&tmpminor);
362
363 if (rc != 0)
364 {
365 regerror(rc, &a->minor_regex, err->data, err->dsize);
366 FREE(&a->major);
367 FREE(&a);
368 return MUTT_CMD_ERROR;
369 }
370
371 mutt_debug(LL_DEBUG3, "added %s/%s [%d]\n", a->major, a->minor, a->major_int);
372
373 mutt_list_insert_tail(head, (char *) a);
374 } while (MoreArgs(s));
375
376 if (!a)
377 return MUTT_CMD_ERROR;
378
379 mutt_debug(LL_NOTIFY, "NT_ATTACH_ADD: %s/%s\n", a->major, a->minor);
381
382 return MUTT_CMD_SUCCESS;
383}
384
393static enum CommandResult parse_unattach_list(struct Buffer *buf, struct Buffer *s,
394 struct ListHead *head, struct Buffer *err)
395{
396 struct AttachMatch *a = NULL;
397 char *tmp = NULL;
398 char *minor = NULL;
399
400 do
401 {
403 FREE(&tmp);
404
405 if (mutt_istr_equal(buf->data, "any"))
406 tmp = mutt_str_dup("*/.*");
407 else if (mutt_istr_equal(buf->data, "none"))
408 tmp = mutt_str_dup("cheap_hack/this_should_never_match");
409 else
410 tmp = mutt_str_dup(buf->data);
411
412 minor = strchr(tmp, '/');
413 if (minor)
414 {
415 *minor = '\0';
416 minor++;
417 }
418 else
419 {
420 minor = "unknown";
421 }
422 const enum ContentType major = mutt_check_mime_type(tmp);
423
424 struct ListNode *np = NULL, *tmp2 = NULL;
425 STAILQ_FOREACH_SAFE(np, head, entries, tmp2)
426 {
427 a = (struct AttachMatch *) np->data;
428 mutt_debug(LL_DEBUG3, "check %s/%s [%d] : %s/%s [%d]\n", a->major,
429 a->minor, a->major_int, tmp, minor, major);
430 if ((a->major_int == major) && mutt_istr_equal(minor, a->minor))
431 {
432 mutt_debug(LL_DEBUG3, "removed %s/%s [%d]\n", a->major, a->minor, a->major_int);
433 mutt_debug(LL_NOTIFY, "NT_ATTACH_DELETE: %s/%s\n", a->major, a->minor);
434
435 regfree(&a->minor_regex);
436 FREE(&a->major);
437 STAILQ_REMOVE(head, np, ListNode, entries);
438 FREE(&np->data);
439 FREE(&np);
440 }
441 }
442
443 } while (MoreArgs(s));
444
445 FREE(&tmp);
446
448
449 return MUTT_CMD_SUCCESS;
450}
451
459static int print_attach_list(struct ListHead *h, const char op, const char *name)
460{
461 struct ListNode *np = NULL;
462 STAILQ_FOREACH(np, h, entries)
463 {
464 printf("attachments %c%s %s/%s\n", op, name,
465 ((struct AttachMatch *) np->data)->major,
466 ((struct AttachMatch *) np->data)->minor);
467 }
468
469 return 0;
470}
471
475enum CommandResult parse_attachments(struct Buffer *buf, struct Buffer *s,
476 intptr_t data, struct Buffer *err)
477{
479 if (!buf->data || (*buf->data == '\0'))
480 {
481 buf_strcpy(err, _("attachments: no disposition"));
482 return MUTT_CMD_WARNING;
483 }
484
485 char *category = buf->data;
486 char op = *category++;
487
488 if (op == '?')
489 {
490 mutt_endwin();
491 fflush(stdout);
492 printf("\n%s\n\n", _("Current attachments settings:"));
493 print_attach_list(&AttachAllow, '+', "A");
495 print_attach_list(&InlineAllow, '+', "I");
498 return MUTT_CMD_SUCCESS;
499 }
500
501 if ((op != '+') && (op != '-'))
502 {
503 op = '+';
504 category--;
505 }
506
507 struct ListHead *head = NULL;
508 if (mutt_istr_startswith("attachment", category))
509 {
510 if (op == '+')
511 head = &AttachAllow;
512 else
513 head = &AttachExclude;
514 }
515 else if (mutt_istr_startswith("inline", category))
516 {
517 if (op == '+')
518 head = &InlineAllow;
519 else
520 head = &InlineExclude;
521 }
522 else
523 {
524 buf_strcpy(err, _("attachments: invalid disposition"));
525 return MUTT_CMD_ERROR;
526 }
527
528 return parse_attach_list(buf, s, head, err);
529}
530
534enum CommandResult parse_unattachments(struct Buffer *buf, struct Buffer *s,
535 intptr_t data, struct Buffer *err)
536{
537 char op;
538 char *p = NULL;
539 struct ListHead *head = NULL;
540
542 if (!buf->data || (*buf->data == '\0'))
543 {
544 buf_strcpy(err, _("unattachments: no disposition"));
545 return MUTT_CMD_WARNING;
546 }
547
548 p = buf->data;
549 op = *p++;
550
551 if (op == '*')
552 {
557
558 mutt_debug(LL_NOTIFY, "NT_ATTACH_DELETE_ALL\n");
560 return 0;
561 }
562
563 if ((op != '+') && (op != '-'))
564 {
565 op = '+';
566 p--;
567 }
568 if (mutt_istr_startswith("attachment", p))
569 {
570 if (op == '+')
571 head = &AttachAllow;
572 else
573 head = &AttachExclude;
574 }
575 else if (mutt_istr_startswith("inline", p))
576 {
577 if (op == '+')
578 head = &InlineAllow;
579 else
580 head = &InlineExclude;
581 }
582 else
583 {
584 buf_strcpy(err, _("unattachments: invalid disposition"));
585 return MUTT_CMD_ERROR;
586 }
587
588 return parse_unattach_list(buf, s, head, err);
589}
590
596void mutt_parse_mime_message(struct Email *e, FILE *fp)
597{
598 const bool right_type = (e->body->type == TYPE_MESSAGE) ||
599 (e->body->type == TYPE_MULTIPART);
600 const bool not_parsed = (e->body->parts == NULL);
601
602 if (right_type && fp && not_parsed)
603 {
604 mutt_parse_part(fp, e->body);
605 if (WithCrypto)
606 {
607 e->security = crypt_query(e->body);
608 }
609 }
610
611 e->attach_valid = false;
612}
static struct ListHead AttachAllow
List of attachment types to be counted.
Definition: attachments.c:56
static int print_attach_list(struct ListHead *h, const char op, const char *name)
Print a list of attachments.
Definition: attachments.c:459
static int count_body_parts(struct Body *b)
Count the MIME Body parts.
Definition: attachments.c:157
static struct ListHead InlineExclude
List of inline types to ignore.
Definition: attachments.c:59
void mutt_parse_mime_message(struct Email *e, FILE *fp)
Parse a MIME email.
Definition: attachments.c:596
void mutt_attachments_reset(struct MailboxView *mv)
Reset the attachment count for all Emails.
Definition: attachments.c:288
static struct ListHead AttachExclude
List of attachment types to be ignored.
Definition: attachments.c:57
static struct AttachMatch * attachmatch_new(void)
Create a new AttachMatch.
Definition: attachments.c:84
void attach_init(void)
Set up the attachments lists.
Definition: attachments.c:106
void attach_cleanup(void)
Free the attachments lists.
Definition: attachments.c:92
static struct Notify * AttachmentsNotify
Notifications: NotifyAttach.
Definition: attachments.c:60
static enum CommandResult parse_unattach_list(struct Buffer *buf, struct Buffer *s, struct ListHead *head, struct Buffer *err)
Parse the "unattachments" command.
Definition: attachments.c:393
int mutt_count_body_parts(struct Email *e, FILE *fp)
Count the MIME Body parts.
Definition: attachments.c:251
static bool count_body_parts_check(struct ListHead *checklist, struct Body *b, bool dflt)
Compares mime types to the ok and except lists.
Definition: attachments.c:122
static enum CommandResult parse_attach_list(struct Buffer *buf, struct Buffer *s, struct ListHead *head, struct Buffer *err)
Parse the "attachments" command.
Definition: attachments.c:313
static struct ListHead InlineAllow
List of inline types to counted.
Definition: attachments.c:58
Miscellaneous email parsing routines.
@ NT_ATTACH_DELETE
Attachment regex has been deleted.
Definition: attachments.h:41
@ NT_ATTACH_DELETE_ALL
All Attachment regexes have been deleted.
Definition: attachments.h:42
@ NT_ATTACH_ADD
Attachment regex has been added.
Definition: attachments.h:40
size_t buf_strcpy(struct Buffer *buf, const char *s)
Copy a string into a Buffer.
Definition: buffer.c:395
CommandResult
Error codes for command_t parse functions.
Definition: command.h:36
@ MUTT_CMD_SUCCESS
Success: Command worked.
Definition: command.h:39
@ MUTT_CMD_ERROR
Error: Can't help the user.
Definition: command.h:37
@ MUTT_CMD_WARNING
Warning: Help given to the user.
Definition: command.h:38
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition: helpers.c:47
Convenience wrapper for the config headers.
Convenience wrapper for the core headers.
SecurityFlags crypt_query(struct Body *b)
Check out the type of encryption used.
Definition: crypt.c:687
int mutt_any_key_to_continue(const char *s)
Prompt the user to 'press any key' and wait.
Definition: curs_lib.c:173
void mutt_endwin(void)
Shutdown curses.
Definition: curs_lib.c:151
void mutt_body_free(struct Body **ptr)
Free a Body.
Definition: body.c:58
Structs that make up an email.
void mutt_parse_part(FILE *fp, struct Body *b)
Parse a MIME part.
Definition: parse.c:1822
enum ContentType mutt_check_mime_type(const char *s)
Check a MIME type string.
Definition: parse.c:366
int parse_extract_token(struct Buffer *dest, struct Buffer *tok, TokenFlags flags)
Extract one token from a string.
Definition: extract.c:50
#define MoreArgs(buf)
Definition: extract.h:32
#define TOKEN_NO_FLAGS
No flags are set.
Definition: extract.h:46
enum CommandResult parse_unattachments(struct Buffer *buf, struct Buffer *s, intptr_t data, struct Buffer *err)
Parse the 'unattachments' command - Implements Command::parse() -.
Definition: attachments.c:534
enum CommandResult parse_attachments(struct Buffer *buf, struct Buffer *s, intptr_t data, struct Buffer *err)
Parse the 'attachments' command - Implements Command::parse() -.
Definition: attachments.c:475
static void attachmatch_free(struct AttachMatch **ptr)
Free an AttachMatch - Implements list_free_t -.
Definition: attachments.c:69
#define mutt_debug(LEVEL,...)
Definition: logging2.h:89
Convenience wrapper for the gui headers.
struct ListNode * mutt_list_insert_tail(struct ListHead *h, char *s)
Append a string to the end of a List.
Definition: list.c:65
void mutt_list_free_type(struct ListHead *h, list_free_t fn)
Free a List of type.
Definition: list.c:144
void(* list_free_t)(void **ptr)
Definition: list.h:50
@ LL_DEBUG3
Log at debug level 3.
Definition: logging2.h:45
@ LL_DEBUG5
Log at debug level 5.
Definition: logging2.h:47
@ LL_NOTIFY
Log of notifications.
Definition: logging2.h:48
#define FREE(x)
Definition: memory.h:55
#define MUTT_MEM_CALLOC(n, type)
Definition: memory.h:40
#define MUTT_MEM_MALLOC(n, type)
Definition: memory.h:41
ContentType
Content-Type.
Definition: mime.h:30
@ TYPE_MESSAGE
Type: 'message/*'.
Definition: mime.h:35
@ TYPE_MULTIPART
Type: 'multipart/*'.
Definition: mime.h:37
@ TYPE_ANY
Type: '*' or '.*'.
Definition: mime.h:40
@ DISP_ATTACH
Content is attached.
Definition: mime.h:63
@ DISP_INLINE
Content is inline.
Definition: mime.h:62
Convenience wrapper for the library headers.
#define _(a)
Definition: message.h:28
struct Notify * notify_new(void)
Create a new notifications handler.
Definition: notify.c:62
bool notify_send(struct Notify *notify, enum NotifyType event_type, int event_subtype, void *event_data)
Send out a notification message.
Definition: notify.c:173
void notify_set_parent(struct Notify *notify, struct Notify *parent)
Set the parent notification handler.
Definition: notify.c:95
void notify_free(struct Notify **ptr)
Free a notification handler.
Definition: notify.c:75
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition: string.c:672
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition: string.c:253
size_t mutt_istr_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix, ignoring case.
Definition: string.c:242
View of a Mailbox.
API for encryption/signing of emails.
#define WithCrypto
Definition: lib.h:122
@ NT_ATTACH
Attachment command changed, NotifyAttach.
Definition: notify_type.h:39
Text parsing functions.
#define STAILQ_REMOVE(head, elm, type, field)
Definition: queue.h:402
#define STAILQ_HEAD_INITIALIZER(head)
Definition: queue.h:324
#define STAILQ_FOREACH(var, head, field)
Definition: queue.h:352
#define STAILQ_EMPTY(head)
Definition: queue.h:348
#define STAILQ_FOREACH_SAFE(var, head, field, tvar)
Definition: queue.h:362
#define REG_COMP(preg, regex, cflags)
Compile a regular expression.
Definition: regex3.h:50
An attachment matching a regex for attachment counter.
Definition: attachments.c:49
const char * minor
Minor mime type, e.g. "html".
Definition: attachments.c:52
regex_t minor_regex
Minor mime type regex.
Definition: attachments.c:53
const char * major
Major mime type, e.g. "text".
Definition: attachments.c:50
enum ContentType major_int
Major mime type, e.g. TYPE_TEXT.
Definition: attachments.c:51
The body of an email.
Definition: body.h:36
struct Body * parts
parts of a multipart or message/rfc822
Definition: body.h:73
struct Body * next
next attachment in the list
Definition: body.h:72
char * subtype
content-type subtype
Definition: body.h:61
unsigned int type
content-type primary type, ContentType
Definition: body.h:40
String manipulation buffer.
Definition: buffer.h:36
size_t dsize
Length of data.
Definition: buffer.h:39
char * data
Pointer to data.
Definition: buffer.h:37
The envelope/body of an email.
Definition: email.h:39
bool attach_valid
true when the attachment count is valid
Definition: email.h:100
SecurityFlags security
bit 0-10: flags, bit 11,12: application, bit 13: traditional pgp See: ncrypt/lib.h pgplib....
Definition: email.h:43
struct Body * body
List of MIME parts.
Definition: email.h:69
short attach_total
Number of qualifying attachments in message, if attach_valid.
Definition: email.h:115
A List node for strings.
Definition: list.h:37
char * data
String.
Definition: list.h:38
View of a Mailbox.
Definition: mview.h:40
struct Mailbox * mailbox
Current Mailbox.
Definition: mview.h:51
A mailbox.
Definition: mailbox.h:79
int msg_count
Total number of messages.
Definition: mailbox.h:88
struct Email ** emails
Array of Emails.
Definition: mailbox.h:96
Container for Accounts, Notifications.
Definition: neomutt.h:42
struct Notify * notify
Notifications handler.
Definition: neomutt.h:43
struct ConfigSubset * sub
Inherited config items.
Definition: neomutt.h:46
Notification API.
Definition: notify.c:53