/*- * Copyright 2016 Vsevolod Stakhov * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "config.h" #include "email_addr.h" #include "message.h" #include "printf.h" #include "smtp_parsers.h" static void rspamd_email_address_unescape (struct rspamd_email_address *addr) { const char *h, *end; char *t, *d; if (addr->user_len == 0) { return; } d = g_malloc (addr->user_len); t = d; h = addr->user; end = h + addr->user_len; while (h < end) { if (*h != '\\') { *t++ = *h; } h ++; } addr->user = d; addr->user_len = t - d; addr->flags |= RSPAMD_EMAIL_ADDR_USER_ALLOCATED; } struct rspamd_email_address * rspamd_email_address_from_smtp (const gchar *str, guint len) { struct rspamd_email_address addr, *ret; gsize nlen; if (str == NULL || len == 0) { return NULL; } rspamd_smtp_addr_parse (str, len, &addr); if (addr.flags & RSPAMD_EMAIL_ADDR_VALID) { ret = g_malloc (sizeof (*ret)); memcpy (ret, &addr, sizeof (addr)); if ((ret->flags & RSPAMD_EMAIL_ADDR_QUOTED) && ret->addr[0] == '"') { if (ret->flags & RSPAMD_EMAIL_ADDR_HAS_BACKSLASH) { /* We also need to unquote user */ rspamd_email_address_unescape (ret); } /* We need to unquote addr */ nlen = ret->domain_len + ret->user_len + 2; ret->addr = g_malloc (nlen + 1); ret->addr_len = rspamd_snprintf ((char *)ret->addr, nlen, "%*s@%*s", (gint)ret->user_len, ret->user, (gint)ret->domain_len, ret->domain); ret->flags |= RSPAMD_EMAIL_ADDR_ADDR_ALLOCATED; } return ret; } return NULL; } void rspamd_email_address_free (struct rspamd_email_address *addr) { if (addr->flags & RSPAMD_EMAIL_ADDR_ADDR_ALLOCATED) { g_free ((void *)addr->addr); } if (addr->flags & RSPAMD_EMAIL_ADDR_USER_ALLOCATED) { g_free ((void *)addr->user); } g_free (addr); } static inline void rspamd_email_address_add (rspamd_mempool_t *pool, GPtrArray *ar, struct rspamd_email_address *addr, GString *name) { struct rspamd_email_address *elt; guint nlen; elt = g_malloc0 (sizeof (*elt)); if (addr != NULL) { memcpy (elt, addr, sizeof (*addr)); } else { elt->addr = ""; elt->domain = ""; elt->raw = "<>"; elt->raw_len = 2; elt->user = ""; elt->flags |= RSPAMD_EMAIL_ADDR_EMPTY; } if ((elt->flags & RSPAMD_EMAIL_ADDR_QUOTED) && elt->addr[0] == '"') { if (elt->flags & RSPAMD_EMAIL_ADDR_HAS_BACKSLASH) { /* We also need to unquote user */ rspamd_email_address_unescape (elt); } /* We need to unquote addr */ nlen = elt->domain_len + elt->user_len + 2; elt->addr = g_malloc (nlen + 1); elt->addr_len = rspamd_snprintf ((char *)elt->addr, nlen, "%*s@%*s", (gint)elt->user_len, elt->user, (gint)elt->domain_len, elt->domain); elt->flags |= RSPAMD_EMAIL_ADDR_ADDR_ALLOCATED; } if (name->len > 0) { elt->name = rspamd_mime_header_decode (pool, name->str, name->len, NULL); } g_ptr_array_add (ar, elt); } /* * Tries to parse an email address that doesn't conform RFC */ static gboolean rspamd_email_address_parse_heuristic (const char *data, size_t len, struct rspamd_email_address *addr) { const gchar *p = data, *at = NULL, *end = data + len; gboolean ret = FALSE; memset (addr, 0, sizeof (*addr)); if (*p == '<' && len > 1) { /* Angled address */ addr->addr_len = rspamd_memcspn (p + 1, ">", len - 1); addr->addr = p + 1; addr->raw = p; addr->raw_len = len; ret = TRUE; p = p + 1; len = addr->addr_len; end = p + len; } else if (len > 0) { addr->addr = p; addr->addr_len = len; addr->raw = p; addr->raw_len = len; ret = TRUE; } if (ret) { at = memchr (p, '@', len); if (at != NULL && at + 1 < end) { addr->domain = at + 1; addr->domain_len = end - (at + 1); addr->user = p; addr->user_len = at - p; } if (rspamd_str_has_8bit (p, len)) { addr->flags |= RSPAMD_EMAIL_ADDR_HAS_8BIT; } } return ret; } static inline gboolean rspamd_email_address_check_and_add (const gchar *start, gsize len, GPtrArray *res, rspamd_mempool_t *pool, GString *ns) { struct rspamd_email_address addr; /* The whole email is likely address */ memset (&addr, 0, sizeof (addr)); rspamd_smtp_addr_parse (start, len, &addr); if (addr.flags & RSPAMD_EMAIL_ADDR_VALID) { rspamd_email_address_add (pool, res, &addr, ns); } else { /* Try heuristic */ if (rspamd_email_address_parse_heuristic (start, len, &addr)) { rspamd_email_address_add (pool, res, &addr, ns); return TRUE; } else { return FALSE; } } return TRUE; } GPtrArray * rspamd_email_address_from_mime (rspamd_mempool_t *pool, const gchar *hdr, guint len, GPtrArray *src) { GPtrArray *res = src; gboolean seen_at = FALSE; const gchar *p = hdr, *end = hdr + len, *c = hdr, *t; GString *ns; gint obraces, ebraces; enum { parse_name = 0, parse_quoted, parse_addr, skip_comment, skip_spaces } state = parse_name, next_state = parse_name; if (res == NULL) { res = g_ptr_array_sized_new (2); rspamd_mempool_add_destructor (pool, rspamd_email_address_list_destroy, res); } ns = g_string_sized_new (127); while (p < end) { switch (state) { case parse_name: if (*p == '"') { /* We need to strip last spaces and update `ns` */ if (p > c) { t = p - 1; while (t > c && g_ascii_isspace (*t)) { t --; } g_string_append_len (ns, c, t - c + 1); } state = parse_quoted; c = p + 1; } else if (*p == '<') { if (p > c) { t = p - 1; while (t > c && g_ascii_isspace (*t)) { t --; } g_string_append_len (ns, c, t - c + 1); } c = p; state = parse_addr; } else if (*p == ',') { if (p > c && seen_at) { /* * Last token must be the address: * e.g. Some name name@domain.com */ t = p - 1; while (t > c && g_ascii_isspace (*t)) { t --; } if (!rspamd_email_address_check_and_add (c, t - c + 1, res, pool, ns)) { rspamd_email_address_add (pool, res, NULL, ns); } /* Cleanup for the next use */ g_string_set_size (ns, 0); seen_at = FALSE; } state = skip_spaces; next_state = parse_name; } else if (*p == '@') { seen_at = TRUE; } else if (*p == '(') { if (p > c) { t = p - 1; while (t > c && g_ascii_isspace (*t)) { t --; } g_string_append_len (ns, c, t - c + 1); if (seen_at) { if (!rspamd_email_address_check_and_add (c, t - c + 1, res, pool, ns)) { rspamd_email_address_add (pool, res, NULL, ns); } g_string_set_size (ns, 0); seen_at = FALSE; } } c = p; obraces = 1; ebraces = 0; state = skip_comment; next_state = parse_name; } p ++; break; case parse_quoted: if (*p == '"') { if (p > c) { g_string_append_len (ns, c, p - c); } state = skip_spaces; next_state = parse_name; } p ++; break; case parse_addr: if (*p == '>') { if (!rspamd_email_address_check_and_add (c, p - c + 1, res, pool, ns)) { rspamd_email_address_add (pool, res, NULL, ns); } /* Cleanup for the next use */ g_string_set_size (ns, 0); seen_at = FALSE; state = skip_spaces; next_state = parse_name; } else if (*p == '@') { seen_at = TRUE; } else if (*p == '(') { obraces = 1; ebraces = 0; state = skip_comment; next_state = parse_addr; } p ++; break; case skip_spaces: if (!g_ascii_isspace (*p)) { c = p; state = next_state; } else { p ++; } break; case skip_comment: if (*p == '(') { obraces ++; } else if (*p == ')') { ebraces ++; } if (obraces == ebraces) { if (next_state == parse_name) { if (ns->len > 0) { /* Include comment in name if it has been seen */ if (p > c) { t = p - 1; while (t > c && g_ascii_isspace (*t)) { t --; } g_string_append_len (ns, c, t - c + 1); } } c = p + 1; } state = next_state; } p ++; break; } } /* Handle leftover */ switch (state) { case parse_name: /* Assume the whole header as name (bad thing) */ if (p > c) { while (p > c && g_ascii_isspace (*p)) { p --; } if (p > c) { if (seen_at) { /* The whole email is likely address */ if (!rspamd_email_address_check_and_add (c, p - c, res, pool, ns)) { if (res->len == 0) { rspamd_email_address_add (pool, res, NULL, ns); } } } else { /* No @ seen */ g_string_append_len (ns, c, p - c); if (res->len == 0) { rspamd_email_address_add (pool, res, NULL, ns); } } } else if (res->len == 0) { rspamd_email_address_add (pool, res, NULL, ns); } } break; case parse_addr: if (p > c) { if (!rspamd_email_address_check_and_add (c, p - c, res, pool, ns)) { if (res->len == 0) { rspamd_email_address_add (pool, res, NULL, ns); } } } break; case parse_quoted: case skip_comment: /* Unfinished quoted string or a comment */ break; default: /* Do nothing */ break; } g_string_free (ns, TRUE); return res; } void rspamd_email_address_list_destroy (gpointer ptr) { GPtrArray *ar = ptr; guint i; struct rspamd_email_address *addr; PTR_ARRAY_FOREACH (ar, i, addr) { rspamd_email_address_free (addr); } g_ptr_array_free (ar, TRUE); } void rspamd_smtp_maybe_process_smtp_comment (struct rspamd_task *task, const char *data, size_t len, struct received_header *rh) { if (!rh->by_hostname) { /* Heuristic to detect IP addresses like in Exim received: * [xxx]:port or [xxx] */ if (*data == '[' && len > 2) { const gchar *p = data + 1; gsize iplen = rspamd_memcspn (p, "]", len - 1); if (iplen > 0) { guchar tbuf[sizeof(struct in6_addr) + sizeof(guint32)]; if (rspamd_parse_inet_address_ip4 (p, iplen, tbuf) || rspamd_parse_inet_address_ip6 (p, iplen, tbuf)) { rh->comment_ip = rspamd_mempool_alloc (task->task_pool, iplen + 1); rspamd_strlcpy (rh->comment_ip, p, iplen + 1); } } } } }