summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorWolfgang Breyha <wbreyha@gmx.net>2019-12-20 14:01:23 +0000
committerJeremy Harris <jgh146exb@wizmail.org>2019-12-21 20:31:03 +0000
commit4a3709fb2cfb1ec26ac60bcdb8bbc078d3377ddc (patch)
treedd82436d629ee4dd98dbbab04678067a91186031
parent847a015ae17b3fa66154088009803636f95b2c6f (diff)
SPF: only require "v=spf1" on TXT DNS records during lookups. Bug 2499
-rw-r--r--src/src/spf.c60
1 files changed, 44 insertions, 16 deletions
diff --git a/src/src/spf.c b/src/src/spf.c
index 1955b5d96..b7041d3e3 100644
--- a/src/src/spf.c
+++ b/src/src/spf.c
@@ -43,17 +43,15 @@ dns_answer * dnsa = store_get_dns_answer();
dns_scan dnss;
SPF_dns_rr_t * spfrr;
-DEBUG(D_receive) debug_printf("SPF_dns_exim_lookup\n");
+DEBUG(D_receive) debug_printf("SPF_dns_exim_lookup '%s'\n", domain);
if (dns_lookup(dnsa, US domain, rr_type, NULL) == DNS_SUCCEED)
for (dns_record * rr = dns_next_rr(dnsa, &dnss, RESET_ANSWERS); rr;
rr = dns_next_rr(dnsa, &dnss, RESET_NEXT))
if ( rr->type == rr_type
- && Ustrncmp(rr->data+1, "v=spf1", 6) == 0)
+ && (rr_type != T_TXT || Ustrncmp(rr->data+1, "v=spf1", 6) == 0))
{
- gstring * g = NULL;
- uschar chunk_len;
- uschar * s;
+ const uschar * s = rr->data;
SPF_dns_rr_t srr = {
.domain = CS rr->name, /* query information */
.domain_buf_len = DNS_MAXNAME,
@@ -71,19 +69,49 @@ if (dns_lookup(dnsa, US domain, rr_type, NULL) == DNS_SUCCEED)
.source = spf_dns_server
};
- for (int off = 0; off < rr->size; off += chunk_len)
+ switch(rr_type)
{
- chunk_len = (rr->data)[off++];
- g = string_catn(g, US ((rr->data)+off), chunk_len);
+ case T_MX:
+ s += 2; /* skip the MX precedence field */
+ case T_PTR:
+ {
+ uschar * buf = store_malloc(256);
+ (void)dn_expand(dnsa->answer, dnsa->answer + dnsa->answerlen, s,
+ (DN_EXPAND_ARG4_TYPE)buf, 256);
+ s = buf;
+ break;
+ }
+
+ case T_TXT:
+ {
+ gstring * g = NULL;
+ uschar chunk_len;
+ for (int off = 0; off < rr->size; off += chunk_len)
+ {
+ if (!(chunk_len = s[off++])) break;
+ g = string_catn(g, s+off, chunk_len);
+ }
+ if (!g)
+ {
+ HDEBUG(D_host_lookup) debug_printf("IP address lookup yielded an "
+ "empty name: treated as non-existent host name\n");
+ continue;
+ }
+ gstring_release_unused(g);
+ s = string_copy_malloc(string_from_gstring(g));
+ break;
+ }
+
+ case T_A:
+ case T_AAAA:
+ default:
+ {
+ uschar * buf = store_malloc(dnsa->answerlen + 1);
+ s = memcpy(buf, s, dnsa->answerlen + 1);
+ break;
+ }
}
- if (!g)
- {
- HDEBUG(D_host_lookup) debug_printf("IP address lookup yielded an "
- "empty name: treated as non-existent host name\n");
- continue;
- }
- gstring_release_unused(g);
- s = string_copy_malloc(string_from_gstring(g));
+ DEBUG(D_receive) debug_printf("SPF_dns_exim_lookup '%s'\n", s);
srr.rr = (void *) &s;
/* spfrr->rr must have been malloc()d for this */