NeoMutt  2025-12-11-911-gd8d604
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
parse.h File Reference

Miscellaneous email parsing routines. More...

#include "config.h"
#include <stdbool.h>
#include <stdio.h>
#include "mime.h"
+ Include dependency graph for parse.h:
+ This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Data Structures

struct  Rfc2369ListHeaders
 Mailing-list actions from RFC 2369 headers. More...
 

Functions

int mutt_check_encoding (const char *c)
 Check the encoding type.
 
enum ContentType mutt_check_mime_type (const char *s)
 Check a MIME type string.
 
char * mutt_extract_message_id (const char *s, size_t *len)
 Find a Message-ID.
 
bool mutt_is_message_type (int type, const char *subtype)
 Determine if a mime type matches a message or not.
 
bool mutt_matches_ignore (const char *s)
 Does the string match the ignore list.
 
void mutt_parse_content_type (const char *s, struct Body *b)
 Parse a content type.
 
bool mutt_parse_mailto (struct Envelope *env, char **body, const char *src)
 Parse a mailto:// url.
 
char * mutt_rfc2369_first_mailto (const char *body)
 Extract the first mailto: URL from an RFC 2369 list.
 
void mutt_rfc2369_list_headers_free (struct Rfc2369ListHeaders *headers)
 Free RFC 2369 mailing-list header values.
 
void mutt_rfc2369_read_headers (FILE *fp, struct Rfc2369ListHeaders *headers)
 Read RFC 2369 mailing-list headers.
 
struct Bodymutt_parse_multipart (FILE *fp, const char *boundary, LOFF_T end_off, bool digest)
 Parse a multipart structure.
 
void mutt_parse_part (FILE *fp, struct Body *b)
 Parse a MIME part.
 
struct Bodymutt_read_mime_header (FILE *fp, bool digest)
 Parse a MIME header.
 
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.
 
struct Bodymutt_rfc822_parse_message (FILE *fp, struct Body *b)
 Parse a Message/RFC822 body.
 
struct Envelopemutt_rfc822_read_header (FILE *fp, struct Email *e, bool user_hdrs, bool weed)
 Parses an RFC822 header.
 
size_t mutt_rfc822_read_line (FILE *fp, struct Buffer *out)
 Read a header line from a file.
 
void mutt_filter_commandline_header_tag (char *header)
 Sanitise characters in a header tag.
 
void mutt_filter_commandline_header_value (char *header)
 Sanitise characters in a header value.
 

Detailed Description

Miscellaneous email parsing routines.

Authors
  • Richard Russon
  • Pietro Cerutti

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 General Public License for more details.

You should have received a copy of the GNU General Public License along with this program. If not, see http://www.gnu.org/licenses/.

Definition in file parse.h.

Function Documentation

◆ mutt_check_encoding()

int mutt_check_encoding ( const char * c)

Check the encoding type.

Parameters
cString to check
Return values
enumContentEncoding, e.g. ENC_QUOTED_PRINTABLE

Definition at line 404 of file parse.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}
@ 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
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
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_check_mime_type()

enum ContentType mutt_check_mime_type ( const char * s)

Check a MIME type string.

Parameters
sString to check
Return values
enumContentType, e.g. TYPE_TEXT

Definition at line 333 of file parse.c.

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}
@ 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
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition string.c:677
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_extract_message_id()

char * mutt_extract_message_id ( const char * s,
size_t * len )

Find a Message-ID.

Parameters
[in]sString to parse
[out]lenNumber of bytes of s parsed
Return values
ptrMessage id found
NULLNo more message ids

Definition at line 368 of file parse.c.

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}
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
char * mutt_strn_dup(const char *begin, size_t len)
Duplicate a sub-string.
Definition string.c:384
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
void rfc2047_decode(char **pd)
Decode any RFC2047-encoded header fields.
Definition rfc2047.c:669
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_is_message_type()

bool mutt_is_message_type ( int type,
const char * subtype )

Determine if a mime type matches a message or not.

Parameters
typeMessage type enum value
subtypeMessage subtype
Return values
trueType is message/news or message/rfc822
falseOtherwise

Definition at line 1556 of file parse.c.

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}
#define NONULL(x)
Definition string2.h:44
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_matches_ignore()

bool mutt_matches_ignore ( const char * s)

Does the string match the ignore list.

Parameters
sString to check
Return values
trueString matches

Checks ignore and unignore using mutt_list_match

Definition at line 320 of file parse.c.

321{
323 ASSERT(mod_data);
324
325 return mutt_list_match(s, &mod_data->ignore) && !mutt_list_match(s, &mod_data->unignore);
326}
bool mutt_list_match(const char *s, struct ListHead *h)
Is the string in the list (see notes)
Definition list.c:194
@ MODULE_ID_EMAIL
ModuleEmail, Email code
Definition module_api.h:64
void * neomutt_get_module_data(struct NeoMutt *n, enum ModuleId id)
Get the private data for a Module.
Definition neomutt.c:663
#define ASSERT(COND)
Definition signal2.h:59
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
Container for Accounts, Notifications.
Definition neomutt.h:41
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_parse_content_type()

void mutt_parse_content_type ( const char * s,
struct Body * b )

Parse a content type.

Parameters
sString to parse
bBody to save the result

WARNING: this function modifies the 's' argument to segregate type, subtype, and parametners, so, e.g., a param of "text/plain" will be changed into "text\0plain".

Definition at line 433 of file parse.c.

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}
const struct Slist * cc_assumed_charset(void)
Get the cached value of $assumed_charset.
bool mutt_isspace(int arg)
Wrapper for isspace(3)
Definition ctype.c:96
enum ContentType mutt_check_mime_type(const char *s)
Check a MIME type string.
Definition parse.c:333
int mutt_check_encoding(const char *c)
Check the encoding type.
Definition parse.c:404
static void parse_parameters(struct ParameterList *pl, const char *s, bool allow_value_spaces)
Parse a list of Parameters.
Definition parse.c:116
const char * mutt_ch_get_default_charset(const struct Slist *const assumed_charset)
Get the default character set.
Definition charset.c:451
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
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
char * xtype
content-type if x-unknown
Definition body.h:62
struct ParameterList parameter
Parameters of the content-type.
Definition body.h:63
char * subtype
content-type subtype
Definition body.h:61
unsigned int encoding
content-transfer-encoding, ContentEncoding
Definition body.h:41
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
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_parse_mailto()

bool mutt_parse_mailto ( struct Envelope * env,
char ** body,
const char * src )

Parse a mailto:// url.

Parameters
[in]envEnvelope to fill
[out]bodyBody to
[in]srcString to parse
Return values
trueSuccess
falseError

Definition at line 1812 of file parse.c.

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}
int mutt_addrlist_parse(struct AddressList *al, const char *s)
Parse a list of email addresses.
Definition address.c:480
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
static bool mailto_header_allowed(const char *s, struct ListHead *h)
Is the string in the list.
Definition parse.c:1790
void mutt_filter_commandline_header_tag(char *header)
Sanitise characters in a header tag.
Definition parse.c:73
void mutt_filter_commandline_header_value(char *header)
Sanitise characters in a header value.
Definition parse.c:93
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
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
Definition string.c:503
#define STAILQ_FOREACH(var, head, field)
Definition queue.h:390
void rfc2047_decode_envelope(struct Envelope *env)
Decode the fields of an Envelope.
Definition rfc2047.c:840
struct ListHead mail_to_allow
Permitted fields in a mailto: url.
Definition module_data.h:38
struct AddressList to
Email's 'To' list.
Definition envelope.h:60
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
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc2369_first_mailto()

char * mutt_rfc2369_first_mailto ( const char * body)

Extract the first mailto: URL from an RFC 2369 list.

Parameters
bodyBody of the header
Return values
ptrFirst mailto: URL found, or NULL if none was found

Definition at line 606 of file parse.c.

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}
enum UrlScheme url_check_scheme(const char *str)
Check the protocol of a URL.
Definition url.c:229
@ U_MAILTO
Url is mailto://.
Definition url.h:45
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc2369_list_headers_free()

void mutt_rfc2369_list_headers_free ( struct Rfc2369ListHeaders * headers)

Free RFC 2369 mailing-list header values.

Parameters
headersHeaders to free

Definition at line 639 of file parse.c.

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}
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
+ Here is the caller graph for this function:

◆ mutt_rfc2369_read_headers()

void mutt_rfc2369_read_headers ( FILE * fp,
struct Rfc2369ListHeaders * headers )

Read RFC 2369 mailing-list headers.

Parameters
fpMessage stream positioned at the start of the headers
headersParsed header values

Definition at line 657 of file parse.c.

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}
bool buf_is_empty(const struct Buffer *buf)
Is the Buffer empty?
Definition buffer.c:291
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
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
void mutt_rfc2369_list_headers_free(struct Rfc2369ListHeaders *headers)
Free RFC 2369 mailing-list header values.
Definition parse.c:639
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 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 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
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
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
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
String manipulation buffer.
Definition buffer.h:36
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_parse_multipart()

struct Body * mutt_parse_multipart ( FILE * fp,
const char * boundary,
LOFF_T end_off,
bool digest )

Parse a multipart structure.

Parameters
fpStream to read from
boundaryBody separator
end_offLength of the multipart body (used when the final boundary is missing to avoid reading too far)
digesttrue if reading a multipart/digest
Return values
ptrNew Body containing parsed structure

Definition at line 1914 of file parse.c.

1915{
1916 int counter = 0;
1917
1918 return parse_multipart(fp, boundary, end_off, digest, &counter);
1919}
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
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_parse_part()

void mutt_parse_part ( FILE * fp,
struct Body * b )

Parse a MIME part.

Parameters
fpFile to read from
bBody to store the results in

Definition at line 1883 of file parse.c.

1884{
1885 int counter = 0;
1886
1887 parse_part(fp, b, &counter);
1888}
static void parse_part(FILE *fp, struct Body *b, int *counter)
Parse a MIME part.
Definition parse.c:1572
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_read_mime_header()

struct Body * mutt_read_mime_header ( FILE * fp,
bool digest )

Parse a MIME header.

Parameters
fpstream to read from
digesttrue if reading subparts of a multipart/digest
Return values
ptrNew Body containing parsed structure

Definition at line 1420 of file parse.c.

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}
struct Body * mutt_body_new(void)
Create a new Body.
Definition body.c:44
void mutt_parse_content_type(const char *s, struct Body *b)
Parse a content type.
Definition parse.c:433
static void parse_content_language(const char *s, struct Body *b)
Read the content's language.
Definition parse.c:304
static void parse_content_disposition(const char *s, struct Body *b)
Parse a content disposition.
Definition parse.c:255
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
#define mutt_debug(LEVEL,...)
Definition logging2.h:91
@ LL_DEBUG1
Log at debug level 1.
Definition logging2.h:45
@ DISP_INLINE
Content is inline.
Definition mime.h:62
The body of an email.
Definition body.h:36
char * content_id
Content-Id (RFC2392)
Definition body.h:58
LOFF_T offset
offset where the actual data begins
Definition body.h:52
struct Envelope * mime_headers
Memory hole protected headers.
Definition body.h:76
char * description
content-description
Definition body.h:55
unsigned int disposition
content-disposition, ContentDisposition
Definition body.h:42
long hdr_offset
Offset in stream where the headers begin.
Definition body.h:81
The header of an Email.
Definition envelope.h:57
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc822_parse_line()

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.

Parameters
envEnvelope of the email
eEmail
nameHeader field name, e.g. 'to'
name_lenMust be equivalent to strlen(name)
bodyHeader field body, e.g. 'john@.nosp@m.exam.nosp@m.ple.c.nosp@m.om'
user_hdrsIf true, save into the Envelope's userhdrs
weedIf true, perform header weeding (filtering)
do_2047If true, perform RFC2047 decoding of the field
Return values
1The field is recognised
0The field is not recognised

Process a line from an email header. Each line that is recognised is parsed and the information put in the Envelope or Header.

Definition at line 734 of file parse.c.

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}
void mutt_addrlist_clear(struct AddressList *al)
Unlink and free all Address in an AddressList.
Definition address.c:1469
void mutt_auto_subscribe(const char *mailto)
Check if user is subscribed to mailing list.
Definition commands.c:49
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
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
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
char * mutt_extract_message_id(const char *s, size_t *len)
Find a Message-ID.
Definition parse.c:368
#define CONTENT_TOO_BIG
Maximum reasonable Content-Length value (1 GiB) to prevent overflow.
Definition parse.c:62
void mutt_env_set_subject(struct Envelope *env, const char *subj)
Set both subject and real_subj to subj.
Definition envelope.c:68
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 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 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 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
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(struct ListHead *h)
Free a List AND its strings.
Definition list.c:123
#define MIN(a, b)
Return the minimum of two values.
Definition memory.h:40
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
void mutt_str_remove_trailing_ws(char *s)
Trim trailing whitespace from a string.
Definition string.c:570
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
LOFF_T length
length (in bytes) of attachment
Definition body.h:53
bool read
Email is read.
Definition email.h:50
unsigned int zminutes
Minutes away from UTC.
Definition email.h:57
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
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
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
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 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
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
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
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc822_parse_message()

struct Body * mutt_rfc822_parse_message ( FILE * fp,
struct Body * b )

Parse a Message/RFC822 body.

Parameters
fpStream to read from
bInfo about the message/rfc822 body part
Return values
ptrNew Body containing parsed message
Note
This assumes that 'b->length' has been set!

Definition at line 1898 of file parse.c.

1899{
1900 int counter = 0;
1901
1902 return rfc822_parse_message(fp, b, &counter);
1903}
static struct Body * rfc822_parse_message(FILE *fp, struct Body *parent, int *counter)
Parse a Message/RFC822 body.
Definition parse.c:1753
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc822_read_header()

struct Envelope * mutt_rfc822_read_header ( FILE * fp,
struct Email * e,
bool user_hdrs,
bool weed )

Parses an RFC822 header.

Parameters
fpStream to read from
eCurrent Email (optional)
user_hdrsIf set, store user headers Used for recall-message and postpone modes
weedIf this parameter is set and the user has activated the $weed option, honor the header weed list for user headers. Used for recall-message
Return values
ptrNewly allocated envelope structure

Caller should free the Envelope using mutt_env_free().

Definition at line 1261 of file parse.c.

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}
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
size_t buf_addstr(struct Buffer *buf, const char *s)
Add a string to a Buffer.
Definition buffer.c:226
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
void mutt_autocrypthdr_free(struct AutocryptHeader **ptr)
Free an AutocryptHeader.
Definition envelope.c:103
bool mutt_file_seek(FILE *fp, LOFF_T offset, int whence)
Wrapper for fseeko with error handling.
Definition file.c:648
@ LL_DEBUG5
Log at debug level 5.
Definition logging2.h:49
int mutt_date_local_tz(time_t t)
Calculate the local timezone in seconds east of UTC.
Definition date.c:220
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
char * data
Pointer to data.
Definition buffer.h:37
struct ReplaceList spam
Regexes and patterns to match spam emails.
Definition module_data.h:40
struct RegexList no_spam
Regexes to identify non-spam emails.
Definition module_data.h:39
LOFF_T offset
Where in the stream does this message begin?
Definition email.h:71
struct Buffer spam
Spam header.
Definition envelope.h:82
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_rfc822_read_line()

size_t mutt_rfc822_read_line ( FILE * fp,
struct Buffer * buf )

Read a header line from a file.

Parameters
fpFile to read from
bufBuffer to store the result
Return values
numNumber of bytes read from fp

Reads an arbitrarily long header field, and looks ahead for continuation lines.

Definition at line 1181 of file parse.c.

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}
static bool mutt_str_is_email_wsp(char c)
Is this a whitespace character (for an email header)
Definition string2.h:111
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_filter_commandline_header_tag()

void mutt_filter_commandline_header_tag ( char * header)

Sanitise characters in a header tag.

Parameters
headerString to sanitise

Definition at line 73 of file parse.c.

74{
75 if (!header)
76 return;
77
78 for (; (*header != '\0'); header++)
79 {
80 if ((*header < 33) || (*header > 126) || (*header == ':'))
81 *header = '?';
82 }
83}
+ Here is the caller graph for this function:

◆ mutt_filter_commandline_header_value()

void mutt_filter_commandline_header_value ( char * header)

Sanitise characters in a header value.

Parameters
headerString to sanitise

It might be preferable to use mutt_filter_unprintable() instead. This filter is being lax, but preventing a header injection via an embedded newline.

Definition at line 93 of file parse.c.

94{
95 if (!header)
96 return;
97
98 for (; (*header != '\0'); header++)
99 {
100 if ((*header == '\n') || (*header == '\r'))
101 *header = ' ';
102 }
103}
+ Here is the caller graph for this function: