NeoMutt  2025-12-11-911-gd8d604
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
parse.c
Go to the documentation of this file.
1
29
35
36#include "config.h"
37#include <errno.h>
38#include <string.h>
39#include <time.h>
40#include "mutt/lib.h"
41#include "address/lib.h"
42#include "config/lib.h"
43#include "core/lib.h"
44#include "alias/lib.h"
45#include "mutt.h"
46#include "parse.h"
47#include "body.h"
48#include "email.h"
49#include "envelope.h"
50#include "from.h"
51#include "mime.h"
52#include "module_data.h"
53#include "parameter.h"
54#include "rfc2047.h"
55#include "rfc2231.h"
56#include "url.h"
57#ifdef USE_AUTOCRYPT
58#include "autocrypt/lib.h"
59#endif
60
62#define CONTENT_TOO_BIG (1 << 30)
63
64static void parse_part(FILE *fp, struct Body *b, int *counter);
65static struct Body *rfc822_parse_message(FILE *fp, struct Body *parent, int *counter);
66static struct Body *parse_multipart(FILE *fp, const char *boundary,
67 LOFF_T end_off, bool digest, int *counter);
68
74{
75 if (!header)
76 return;
77
78 for (; (*header != '\0'); header++)
79 {
80 if ((*header < 33) || (*header > 126) || (*header == ':'))
81 *header = '?';
82 }
83}
84
94{
95 if (!header)
96 return;
97
98 for (; (*header != '\0'); header++)
99 {
100 if ((*header == '\n') || (*header == '\r'))
101 *header = ' ';
102 }
103}
104
116static void parse_parameters(struct ParameterList *pl, const char *s, bool allow_value_spaces)
117{
118 struct Parameter *pnew = NULL;
119 const char *p = NULL;
120 size_t i;
121
122 struct Buffer *buf = buf_pool_get();
123 /* allow_value_spaces, especially with autocrypt keydata, can result
124 * in quite large parameter values. avoid frequent reallocs by
125 * pre-sizing */
126 if (allow_value_spaces)
127 buf_alloc(buf, mutt_str_len(s));
128
129 mutt_debug(LL_DEBUG2, "'%s'\n", s);
130
131 const bool assumed = !slist_is_empty(cc_assumed_charset());
132 while (*s)
133 {
134 buf_reset(buf);
135
136 p = strpbrk(s, "=;");
137 if (!p)
138 {
139 mutt_debug(LL_DEBUG1, "malformed parameter: %s\n", s);
140 goto bail;
141 }
142
143 /* if we hit a ; now the parameter has no value, just skip it */
144 if (*p != ';')
145 {
146 i = p - s;
147 /* remove whitespace from the end of the attribute name */
148 while ((i > 0) && mutt_str_is_email_wsp(s[i - 1]))
149 i--;
150
151 /* the check for the missing parameter token is here so that we can skip
152 * over any quoted value that may be present. */
153 if (i == 0)
154 {
155 mutt_debug(LL_DEBUG1, "missing attribute: %s\n", s);
156 pnew = NULL;
157 }
158 else
159 {
160 pnew = mutt_param_new();
161 pnew->attribute = mutt_strn_dup(s, i);
162 }
163
164 do
165 {
166 s = mutt_str_skip_email_wsp(p + 1); /* skip over the =, or space if we loop */
167
168 if (*s == '"')
169 {
170 bool state_ascii = true;
171 s++;
172 for (; *s; s++)
173 {
174 if (assumed)
175 {
176 // As iso-2022-* has a character of '"' with non-ascii state, ignore it
177 if (*s == 0x1b)
178 {
179 if ((s[1] == '(') && ((s[2] == 'B') || (s[2] == 'J')))
180 state_ascii = true;
181 else
182 state_ascii = false;
183 }
184 }
185 if (state_ascii && (*s == '"'))
186 break;
187 if (*s == '\\')
188 {
189 if (s[1])
190 {
191 s++;
192 /* Quote the next character */
193 buf_addch(buf, *s);
194 }
195 }
196 else
197 {
198 buf_addch(buf, *s);
199 }
200 }
201 if (*s)
202 s++; /* skip over the " */
203 }
204 else
205 {
206 for (; *s && *s != ' ' && *s != ';'; s++)
207 buf_addch(buf, *s);
208 }
209
210 p = s;
211 } while (allow_value_spaces && (*s == ' '));
212
213 /* if the attribute token was missing, 'new' will be NULL */
214 if (pnew)
215 {
216 pnew->value = buf_strdup(buf);
217
218 mutt_debug(LL_DEBUG2, "parse_parameter: '%s' = '%s'\n",
219 pnew->attribute ? pnew->attribute : "", pnew->value ? pnew->value : "");
220
221 /* Add this parameter to the list */
222 TAILQ_INSERT_HEAD(pl, pnew, entries);
223 }
224 }
225 else
226 {
227 mutt_debug(LL_DEBUG1, "parameter with no value: %s\n", s);
228 s = p;
229 }
230
231 /* Find the next parameter */
232 if ((*s != ';') && !(s = strchr(s, ';')))
233 break; /* no more parameters */
234
235 do
236 {
237 /* Move past any leading whitespace. the +1 skips over the semicolon */
238 s = mutt_str_skip_email_wsp(s + 1);
239 } while (*s == ';'); /* skip empty parameters */
240 }
241
242bail:
243
245 buf_pool_release(&buf);
246}
247
255static void parse_content_disposition(const char *s, struct Body *b)
256{
257 struct ParameterList pl = TAILQ_HEAD_INITIALIZER(pl);
258
259 if (mutt_istr_startswith(s, "inline"))
261 else if (mutt_istr_startswith(s, "form-data"))
263 else
265
266 /* Check to see if a default filename was given */
267 s = strchr(s, ';');
268 if (s)
269 {
270 s = mutt_str_skip_email_wsp(s + 1);
271 parse_parameters(&pl, s, false);
272 s = mutt_param_get(&pl, "filename");
273 if (s)
275 s = mutt_param_get(&pl, "name");
276 if (s)
278 mutt_param_free(&pl);
279 }
280}
281
287static void parse_references(struct ListHead *head, const char *s)
288{
289 if (!head)
290 return;
291
292 char *m = NULL;
293 for (size_t off = 0; (m = mutt_extract_message_id(s, &off)); s += off)
294 {
295 mutt_list_insert_head(head, m);
296 }
297}
298
304static void parse_content_language(const char *s, struct Body *b)
305{
306 if (!s || !b)
307 return;
308
309 mutt_debug(LL_DEBUG2, "RFC8255 >> Content-Language set to %s\n", s);
311}
312
320bool mutt_matches_ignore(const char *s)
321{
323 ASSERT(mod_data);
324
325 return mutt_list_match(s, &mod_data->ignore) && !mutt_list_match(s, &mod_data->unignore);
326}
327
334{
335 if (mutt_istr_equal("text", s))
336 return TYPE_TEXT;
337 if (mutt_istr_equal("multipart", s))
338 return TYPE_MULTIPART;
339 if (mutt_istr_equal("x-sun-attachment", s))
340 return TYPE_MULTIPART;
341 if (mutt_istr_equal("application", s))
342 return TYPE_APPLICATION;
343 if (mutt_istr_equal("message", s))
344 return TYPE_MESSAGE;
345 if (mutt_istr_equal("image", s))
346 return TYPE_IMAGE;
347 if (mutt_istr_equal("audio", s))
348 return TYPE_AUDIO;
349 if (mutt_istr_equal("video", s))
350 return TYPE_VIDEO;
351 if (mutt_istr_equal("model", s))
352 return TYPE_MODEL;
353 if (mutt_istr_equal("*", s))
354 return TYPE_ANY;
355 if (mutt_istr_equal(".*", s))
356 return TYPE_ANY;
357
358 return TYPE_OTHER;
359}
360
368char *mutt_extract_message_id(const char *s, size_t *len)
369{
370 if (!s || (*s == '\0'))
371 return NULL;
372
373 char *decoded = mutt_str_dup(s);
374 rfc2047_decode(&decoded);
375
376 char *res = NULL;
377
378 for (const char *p = decoded, *beg = NULL; *p; p++)
379 {
380 if (*p == '<')
381 {
382 beg = p;
383 continue;
384 }
385
386 if (beg && (*p == '>'))
387 {
388 if (len)
389 *len = p - decoded + 1;
390 res = mutt_strn_dup(beg, (p + 1) - beg);
391 break;
392 }
393 }
394
395 FREE(&decoded);
396 return res;
397}
398
404int mutt_check_encoding(const char *c)
405{
406 if (mutt_istr_startswith(c, "7bit"))
407 return ENC_7BIT;
408 if (mutt_istr_startswith(c, "8bit"))
409 return ENC_8BIT;
410 if (mutt_istr_startswith(c, "binary"))
411 return ENC_BINARY;
412 if (mutt_istr_startswith(c, "quoted-printable"))
414 if (mutt_istr_startswith(c, "base64"))
415 return ENC_BASE64;
416 if (mutt_istr_startswith(c, "x-uuencode"))
417 return ENC_UUENCODED;
418 if (mutt_istr_startswith(c, "uuencode"))
419 return ENC_UUENCODED;
420 return ENC_OTHER;
421}
422
433void mutt_parse_content_type(const char *s, struct Body *b)
434{
435 if (!s || !b)
436 return;
437
438 FREE(&b->subtype);
440
441 /* First extract any existing parameters */
442 char *pc = strchr(s, ';');
443 if (pc)
444 {
445 *pc++ = 0;
446 while (*pc && mutt_isspace(*pc))
447 pc++;
448 parse_parameters(&b->parameter, pc, false);
449
450 /* Some pre-RFC1521 gateways still use the "name=filename" convention,
451 * but if a filename has already been set in the content-disposition,
452 * let that take precedence, and don't set it here */
453 pc = mutt_param_get(&b->parameter, "name");
454 if (pc && !b->filename)
455 b->filename = mutt_str_dup(pc);
456
457 /* this is deep and utter perversion */
458 pc = mutt_param_get(&b->parameter, "conversions");
459 if (pc)
461 }
462
463 /* Now get the subtype */
464 char *subtype = strchr(s, '/');
465 if (subtype)
466 {
467 *subtype++ = '\0';
468 for (pc = subtype; *pc && !mutt_isspace(*pc) && (*pc != ';'); pc++)
469 ; // do nothing
470
471 *pc = '\0';
472 mutt_str_replace(&b->subtype, subtype);
473 }
474
475 /* Finally, get the major type */
477
478 if (mutt_istr_equal("x-sun-attachment", s))
479 mutt_str_replace(&b->subtype, "x-sun-attachment");
480
481 if (b->type == TYPE_OTHER)
482 {
483 mutt_str_replace(&b->xtype, s);
484 }
485
486 if (!b->subtype)
487 {
488 /* Some older non-MIME mailers (i.e., mailtool, elm) have a content-type
489 * field, so we can attempt to convert the type to Body here. */
490 if (b->type == TYPE_TEXT)
491 {
492 b->subtype = mutt_str_dup("plain");
493 }
494 else if (b->type == TYPE_AUDIO)
495 {
496 b->subtype = mutt_str_dup("basic");
497 }
498 else if (b->type == TYPE_MESSAGE)
499 {
500 b->subtype = mutt_str_dup("rfc822");
501 }
502 else if (b->type == TYPE_OTHER)
503 {
504 char buf[128] = { 0 };
505
507 snprintf(buf, sizeof(buf), "x-%s", s);
508 b->subtype = mutt_str_dup(buf);
509 }
510 else
511 {
512 b->subtype = mutt_str_dup("x-unknown");
513 }
514 }
515
516 /* Default character set for text types. */
517 if (b->type == TYPE_TEXT)
518 {
519 pc = mutt_param_get(&b->parameter, "charset");
520 if (pc)
521 {
522 /* Microsoft Outlook seems to think it is necessary to repeat
523 * charset=, strip it off not to confuse ourselves */
524 if (mutt_istrn_equal(pc, "charset=", sizeof("charset=") - 1))
525 mutt_param_set(&b->parameter, "charset", pc + (sizeof("charset=") - 1));
526 }
527 else
528 {
529 mutt_param_set(&b->parameter, "charset",
531 }
532 }
533}
534
535#ifdef USE_AUTOCRYPT
542static struct AutocryptHeader *parse_autocrypt(struct AutocryptHeader *head, const char *s)
543{
544 struct AutocryptHeader *autocrypt = mutt_autocrypthdr_new();
545 autocrypt->next = head;
546
547 struct ParameterList pl = TAILQ_HEAD_INITIALIZER(pl);
548 parse_parameters(&pl, s, true);
549 if (TAILQ_EMPTY(&pl))
550 {
551 autocrypt->invalid = true;
552 goto cleanup;
553 }
554
555 struct Parameter *p = NULL;
556 TAILQ_FOREACH(p, &pl, entries)
557 {
558 if (mutt_istr_equal(p->attribute, "addr"))
559 {
560 if (autocrypt->addr)
561 {
562 autocrypt->invalid = true;
563 goto cleanup;
564 }
565 autocrypt->addr = p->value;
566 p->value = NULL;
567 }
568 else if (mutt_istr_equal(p->attribute, "prefer-encrypt"))
569 {
570 if (mutt_istr_equal(p->value, "mutual"))
571 autocrypt->prefer_encrypt = true;
572 }
573 else if (mutt_istr_equal(p->attribute, "keydata"))
574 {
575 if (autocrypt->keydata)
576 {
577 autocrypt->invalid = true;
578 goto cleanup;
579 }
580 autocrypt->keydata = p->value;
581 p->value = NULL;
582 }
583 else if (p->attribute && (p->attribute[0] != '_'))
584 {
585 autocrypt->invalid = true;
586 goto cleanup;
587 }
588 }
589
590 /* Checking the addr against From, and for multiple valid headers
591 * occurs later, after all the headers are parsed. */
592 if (!autocrypt->addr || !autocrypt->keydata)
593 autocrypt->invalid = true;
594
595cleanup:
596 mutt_param_free(&pl);
597 return autocrypt;
598}
599#endif
600
606char *mutt_rfc2369_first_mailto(const char *body)
607{
608 if (!body)
609 return NULL;
610
611 for (const char *beg = body, *end = NULL; beg; beg = strchr(end, ','))
612 {
613 beg = strchr(beg, '<');
614 if (!beg)
615 {
616 break;
617 }
618 beg++;
619 end = strchr(beg, '>');
620 if (!end)
621 {
622 break;
623 }
624
625 char *mlist = mutt_strn_dup(beg, end - beg);
626 if (url_check_scheme(mlist) == U_MAILTO)
627 {
628 return mlist;
629 }
630 FREE(&mlist);
631 }
632 return NULL;
633}
634
640{
641 if (!headers)
642 return;
643
644 FREE(&headers->archive);
645 FREE(&headers->help);
646 FREE(&headers->owner);
647 FREE(&headers->post);
648 FREE(&headers->subscribe);
649 FREE(&headers->unsubscribe);
650}
651
657void mutt_rfc2369_read_headers(FILE *fp, struct Rfc2369ListHeaders *headers)
658{
659 if (!fp || !headers)
660 return;
661
663
664 struct Buffer *line = buf_pool_get();
665 while (true)
666 {
667 size_t len = mutt_rfc822_read_line(fp, line);
668 if ((len == 0) || buf_is_empty(line))
669 break;
670
671 const char *lines = buf_string(line);
672 const char *p = strpbrk(lines, ": \t");
673 if (!p || (*p != ':'))
674 {
675 time_t t = 0;
676
677 /* Some MTAs quote the original mbox separator. */
678 if (mutt_str_startswith(lines, ">From "))
679 continue;
680 if (is_from(lines, NULL, 0, &t))
681 continue;
682
683 break;
684 }
685
686 const size_t name_len = p - lines;
687 const char *body = mutt_str_skip_whitespace(p + 1);
688 if (*body == '\0')
689 continue;
690
691 char **value = NULL;
692 if ((name_len == 12) && eqi12(lines, "List-Archive"))
693 value = &headers->archive;
694 else if ((name_len == 9) && eqi9(lines, "List-Help"))
695 value = &headers->help;
696 else if ((name_len == 10) && eqi10(lines, "List-Owner"))
697 value = &headers->owner;
698 else if ((name_len == 9) && eqi9(lines, "List-Post"))
699 value = &headers->post;
700 else if ((name_len == 14) && eqi14(lines, "List-Subscribe"))
701 value = &headers->subscribe;
702 else if ((name_len == 16) && eqi16(lines, "List-Unsubscribe"))
703 value = &headers->unsubscribe;
704
705 if (!value)
706 continue;
707
708 char *mailto = mutt_rfc2369_first_mailto(body);
709 if (mailto)
710 {
711 FREE(value);
712 *value = mailto;
713 }
714 }
715 buf_pool_release(&line);
716}
717
734int mutt_rfc822_parse_line(struct Envelope *env, struct Email *e,
735 const char *name, size_t name_len, const char *body,
736 bool user_hdrs, bool weed, bool do_2047)
737{
738 if (!env || !name)
739 return 0;
740
741 bool matched = false;
742
743 switch (name[0] | 0x20)
744 {
745 case 'a':
746 if ((name_len == 13) && eqi12(name + 1, "pparently-to"))
747 {
748 mutt_addrlist_parse(&env->to, body);
749 matched = true;
750 }
751 else if ((name_len == 15) && eqi14(name + 1, "pparently-from"))
752 {
753 mutt_addrlist_parse(&env->from, body);
754 matched = true;
755 }
756#ifdef USE_AUTOCRYPT
757 else if ((name_len == 9) && eqi8(name + 1, "utocrypt"))
758 {
759 const bool c_autocrypt = cs_subset_bool(NeoMutt->sub, "autocrypt");
760 if (c_autocrypt)
761 {
762 env->autocrypt = parse_autocrypt(env->autocrypt, body);
763 matched = true;
764 }
765 }
766 else if ((name_len == 16) && eqi15(name + 1, "utocrypt-gossip"))
767 {
768 const bool c_autocrypt = cs_subset_bool(NeoMutt->sub, "autocrypt");
769 if (c_autocrypt)
770 {
772 matched = true;
773 }
774 }
775#endif
776 break;
777
778 case 'b':
779 if ((name_len == 3) && eqi2(name + 1, "cc"))
780 {
781 mutt_addrlist_parse(&env->bcc, body);
782 matched = true;
783 }
784 break;
785
786 case 'c':
787 if ((name_len == 2) && eqi1(name + 1, "c"))
788 {
789 mutt_addrlist_parse(&env->cc, body);
790 matched = true;
791 }
792 else
793 {
794 if ((name_len >= 12) && eqi8(name, "content-"))
795 {
796 if ((name_len == 12) && eqi4(name + 8, "type"))
797 {
798 if (e)
800 matched = true;
801 }
802 else if ((name_len == 16) && eqi8(name + 8, "language"))
803 {
804 if (e)
806 matched = true;
807 }
808 else if ((name_len == 25) && eqi17(name + 8, "transfer-encoding"))
809 {
810 if (e)
812 matched = true;
813 }
814 else if ((name_len == 14) && eqi8(name + 6, "t-length"))
815 {
816 if (e)
817 {
818 unsigned long len = 0;
819 e->body->length = mutt_str_atoul(body, &len) ? MIN(len, CONTENT_TOO_BIG) : -1;
820 }
821 matched = true;
822 }
823 else if ((name_len == 19) && eqi11(name + 8, "description"))
824 {
825 if (e)
826 {
829 }
830 matched = true;
831 }
832 else if ((name_len == 19) && eqi11(name + 8, "disposition"))
833 {
834 if (e)
836 matched = true;
837 }
838 }
839 }
840 break;
841
842 case 'd':
843 if ((name_len != 4) || !eqi4(name, "date"))
844 break;
845
846 mutt_str_replace(&env->date, body);
847 if (e)
848 {
849 struct Tz tz = { 0 };
850 // the caller will check e->date_sent for -1
851 e->date_sent = mutt_date_parse_date(body, &tz);
852 if (e->date_sent > 0)
853 {
854 e->zhours = tz.zhours;
855 e->zminutes = tz.zminutes;
856 e->zoccident = tz.zoccident;
857 }
858 }
859 matched = true;
860 break;
861
862 case 'e':
863 if ((name_len == 7) && eqi6(name + 1, "xpires") && e)
864 {
865 const time_t expired = mutt_date_parse_date(body, NULL);
866 if ((expired != -1) && (expired < mutt_date_now()))
867 {
868 e->expired = true;
869 }
870 }
871 break;
872
873 case 'f':
874 if ((name_len == 4) && eqi4(name, "from"))
875 {
876 mutt_addrlist_parse(&env->from, body);
877 matched = true;
878 }
879 else if ((name_len == 11) && eqi10(name + 1, "ollowup-to"))
880 {
881 if (!env->followup_to)
882 {
885 }
886 matched = true;
887 }
888 break;
889
890 case 'i':
891 if ((name_len != 11) || !eqi10(name + 1, "n-reply-to"))
892 break;
893
895 char *body2 = mutt_str_dup(body); // Create a mutable copy
897 parse_references(&env->in_reply_to, body2);
898 FREE(&body2);
899 matched = true;
900 break;
901
902 case 'l':
903 if ((name_len == 5) && eqi4(name + 1, "ines"))
904 {
905 if (e)
906 {
907 unsigned int ui = 0; // we don't want a negative number of lines
908 mutt_str_atoui(body, &ui);
909 e->lines = ui;
910 }
911
912 matched = true;
913 }
914 else if ((name_len == 9) && eqi8(name + 1, "ist-post"))
915 {
916 /* RFC2369 */
917 if (!mutt_strn_equal(mutt_str_skip_whitespace(body), "NO", 2))
918 {
919 char *mailto = mutt_rfc2369_first_mailto(body);
920 if (mailto)
921 {
922 FREE(&env->list_post);
923 env->list_post = mailto;
924 const bool c_auto_subscribe = cs_subset_bool(NeoMutt->sub, "auto_subscribe");
925 if (c_auto_subscribe)
927 }
928 }
929 matched = true;
930 }
931 else if ((name_len == 14) && eqi13(name + 1, "ist-subscribe"))
932 {
933 /* RFC2369 */
934 char *mailto = mutt_rfc2369_first_mailto(body);
935 if (mailto)
936 {
937 FREE(&env->list_subscribe);
938 env->list_subscribe = mailto;
939 }
940 matched = true;
941 }
942 else if ((name_len == 16) && eqi15(name + 1, "ist-unsubscribe"))
943 {
944 /* RFC2369 */
945 char *mailto = mutt_rfc2369_first_mailto(body);
946 if (mailto)
947 {
948 FREE(&env->list_unsubscribe);
949 env->list_unsubscribe = mailto;
950 }
951 matched = true;
952 }
953 break;
954
955 case 'm':
956 if ((name_len == 12) && eqi11(name + 1, "ime-version"))
957 {
958 if (e)
959 e->mime = true;
960 matched = true;
961 }
962 else if ((name_len == 10) && eqi9(name + 1, "essage-id"))
963 {
964 /* We add a new "Message-ID:" when building a message */
965 FREE(&env->message_id);
966 env->message_id = mutt_extract_message_id(body, NULL);
967 matched = true;
968 }
969 else
970 {
971 if ((name_len >= 13) && eqi4(name + 1, "ail-"))
972 {
973 if ((name_len == 13) && eqi8(name + 5, "reply-to"))
974 {
975 /* override the Reply-To: field */
977 mutt_addrlist_parse(&env->reply_to, body);
978 matched = true;
979 }
980 else if ((name_len == 16) && eqi11(name + 5, "followup-to"))
981 {
983 matched = true;
984 }
985 }
986 }
987 break;
988
989 case 'n':
990 if ((name_len == 10) && eqi9(name + 1, "ewsgroups"))
991 {
992 FREE(&env->newsgroups);
995 matched = true;
996 }
997 break;
998
999 case 'o':
1000 /* field 'Organization:' saves only for pager! */
1001 if ((name_len == 12) && eqi11(name + 1, "rganization"))
1002 {
1003 if (!env->organization && !mutt_istr_equal(body, "unknown"))
1004 env->organization = mutt_str_dup(body);
1005 }
1006 break;
1007
1008 case 'r':
1009 if ((name_len == 10) && eqi9(name + 1, "eferences"))
1010 {
1012 parse_references(&env->references, body);
1013 matched = true;
1014 }
1015 else if ((name_len == 8) && eqi8(name, "reply-to"))
1016 {
1017 mutt_addrlist_parse(&env->reply_to, body);
1018 matched = true;
1019 }
1020 else if ((name_len == 11) && eqi10(name + 1, "eturn-path"))
1021 {
1022 mutt_addrlist_parse(&env->return_path, body);
1023 matched = true;
1024 }
1025 else if ((name_len == 8) && eqi8(name, "received"))
1026 {
1027 if (e && (e->received == 0))
1028 {
1029 const char *d = strrchr(body, ';');
1030 if (d)
1031 {
1032 d = mutt_str_skip_email_wsp(d + 1);
1033 // the caller will check e->received for -1
1034 e->received = mutt_date_parse_date(d, NULL);
1035 }
1036 }
1037 }
1038 break;
1039
1040 case 's':
1041 if ((name_len == 7) && eqi6(name + 1, "ubject"))
1042 {
1043 if (!env->subject)
1044 mutt_env_set_subject(env, body);
1045 matched = true;
1046 }
1047 else if ((name_len == 6) && eqi5(name + 1, "ender"))
1048 {
1049 mutt_addrlist_parse(&env->sender, body);
1050 matched = true;
1051 }
1052 else if ((name_len == 6) && eqi5(name + 1, "tatus"))
1053 {
1054 if (e)
1055 {
1056 while (*body)
1057 {
1058 switch (*body)
1059 {
1060 case 'O':
1061 {
1062 e->old = true;
1063 break;
1064 }
1065 case 'R':
1066 e->read = true;
1067 break;
1068 case 'r':
1069 e->replied = true;
1070 break;
1071 }
1072 body++;
1073 }
1074 }
1075 matched = true;
1076 }
1077 else if (e && (name_len == 10) && eqi1(name + 1, "u") &&
1078 (eqi8(name + 2, "persedes") || eqi8(name + 2, "percedes")))
1079 {
1080 FREE(&env->supersedes);
1081 env->supersedes = mutt_str_dup(body);
1082 }
1083 break;
1084
1085 case 't':
1086 if ((name_len == 2) && eqi1(name + 1, "o"))
1087 {
1088 mutt_addrlist_parse(&env->to, body);
1089 matched = true;
1090 }
1091 break;
1092
1093 case 'x':
1094 if ((name_len == 8) && eqi8(name, "x-status"))
1095 {
1096 if (e)
1097 {
1098 while (*body)
1099 {
1100 switch (*body)
1101 {
1102 case 'A':
1103 e->replied = true;
1104 break;
1105 case 'D':
1106 e->deleted = true;
1107 break;
1108 case 'F':
1109 e->flagged = true;
1110 break;
1111 default:
1112 break;
1113 }
1114 body++;
1115 }
1116 }
1117 matched = true;
1118 }
1119 else if ((name_len == 7) && eqi6(name + 1, "-label"))
1120 {
1121 FREE(&env->x_label);
1122 env->x_label = mutt_str_dup(body);
1123 matched = true;
1124 }
1125 else if ((name_len == 12) && eqi11(name + 1, "-comment-to"))
1126 {
1127 if (!env->x_comment_to)
1128 env->x_comment_to = mutt_str_dup(body);
1129 matched = true;
1130 }
1131 else if ((name_len == 4) && eqi4(name, "xref"))
1132 {
1133 if (!env->xref)
1134 env->xref = mutt_str_dup(body);
1135 matched = true;
1136 }
1137 else if ((name_len == 13) && eqi12(name + 1, "-original-to"))
1138 {
1140 matched = true;
1141 }
1142 break;
1143
1144 default:
1145 break;
1146 }
1147
1148 /* Keep track of the user-defined headers */
1149 if (!matched && user_hdrs)
1150 {
1151 const bool c_weed = cs_subset_bool(NeoMutt->sub, "weed");
1152 char *dup = NULL;
1153 mutt_str_asprintf(&dup, "%s: %s", name, body);
1154
1155 if (!weed || !c_weed || !mutt_matches_ignore(dup))
1156 {
1157 struct ListNode *np = mutt_list_insert_tail(&env->userhdrs, dup);
1158 if (do_2047)
1159 {
1160 rfc2047_decode(&np->data);
1161 }
1162 }
1163 else
1164 {
1165 FREE(&dup);
1166 }
1167 }
1168
1169 return matched;
1170}
1171
1181size_t mutt_rfc822_read_line(FILE *fp, struct Buffer *buf)
1182{
1183 if (!fp || !buf)
1184 return 0;
1185
1186 size_t read = 0;
1187 char line[1024] = { 0 }; /* RFC2822 specifies a maximum line length of 998 */
1188
1189 buf_reset(buf);
1190 while (true)
1191 {
1192 if (!fgets(line, sizeof(line), fp))
1193 {
1194 return 0;
1195 }
1196
1197 const size_t linelen = mutt_str_len(line);
1198 if (linelen == 0)
1199 {
1200 break;
1201 }
1202
1203 if (mutt_str_is_email_wsp(line[0]) && buf_is_empty(buf))
1204 {
1205 read = linelen;
1206 break;
1207 }
1208
1209 read += linelen;
1210
1211 size_t off = linelen - 1;
1212 if (line[off] == '\n')
1213 {
1214 /* We did get a full line: remove trailing space */
1215 do
1216 {
1217 line[off] = '\0';
1218 } while (off && mutt_str_is_email_wsp(line[--off]));
1219
1220 /* check to see if the next line is a continuation line */
1221 int ch = fgetc(fp);
1222 if ((ch != ' ') && (ch != '\t'))
1223 {
1224 /* next line is a separate header field or EOH */
1225 ungetc(ch, fp);
1226 buf_addstr(buf, line);
1227 break;
1228 }
1229 read++;
1230
1231 /* eat tabs and spaces from the beginning of the continuation line */
1232 while (((ch = fgetc(fp)) == ' ') || (ch == '\t'))
1233 {
1234 read++;
1235 }
1236
1237 ungetc(ch, fp);
1238 line[off + 1] = ' '; /* string is still terminated because we removed
1239 at least one whitespace char above */
1240 }
1241
1242 buf_addstr(buf, line);
1243 }
1244
1245 return read;
1246}
1247
1261struct Envelope *mutt_rfc822_read_header(FILE *fp, struct Email *e, bool user_hdrs, bool weed)
1262{
1263 if (!fp)
1264 return NULL;
1265
1266 struct Envelope *env = mutt_env_new();
1267 char *p = NULL;
1268 LOFF_T loc = e ? e->offset : ftello(fp);
1269 if (loc < 0)
1270 {
1271 mutt_debug(LL_DEBUG1, "ftello: %s (errno %d)\n", strerror(errno), errno);
1272 loc = 0;
1273 }
1274
1275 struct Buffer *line = buf_pool_get();
1276
1277 if (e)
1278 {
1279 if (!e->body)
1280 {
1281 e->body = mutt_body_new();
1282
1283 /* set the defaults from RFC1521 */
1284 e->body->type = TYPE_TEXT;
1285 e->body->subtype = mutt_str_dup("plain");
1286 e->body->encoding = ENC_7BIT;
1287 e->body->length = -1;
1288
1289 /* RFC2183 says this is arbitrary */
1291 }
1292 }
1293
1295 ASSERT(mod_data);
1296
1297 while (true)
1298 {
1299 LOFF_T line_start_loc = loc;
1300 size_t len = mutt_rfc822_read_line(fp, line);
1301 if (buf_is_empty(line))
1302 {
1303 break;
1304 }
1305 loc += len;
1306 const char *lines = buf_string(line);
1307 p = strpbrk(lines, ": \t");
1308 if (!p || (*p != ':'))
1309 {
1310 time_t t = 0;
1311
1312 /* some bogus MTAs will quote the original "From " line */
1313 if (mutt_str_startswith(lines, ">From "))
1314 {
1315 continue; /* just ignore */
1316 }
1317 else if (is_from(lines, NULL, 0, &t))
1318 {
1319 /* MH sometimes has the From_ line in the middle of the header! */
1320 if (e && (e->received == 0))
1321 e->received = t - mutt_date_local_tz(t);
1322 continue;
1323 }
1324
1325 /* We need to seek back to the start of the body. Note that we
1326 * keep track of loc ourselves, since calling ftello() incurs
1327 * a syscall, which can be expensive to do for every single line */
1328 (void) mutt_file_seek(fp, line_start_loc, SEEK_SET);
1329 break; /* end of header */
1330 }
1331 size_t name_len = p - lines;
1332
1333 char buf[1024] = { 0 };
1334 if (mutt_replacelist_match(&mod_data->spam, buf, sizeof(buf), lines))
1335 {
1336 if (!mutt_regexlist_match(&mod_data->no_spam, lines))
1337 {
1338 /* if spam tag already exists, figure out how to amend it */
1339 if ((!buf_is_empty(&env->spam)) && (*buf != '\0'))
1340 {
1341 /* If `$spam_separator` defined, append with separator */
1342 const char *const c_spam_separator = cs_subset_string(NeoMutt->sub, "spam_separator");
1343 if (c_spam_separator)
1344 {
1345 buf_addstr(&env->spam, c_spam_separator);
1346 buf_addstr(&env->spam, buf);
1347 }
1348 else /* overwrite */
1349 {
1350 buf_reset(&env->spam);
1351 buf_addstr(&env->spam, buf);
1352 }
1353 }
1354 else if (buf_is_empty(&env->spam) && (*buf != '\0'))
1355 {
1356 /* spam tag is new, and match expr is non-empty; copy */
1357 buf_addstr(&env->spam, buf);
1358 }
1359 else if (buf_is_empty(&env->spam))
1360 {
1361 /* match expr is empty; plug in null string if no existing tag */
1362 buf_addstr(&env->spam, "");
1363 }
1364
1365 if (!buf_is_empty(&env->spam))
1366 mutt_debug(LL_DEBUG5, "spam = %s\n", env->spam.data);
1367 }
1368 }
1369
1370 *p = '\0';
1371 p = mutt_str_skip_email_wsp(p + 1);
1372 if (*p == '\0')
1373 continue; /* skip empty header fields */
1374
1375 mutt_rfc822_parse_line(env, e, lines, name_len, p, user_hdrs, weed, true);
1376 }
1377
1378 buf_pool_release(&line);
1379
1380 if (e)
1381 {
1382 e->body->hdr_offset = e->offset;
1383 e->body->offset = ftello(fp);
1384
1386
1387 if (e->received < 0)
1388 {
1389 mutt_debug(LL_DEBUG1, "resetting invalid received time to 0\n");
1390 e->received = 0;
1391 }
1392
1393 /* check for missing or invalid date */
1394 if (e->date_sent <= 0)
1395 {
1396 mutt_debug(LL_DEBUG1, "no date found, using received time from msg separator\n");
1397 e->date_sent = e->received;
1398 }
1399
1400#ifdef USE_AUTOCRYPT
1401 const bool c_autocrypt = cs_subset_bool(NeoMutt->sub, "autocrypt");
1402 if (c_autocrypt)
1403 {
1405 /* No sense in taking up memory after the header is processed */
1407 }
1408#endif
1409 }
1410
1411 return env;
1412}
1413
1420struct Body *mutt_read_mime_header(FILE *fp, bool digest)
1421{
1422 if (!fp)
1423 return NULL;
1424
1425 struct Body *b = mutt_body_new();
1426 struct Envelope *env = mutt_env_new();
1427 char *c = NULL;
1428 struct Buffer *buf = buf_pool_get();
1429 bool matched = false;
1430
1431 b->hdr_offset = ftello(fp);
1432
1433 b->encoding = ENC_7BIT; /* default from RFC1521 */
1434 b->type = digest ? TYPE_MESSAGE : TYPE_TEXT;
1436
1437 while (mutt_rfc822_read_line(fp, buf) != 0)
1438 {
1439 const char *line = buf_string(buf);
1440 /* Find the value of the current header */
1441 c = strchr(line, ':');
1442 if (c)
1443 {
1444 *c = '\0';
1445 c = mutt_str_skip_email_wsp(c + 1);
1446 if (*c == '\0')
1447 {
1448 mutt_debug(LL_DEBUG1, "skipping empty header field: %s\n", line);
1449 continue;
1450 }
1451 }
1452 else
1453 {
1454 mutt_debug(LL_DEBUG1, "bogus MIME header: %s\n", line);
1455 break;
1456 }
1457
1458 size_t plen = mutt_istr_startswith(line, "content-");
1459 if (plen != 0)
1460 {
1461 if (mutt_istr_equal("type", line + plen))
1462 {
1464 }
1465 else if (mutt_istr_equal("language", line + plen))
1466 {
1468 }
1469 else if (mutt_istr_equal("transfer-encoding", line + plen))
1470 {
1472 }
1473 else if (mutt_istr_equal("disposition", line + plen))
1474 {
1476 }
1477 else if (mutt_istr_equal("description", line + plen))
1478 {
1481 }
1482 else if (mutt_istr_equal("id", line + plen))
1483 {
1484 // strip <angle braces> from Content-ID: header
1485 char *id = c;
1486 int cid_len = mutt_str_len(c);
1487 if (cid_len > 2)
1488 {
1489 if (id[0] == '<')
1490 {
1491 id++;
1492 cid_len--;
1493 }
1494 if (id[cid_len - 1] == '>')
1495 id[cid_len - 1] = '\0';
1496 }
1498 }
1499 }
1500 else if ((plen = mutt_istr_startswith(line, "x-sun-")))
1501 {
1502 if (mutt_istr_equal("data-type", line + plen))
1503 {
1505 }
1506 else if (mutt_istr_equal("encoding-info", line + plen))
1507 {
1509 }
1510 else if (mutt_istr_equal("content-lines", line + plen))
1511 {
1512 mutt_param_set(&b->parameter, "content-lines", c);
1513 }
1514 else if (mutt_istr_equal("data-description", line + plen))
1515 {
1518 }
1519 }
1520 else
1521 {
1522 if (mutt_rfc822_parse_line(env, NULL, line, strlen(line), c, false, false, false))
1523 {
1524 matched = true;
1525 }
1526 }
1527 }
1528 b->offset = ftello(fp); /* Mark the start of the real data */
1529 if ((b->type == TYPE_TEXT) && !b->subtype)
1530 b->subtype = mutt_str_dup("plain");
1531 else if ((b->type == TYPE_MESSAGE) && !b->subtype)
1532 b->subtype = mutt_str_dup("rfc822");
1533
1534 buf_pool_release(&buf);
1535
1536 if (matched)
1537 {
1538 b->mime_headers = env;
1540 }
1541 else
1542 {
1543 mutt_env_free(&env);
1544 }
1545
1546 return b;
1547}
1548
1556bool mutt_is_message_type(int type, const char *subtype)
1557{
1558 if (type != TYPE_MESSAGE)
1559 return false;
1560
1561 subtype = NONULL(subtype);
1562 return (mutt_istr_equal(subtype, "rfc822") ||
1563 mutt_istr_equal(subtype, "news") || mutt_istr_equal(subtype, "global"));
1564}
1565
1572static void parse_part(FILE *fp, struct Body *b, int *counter)
1573{
1574 if (!fp || !b)
1575 return;
1576
1577 const char *bound = NULL;
1578 static unsigned short recurse_level = 0;
1579
1580 if (recurse_level >= MUTT_MIME_MAX_DEPTH)
1581 {
1582 mutt_debug(LL_DEBUG1, "recurse level too deep. giving up\n");
1583 return;
1584 }
1585 recurse_level++;
1586
1587 switch (b->type)
1588 {
1589 case TYPE_MULTIPART:
1590 if (mutt_istr_equal(b->subtype, "x-sun-attachment"))
1591 bound = "--------";
1592 else
1593 bound = mutt_param_get(&b->parameter, "boundary");
1594
1595 if (!mutt_file_seek(fp, b->offset, SEEK_SET))
1596 {
1597 goto bail;
1598 }
1599 b->parts = parse_multipart(fp, bound, b->offset + b->length,
1600 mutt_istr_equal("digest", b->subtype), counter);
1601 break;
1602
1603 case TYPE_MESSAGE:
1604 if (!b->subtype)
1605 break;
1606
1607 if (!mutt_file_seek(fp, b->offset, SEEK_SET))
1608 {
1609 goto bail;
1610 }
1611 if (mutt_is_message_type(b->type, b->subtype))
1612 b->parts = rfc822_parse_message(fp, b, counter);
1613 else if (mutt_istr_equal(b->subtype, "external-body"))
1614 b->parts = mutt_read_mime_header(fp, 0);
1615 else
1616 goto bail;
1617 break;
1618
1619 default:
1620 goto bail;
1621 }
1622
1623 /* try to recover from parsing error */
1624 if (!b->parts)
1625 {
1626 b->type = TYPE_TEXT;
1627 mutt_str_replace(&b->subtype, "plain");
1628 }
1629bail:
1630 recurse_level--;
1631}
1632
1643static struct Body *parse_multipart(FILE *fp, const char *boundary,
1644 LOFF_T end_off, bool digest, int *counter)
1645{
1646 if (!fp)
1647 return NULL;
1648
1649 if (!boundary)
1650 {
1651 mutt_error(_("multipart message has no boundary parameter"));
1652 return NULL;
1653 }
1654
1655 char buf[1024] = { 0 };
1656 struct Body *head = NULL, *last = NULL, *new_body = NULL;
1657 bool final = false; /* did we see the ending boundary? */
1658
1659 const size_t blen = mutt_str_len(boundary);
1660 while ((ftello(fp) < end_off) && fgets(buf, sizeof(buf), fp))
1661 {
1662 const size_t len = mutt_str_len(buf);
1663
1664 const size_t crlf = ((len > 1) && (buf[len - 2] == '\r')) ? 1 : 0;
1665
1666 if ((buf[0] == '-') && (buf[1] == '-') && mutt_str_startswith(buf + 2, boundary))
1667 {
1668 if (last)
1669 {
1670 last->length = ftello(fp) - last->offset - len - 1 - crlf;
1671 if (last->parts && (last->parts->length == 0))
1672 last->parts->length = ftello(fp) - last->parts->offset - len - 1 - crlf;
1673 /* if the body is empty, we can end up with a -1 length */
1674 if (last->length < 0)
1675 last->length = 0;
1676 }
1677
1678 if (len > 0)
1679 {
1680 /* Remove any trailing whitespace, up to the length of the boundary */
1681 for (size_t i = len - 1; mutt_isspace(buf[i]) && (i >= (blen + 2)); i--)
1682 buf[i] = '\0';
1683 }
1684
1685 /* Check for the end boundary */
1686 if (mutt_str_equal(buf + blen + 2, "--"))
1687 {
1688 final = true;
1689 break; /* done parsing */
1690 }
1691 else if (buf[2 + blen] == '\0')
1692 {
1693 new_body = mutt_read_mime_header(fp, digest);
1694 if (!new_body)
1695 break;
1696
1697 if (mutt_param_get(&new_body->parameter, "content-lines"))
1698 {
1699 int lines = 0;
1700 mutt_str_atoi(mutt_param_get(&new_body->parameter, "content-lines"), &lines);
1701 for (; lines > 0; lines--)
1702 if ((ftello(fp) >= end_off) || !fgets(buf, sizeof(buf), fp))
1703 break;
1704 }
1705
1706 /* Consistency checking - catch bad attachment end boundaries */
1707 if (new_body->offset > end_off)
1708 {
1709 mutt_body_free(&new_body);
1710 break;
1711 }
1712 if (head)
1713 {
1714 last->next = new_body;
1715 last = new_body;
1716 }
1717 else
1718 {
1719 last = new_body;
1720 head = new_body;
1721 }
1722
1723 /* It seems more intuitive to add the counter increment to
1724 * parse_part(), but we want to stop the case where a multipart
1725 * contains thousands of tiny parts before the memory and data
1726 * structures are allocated. */
1727 if (++(*counter) >= MUTT_MIME_MAX_PARTS)
1728 break;
1729 }
1730 }
1731 }
1732
1733 /* in case of missing end boundary, set the length to something reasonable */
1734 if (last && (last->length == 0) && !final)
1735 last->length = end_off - last->offset;
1736
1737 /* parse recursive MIME parts */
1738 for (last = head; last; last = last->next)
1739 parse_part(fp, last, counter);
1740
1741 return head;
1742}
1743
1753static struct Body *rfc822_parse_message(FILE *fp, struct Body *parent, int *counter)
1754{
1755 if (!fp || !parent)
1756 return NULL;
1757
1758 parent->email = email_new();
1759 parent->email->offset = ftello(fp);
1760 parent->email->env = mutt_rfc822_read_header(fp, parent->email, false, false);
1761 struct Body *msg = parent->email->body;
1762
1763 /* ignore the length given in the content-length since it could be wrong
1764 * and we already have the info to calculate the correct length */
1765 /* if (msg->length == -1) */
1766 msg->length = parent->length - (msg->offset - parent->offset);
1767
1768 /* if body of this message is empty, we can end up with a negative length */
1769 if (msg->length < 0)
1770 msg->length = 0;
1771
1772 parse_part(fp, msg, counter);
1773 return msg;
1774}
1775
1790static bool mailto_header_allowed(const char *s, struct ListHead *h)
1791{
1792 if (!h)
1793 return false;
1794
1795 struct ListNode *np = NULL;
1796 STAILQ_FOREACH(np, h, entries)
1797 {
1798 if ((*np->data == '*') || mutt_istr_equal(s, np->data))
1799 return true;
1800 }
1801 return false;
1802}
1803
1812bool mutt_parse_mailto(struct Envelope *env, char **body, const char *src)
1813{
1814 if (!env || !src)
1815 return false;
1816
1817 struct Url *url = url_parse(src);
1818 if (!url)
1819 return false;
1820
1821 if (url->host)
1822 {
1823 /* this is not a path-only URL */
1824 url_free(&url);
1825 return false;
1826 }
1827
1828 mutt_addrlist_parse(&env->to, url->path);
1829
1831 ASSERT(mod_data);
1832
1833 struct UrlQuery *np;
1834 STAILQ_FOREACH(np, &url->query_strings, entries)
1835 {
1837 const char *tag = np->name;
1838 char *value = np->value;
1839 /* Determine if this header field is on the allowed list. Since NeoMutt
1840 * interprets some header fields specially (such as
1841 * "Attach: ~/.gnupg/secring.gpg"), care must be taken to ensure that
1842 * only safe fields are allowed.
1843 *
1844 * RFC2368, "4. Unsafe headers"
1845 * The user agent interpreting a mailto URL SHOULD choose not to create
1846 * a message if any of the headers are considered dangerous; it may also
1847 * choose to create a message with only a subset of the headers given in
1848 * the URL. */
1849 if (mailto_header_allowed(tag, &mod_data->mail_to_allow))
1850 {
1851 if (mutt_istr_equal(tag, "body"))
1852 {
1853 if (body)
1854 mutt_str_replace(body, value);
1855 }
1856 else
1857 {
1858 char *scratch = NULL;
1859 size_t taglen = mutt_str_len(tag);
1860
1862 mutt_str_asprintf(&scratch, "%s: %s", tag, value);
1863 scratch[taglen] = 0; /* overwrite the colon as mutt_rfc822_parse_line expects */
1864 value = mutt_str_skip_email_wsp(&scratch[taglen + 1]);
1865 mutt_rfc822_parse_line(env, NULL, scratch, taglen, value, true, false, true);
1866 FREE(&scratch);
1867 }
1868 }
1869 }
1870
1871 /* RFC2047 decode after the RFC822 parsing */
1873
1874 url_free(&url);
1875 return true;
1876}
1877
1883void mutt_parse_part(FILE *fp, struct Body *b)
1884{
1885 int counter = 0;
1886
1887 parse_part(fp, b, &counter);
1888}
1889
1898struct Body *mutt_rfc822_parse_message(FILE *fp, struct Body *b)
1899{
1900 int counter = 0;
1901
1902 return rfc822_parse_message(fp, b, &counter);
1903}
1904
1914struct Body *mutt_parse_multipart(FILE *fp, const char *boundary, LOFF_T end_off, bool digest)
1915{
1916 int counter = 0;
1917
1918 return parse_multipart(fp, boundary, end_off, digest, &counter);
1919}
void mutt_addrlist_clear(struct AddressList *al)
Unlink and free all Address in an AddressList.
Definition address.c:1469
int mutt_addrlist_parse(struct AddressList *al, const char *s)
Parse a list of email addresses.
Definition address.c:480
Email Address Handling.
void mutt_auto_subscribe(const char *mailto)
Check if user is subscribed to mailing list.
Definition commands.c:49
Email Aliases.
const char * mutt_str_atoul(const char *str, unsigned long *dst)
Convert ASCII string to an unsigned long.
Definition atoi.c:243
const char * mutt_str_atoui(const char *str, unsigned int *dst)
Convert ASCII string to an unsigned integer.
Definition atoi.c:217
const char * mutt_str_atoi(const char *str, int *dst)
Convert ASCII string to an integer.
Definition atoi.c:191
Autocrypt end-to-end encryption.
int mutt_autocrypt_process_autocrypt_header(struct Email *e, struct Envelope *env)
Parse an Autocrypt email header.
Definition autocrypt.c:263
void buf_reset(struct Buffer *buf)
Reset an existing Buffer.
Definition buffer.c:76
bool buf_is_empty(const struct Buffer *buf)
Is the Buffer empty?
Definition buffer.c:291
size_t buf_addch(struct Buffer *buf, char c)
Add a single character to a Buffer.
Definition buffer.c:241
size_t buf_addstr(struct Buffer *buf, const char *s)
Add a string to a Buffer.
Definition buffer.c:226
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
Definition buffer.c:571
void buf_alloc(struct Buffer *buf, size_t new_size)
Make sure a buffer can store at least new_size bytes.
Definition buffer.c:337
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
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.
const struct Slist * cc_assumed_charset(void)
Get the cached value of $assumed_charset.
Convenience wrapper for the core headers.
bool mutt_isspace(int arg)
Wrapper for isspace(3)
Definition ctype.c:96
void mutt_body_free(struct Body **ptr)
Free a Body.
Definition body.c:58
struct Body * mutt_body_new(void)
Create a new Body.
Definition body.c:44
Representation of the body of an email.
struct Email * email_new(void)
Create a new Email.
Definition email.c:77
Email private Module data.
int mutt_rfc822_parse_line(struct Envelope *env, struct Email *e, const char *name, size_t name_len, const char *body, bool user_hdrs, bool weed, bool do_2047)
Parse an email header.
Definition parse.c:734
struct Body * mutt_rfc822_parse_message(FILE *fp, struct Body *b)
Parse a Message/RFC822 body.
Definition parse.c:1898
void mutt_parse_part(FILE *fp, struct Body *b)
Parse a MIME part.
Definition parse.c:1883
void mutt_parse_content_type(const char *s, struct Body *b)
Parse a content type.
Definition parse.c:433
size_t mutt_rfc822_read_line(FILE *fp, struct Buffer *buf)
Read a header line from a file.
Definition parse.c:1181
char * mutt_rfc2369_first_mailto(const char *body)
Extract the first mailto: URL from an RFC 2369 list.
Definition parse.c:606
struct Body * mutt_read_mime_header(FILE *fp, bool digest)
Parse a MIME header.
Definition parse.c:1420
static struct AutocryptHeader * parse_autocrypt(struct AutocryptHeader *head, const char *s)
Parse an Autocrypt header line.
Definition parse.c:542
static void parse_references(struct ListHead *head, const char *s)
Parse references from an email header.
Definition parse.c:287
bool mutt_matches_ignore(const char *s)
Does the string match the ignore list.
Definition parse.c:320
enum ContentType mutt_check_mime_type(const char *s)
Check a MIME type string.
Definition parse.c:333
static struct Body * rfc822_parse_message(FILE *fp, struct Body *parent, int *counter)
Parse a Message/RFC822 body.
Definition parse.c:1753
struct Envelope * mutt_rfc822_read_header(FILE *fp, struct Email *e, bool user_hdrs, bool weed)
Parses an RFC822 header.
Definition parse.c:1261
static void parse_content_language(const char *s, struct Body *b)
Read the content's language.
Definition parse.c:304
static bool mailto_header_allowed(const char *s, struct ListHead *h)
Is the string in the list.
Definition parse.c:1790
bool mutt_parse_mailto(struct Envelope *env, char **body, const char *src)
Parse a mailto:// url.
Definition parse.c:1812
int mutt_check_encoding(const char *c)
Check the encoding type.
Definition parse.c:404
static void parse_content_disposition(const char *s, struct Body *b)
Parse a content disposition.
Definition parse.c:255
struct Body * mutt_parse_multipart(FILE *fp, const char *boundary, LOFF_T end_off, bool digest)
Parse a multipart structure.
Definition parse.c:1914
void mutt_filter_commandline_header_tag(char *header)
Sanitise characters in a header tag.
Definition parse.c:73
char * mutt_extract_message_id(const char *s, size_t *len)
Find a Message-ID.
Definition parse.c:368
static void parse_part(FILE *fp, struct Body *b, int *counter)
Parse a MIME part.
Definition parse.c:1572
static struct Body * parse_multipart(FILE *fp, const char *boundary, LOFF_T end_off, bool digest, int *counter)
Parse a multipart structure.
Definition parse.c:1643
bool mutt_is_message_type(int type, const char *subtype)
Determine if a mime type matches a message or not.
Definition parse.c:1556
void mutt_rfc2369_list_headers_free(struct Rfc2369ListHeaders *headers)
Free RFC 2369 mailing-list header values.
Definition parse.c:639
void mutt_filter_commandline_header_value(char *header)
Sanitise characters in a header value.
Definition parse.c:93
static void parse_parameters(struct ParameterList *pl, const char *s, bool allow_value_spaces)
Parse a list of Parameters.
Definition parse.c:116
void mutt_rfc2369_read_headers(FILE *fp, struct Rfc2369ListHeaders *headers)
Read RFC 2369 mailing-list headers.
Definition parse.c:657
#define CONTENT_TOO_BIG
Maximum reasonable Content-Length value (1 GiB) to prevent overflow.
Definition parse.c:62
Miscellaneous email parsing routines.
Representation of an email.
void mutt_env_free(struct Envelope **ptr)
Free an Envelope.
Definition envelope.c:125
struct Envelope * mutt_env_new(void)
Create a new Envelope.
Definition envelope.c:45
void mutt_env_set_subject(struct Envelope *env, const char *subj)
Set both subject and real_subj to subj.
Definition envelope.c:68
struct AutocryptHeader * mutt_autocrypthdr_new(void)
Create a new AutocryptHeader.
Definition envelope.c:94
void mutt_autocrypthdr_free(struct AutocryptHeader **ptr)
Free an AutocryptHeader.
Definition envelope.c:103
Representation of an email header (envelope)
static bool eqi17(const char *a, const char b[17])
eqi17 - Compare two 17-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:207
static bool eqi9(const char *a, const char b[9])
eqi9 - Compare two 9-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:159
static bool eqi10(const char *a, const char b[10])
eqi10 - Compare two 10-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:165
static bool eqi8(const char *a, const char b[8])
Compare two 8-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons.
Definition eqi.h:124
static bool eqi11(const char *a, const char b[11])
eqi11 - Compare two 11-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:171
static bool eqi6(const char *a, const char b[6])
eqi6 - Compare two 6-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:151
static bool eqi14(const char *a, const char b[14])
eqi14 - Compare two 14-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:189
static bool eqi13(const char *a, const char b[13])
eqi13 - Compare two 13-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:183
static bool eqi4(const char *a, const char b[4])
Compare two 4-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons.
Definition eqi.h:106
static bool eqi5(const char *a, const char b[5])
eqi5 - Compare two 5-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:145
static bool eqi12(const char *a, const char b[12])
eqi12 - Compare two 12-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:177
static bool eqi16(const char *a, const char b[16])
eqi16 - Compare two 16-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:201
static bool eqi15(const char *a, const char b[15])
eqi15 - Compare two 15-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons
Definition eqi.h:195
static bool eqi1(const char *a, const char b[1])
Compare two 1-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons.
Definition eqi.h:78
static bool eqi2(const char *a, const char b[2])
Compare two 2-byte strings, ignoring case - See: Case-insensitive fixed-chunk comparisons.
Definition eqi.h:90
bool mutt_file_seek(FILE *fp, LOFF_T offset, int whence)
Wrapper for fseeko with error handling.
Definition file.c:648
bool is_from(const char *s, char *path, size_t pathlen, time_t *tp)
Is a string a 'From' header line?
Definition from.c:49
Determine who the email is from.
#define mutt_error(...)
Definition logging2.h:94
#define mutt_debug(LEVEL,...)
Definition logging2.h:91
struct ListNode * mutt_list_insert_tail(struct ListHead *h, char *s)
Append a string to the end of a List.
Definition list.c:65
struct ListNode * mutt_list_insert_head(struct ListHead *h, char *s)
Insert a string at the beginning of a List.
Definition list.c:46
void mutt_list_free(struct ListHead *h)
Free a List AND its strings.
Definition list.c:123
bool mutt_list_match(const char *s, struct ListHead *h)
Is the string in the list (see notes)
Definition list.c:194
@ LL_DEBUG5
Log at debug level 5.
Definition logging2.h:49
@ LL_DEBUG2
Log at debug level 2.
Definition logging2.h:46
@ LL_DEBUG1
Log at debug level 1.
Definition logging2.h:45
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
#define MIN(a, b)
Return the minimum of two values.
Definition memory.h:40
Constants and macros for managing MIME encoding.
@ ENC_7BIT
7-bit text
Definition mime.h:49
@ ENC_UUENCODED
UUEncoded text.
Definition mime.h:54
@ ENC_OTHER
Encoding unknown.
Definition mime.h:48
@ ENC_BINARY
Binary.
Definition mime.h:53
@ ENC_BASE64
Base-64 encoded text.
Definition mime.h:52
@ ENC_8BIT
8-bit text
Definition mime.h:50
@ ENC_QUOTED_PRINTABLE
Quoted-printable text.
Definition mime.h:51
#define MUTT_MIME_MAX_DEPTH
Maximum nesting depth for MIME parts to prevent stack overflow.
Definition mime.h:69
ContentType
Content-Type.
Definition mime.h:30
@ TYPE_AUDIO
Type: 'audio/*'.
Definition mime.h:32
@ TYPE_IMAGE
Type: 'image/*'.
Definition mime.h:34
@ TYPE_OTHER
Unknown Content-Type.
Definition mime.h:31
@ TYPE_MESSAGE
Type: 'message/*'.
Definition mime.h:35
@ TYPE_MODEL
Type: 'model/*'.
Definition mime.h:36
@ TYPE_MULTIPART
Type: 'multipart/*'.
Definition mime.h:37
@ TYPE_APPLICATION
Type: 'application/*'.
Definition mime.h:33
@ TYPE_TEXT
Type: 'text/*'.
Definition mime.h:38
@ TYPE_ANY
Type: '' or '.'.
Definition mime.h:40
@ TYPE_VIDEO
Type: 'video/*'.
Definition mime.h:39
@ DISP_ATTACH
Content is attached.
Definition mime.h:63
@ DISP_INLINE
Content is inline.
Definition mime.h:62
@ DISP_FORM_DATA
Content is form-data.
Definition mime.h:64
#define MUTT_MIME_MAX_PARTS
Maximum number of MIME parts to process to prevent DoS.
Definition mime.h:71
@ MODULE_ID_EMAIL
ModuleEmail, Email code
Definition module_api.h:64
const char * mutt_ch_get_default_charset(const struct Slist *const assumed_charset)
Get the default character set.
Definition charset.c:451
int mutt_date_local_tz(time_t t)
Calculate the local timezone in seconds east of UTC.
Definition date.c:220
time_t mutt_date_now(void)
Return the number of seconds since the Unix epoch.
Definition date.c:457
time_t mutt_date_parse_date(const char *s, struct Tz *tz_out)
Parse a date string in RFC822 format.
Definition date.c:717
Convenience wrapper for the library headers.
#define _(a)
Definition message.h:28
bool mutt_replacelist_match(struct ReplaceList *rl, char *buf, size_t buflen, const char *str)
Does a string match a pattern?
Definition regex.c:478
bool mutt_regexlist_match(struct RegexList *rl, const char *str)
Does a string match any Regex in the list?
Definition regex.c:200
bool slist_is_empty(const struct Slist *list)
Is the slist empty?
Definition slist.c:140
char * mutt_strn_dup(const char *begin, size_t len)
Duplicate a sub-string.
Definition string.c:384
void mutt_str_remove_trailing_ws(char *s)
Trim trailing whitespace from a string.
Definition string.c:570
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition string.c:677
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
int mutt_str_asprintf(char **strp, const char *fmt,...)
Definition string.c:808
char * mutt_str_skip_email_wsp(const char *s)
Skip over whitespace as defined by RFC5322.
Definition string.c:613
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
Definition string.c:665
bool mutt_strn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings (to a maximum), safely.
Definition string.c:429
size_t mutt_str_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix.
Definition string.c:234
char * mutt_str_skip_whitespace(const char *p)
Find the first non-whitespace character in a string.
Definition string.c:556
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
Definition string.c:503
size_t mutt_istr_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix, ignoring case.
Definition string.c:246
bool mutt_istrn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings ignoring case (to a maximum), safely.
Definition string.c:457
char * mutt_str_replace(char **p, const char *s)
Replace one string with another.
Definition string.c:284
Many unsorted constants and some structs.
void * neomutt_get_module_data(struct NeoMutt *n, enum ModuleId id)
Get the private data for a Module.
Definition neomutt.c:663
char * mutt_param_get(const struct ParameterList *pl, const char *s)
Find a matching Parameter.
Definition parameter.c:85
void mutt_param_set(struct ParameterList *pl, const char *attribute, const char *value)
Set a Parameter.
Definition parameter.c:111
void mutt_param_free(struct ParameterList *pl)
Free a ParameterList.
Definition parameter.c:62
struct Parameter * mutt_param_new(void)
Create a new Parameter.
Definition parameter.c:40
Store attributes associated with a MIME part.
struct Buffer * buf_pool_get(void)
Get a Buffer from the pool.
Definition pool.c:91
void buf_pool_release(struct Buffer **ptr)
Return a Buffer to the pool.
Definition pool.c:111
#define TAILQ_FOREACH(var, head, field)
Definition queue.h:782
#define STAILQ_FOREACH(var, head, field)
Definition queue.h:390
#define TAILQ_HEAD_INITIALIZER(head)
Definition queue.h:694
#define TAILQ_EMPTY(head)
Definition queue.h:778
#define TAILQ_INSERT_HEAD(head, elm, field)
Definition queue.h:853
void rfc2047_decode_envelope(struct Envelope *env)
Decode the fields of an Envelope.
Definition rfc2047.c:840
void rfc2047_decode(char **pd)
Decode any RFC2047-encoded header fields.
Definition rfc2047.c:669
RFC2047 MIME extensions encoding / decoding routines.
void rfc2231_decode_parameters(struct ParameterList *pl)
Decode a Parameter list.
Definition rfc2231.c:241
RFC2231 MIME Charset routines.
#define ASSERT(COND)
Definition signal2.h:59
#define NONULL(x)
Definition string2.h:44
static bool mutt_str_is_email_wsp(char c)
Is this a whitespace character (for an email header)
Definition string2.h:111
Parse Autocrypt header info.
Definition envelope.h:44
bool invalid
Header is invalid.
Definition envelope.h:48
struct AutocryptHeader * next
Linked list.
Definition envelope.h:49
char * keydata
PGP Key data.
Definition envelope.h:46
bool prefer_encrypt
User prefers encryption.
Definition envelope.h:47
char * addr
Email address.
Definition envelope.h:45
The body of an email.
Definition body.h:36
char * language
content-language (RFC8255)
Definition body.h:78
char * content_id
Content-Id (RFC2392)
Definition body.h:58
struct Body * parts
parts of a multipart or message/rfc822
Definition body.h:73
LOFF_T offset
offset where the actual data begins
Definition body.h:52
char * xtype
content-type if x-unknown
Definition body.h:62
struct Envelope * mime_headers
Memory hole protected headers.
Definition body.h:76
LOFF_T length
length (in bytes) of attachment
Definition body.h:53
struct ParameterList parameter
Parameters of the content-type.
Definition body.h:63
struct Email * email
header information for message/rfc822
Definition body.h:74
char * description
content-description
Definition body.h:55
unsigned int disposition
content-disposition, ContentDisposition
Definition body.h:42
struct Body * next
next attachment in the list
Definition body.h:72
char * subtype
content-type subtype
Definition body.h:61
unsigned int encoding
content-transfer-encoding, ContentEncoding
Definition body.h:41
long hdr_offset
Offset in stream where the headers begin.
Definition body.h:81
char * form_name
Content-Disposition form-data name param.
Definition body.h:60
unsigned int type
content-type primary type, ContentType
Definition body.h:40
char * filename
When sending a message, this is the file to which this structure refers.
Definition body.h:59
String manipulation buffer.
Definition buffer.h:36
char * data
Pointer to data.
Definition buffer.h:37
Email private Module data.
Definition module_data.h:32
struct ListHead unignore
Header patterns to unignore.
Definition module_data.h:43
struct ListHead ignore
Header patterns to ignore.
Definition module_data.h:37
struct ReplaceList spam
Regexes and patterns to match spam emails.
Definition module_data.h:40
struct ListHead mail_to_allow
Permitted fields in a mailto: url.
Definition module_data.h:38
struct RegexList no_spam
Regexes to identify non-spam emails.
Definition module_data.h:39
The envelope/body of an email.
Definition email.h:39
bool read
Email is read.
Definition email.h:50
unsigned int zminutes
Minutes away from UTC.
Definition email.h:57
struct Envelope * env
Envelope information.
Definition email.h:68
bool mime
Has a MIME-Version header?
Definition email.h:48
int lines
How many lines in the body of this message?
Definition email.h:62
struct Body * body
List of MIME parts.
Definition email.h:69
bool old
Email is seen, but unread.
Definition email.h:49
bool zoccident
True, if west of UTC, False if east.
Definition email.h:58
LOFF_T offset
Where in the stream does this message begin?
Definition email.h:71
bool flagged
Marked important?
Definition email.h:47
unsigned int zhours
Hours away from UTC.
Definition email.h:56
time_t date_sent
Time when the message was sent (UTC)
Definition email.h:60
bool replied
Email has been replied to.
Definition email.h:51
bool expired
Already expired?
Definition email.h:46
bool deleted
Email is deleted.
Definition email.h:78
time_t received
Time when the message was placed in the mailbox.
Definition email.h:61
The header of an Email.
Definition envelope.h:57
struct ListHead userhdrs
user defined headers
Definition envelope.h:85
char * supersedes
Supersedes header.
Definition envelope.h:74
char * list_subscribe
This stores a mailto URL, or nothing.
Definition envelope.h:68
struct AddressList return_path
Return path for the Email.
Definition envelope.h:58
char *const subject
Email's subject.
Definition envelope.h:70
struct AddressList to
Email's 'To' list.
Definition envelope.h:60
char * followup_to
List of 'followup-to' fields.
Definition envelope.h:80
struct AddressList reply_to
Email's 'reply-to'.
Definition envelope.h:64
char * message_id
Message ID.
Definition envelope.h:73
char * x_comment_to
List of 'X-comment-to' fields.
Definition envelope.h:81
struct AddressList x_original_to
Email's 'X-Original-to'.
Definition envelope.h:66
struct AutocryptHeader * autocrypt_gossip
Autocrypt Gossip header.
Definition envelope.h:88
char * newsgroups
List of newsgroups.
Definition envelope.h:78
struct AddressList mail_followup_to
Email's 'mail-followup-to'.
Definition envelope.h:65
struct AddressList cc
Email's 'Cc' list.
Definition envelope.h:61
struct AddressList sender
Email's sender.
Definition envelope.h:63
struct ListHead references
message references (in reverse order)
Definition envelope.h:83
struct AutocryptHeader * autocrypt
Autocrypt header.
Definition envelope.h:87
struct Buffer spam
Spam header.
Definition envelope.h:82
struct ListHead in_reply_to
in-reply-to header content
Definition envelope.h:84
struct AddressList bcc
Email's 'Bcc' list.
Definition envelope.h:62
char * xref
List of cross-references.
Definition envelope.h:79
char * organization
Organisation header.
Definition envelope.h:77
char * x_label
X-Label.
Definition envelope.h:76
char * list_post
This stores a mailto URL, or nothing.
Definition envelope.h:67
char * date
Sent date.
Definition envelope.h:75
char * list_unsubscribe
This stores a mailto URL, or nothing.
Definition envelope.h:69
struct AddressList from
Email's 'From' list.
Definition envelope.h:59
A List node for strings.
Definition list.h:37
char * data
String.
Definition list.h:38
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
Attribute associated with a MIME part.
Definition parameter.h:33
char * attribute
Parameter name.
Definition parameter.h:34
char * value
Parameter value.
Definition parameter.h:35
Mailing-list actions from RFC 2369 headers.
Definition parse.h:41
char * archive
List-Archive.
Definition parse.h:42
char * help
List-Help.
Definition parse.h:43
char * post
List-Post.
Definition parse.h:45
char * owner
List-Owner.
Definition parse.h:44
char * subscribe
List-Subscribe.
Definition parse.h:46
char * unsubscribe
List-Unsubscribe.
Definition parse.h:47
List of recognised Timezones.
Definition date.h:53
unsigned char zminutes
Minutes away from UTC.
Definition date.h:56
bool zoccident
True if west of UTC, False if East.
Definition date.h:57
unsigned char zhours
Hours away from UTC.
Definition date.h:55
Parsed Query String.
Definition url.h:58
char * name
Query name.
Definition url.h:59
char * value
Query value.
Definition url.h:60
A parsed URL proto://user:password@host:port/path?a=1&b=2
Definition url.h:69
struct UrlQueryList query_strings
List of query strings.
Definition url.h:76
char * host
Host.
Definition url.h:73
char * src
Raw URL string.
Definition url.h:77
char * path
Path.
Definition url.h:75
struct Url * url_parse(const char *src)
Fill in Url.
Definition url.c:242
void url_free(struct Url **ptr)
Free the contents of a URL.
Definition url.c:124
enum UrlScheme url_check_scheme(const char *str)
Check the protocol of a URL.
Definition url.c:229
Parse and identify different URL schemes.
@ U_MAILTO
Url is mailto://.
Definition url.h:45