1
1
mirror of https://github.com/systemd/systemd-stable.git synced 2025-01-10 01:17:44 +03:00

resolve: always request records to validate negative answer

Otherwise, dns_transaction_requires_nsec() may not find no required
transaction, and return true. That sets
`answer_dnssec_result = DNSSEC_NO_SIGNATURE`, and the entire transaction fails.

Fixes #21414.
This commit is contained in:
Yu Watanabe 2022-04-19 03:58:50 +09:00
parent cd2cdba2fe
commit 26b23d1187

View File

@ -2213,7 +2213,7 @@ static int dns_transaction_negative_trust_anchor_lookup(DnsTransaction *t, const
return link_negative_trust_anchor_lookup(t->scope->link, name); return link_negative_trust_anchor_lookup(t->scope->link, name);
} }
static int dns_transaction_has_unsigned_negative_answer(DnsTransaction *t) { static int dns_transaction_has_negative_answer(DnsTransaction *t) {
int r; int r;
assert(t); assert(t);
@ -2232,14 +2232,7 @@ static int dns_transaction_has_unsigned_negative_answer(DnsTransaction *t) {
r = dns_transaction_negative_trust_anchor_lookup(t, dns_resource_key_name(dns_transaction_key(t))); r = dns_transaction_negative_trust_anchor_lookup(t, dns_resource_key_name(dns_transaction_key(t)));
if (r < 0) if (r < 0)
return r; return r;
if (r > 0) return !r;
return false;
/* The answer does not contain any RRs that match to the
* question. If so, let's see if there are any NSEC/NSEC3 RRs
* included. If not, the answer is unsigned. */
return !dns_answer_contains_nsec_or_nsec3(t->answer);
} }
static int dns_transaction_is_primary_response(DnsTransaction *t, DnsResourceRecord *rr) { static int dns_transaction_is_primary_response(DnsTransaction *t, DnsResourceRecord *rr) {
@ -2563,14 +2556,15 @@ int dns_transaction_request_dnssec_keys(DnsTransaction *t) {
* we got. Now, let's request what we need to validate what we * we got. Now, let's request what we need to validate what we
* didn't get... */ * didn't get... */
r = dns_transaction_has_unsigned_negative_answer(t); r = dns_transaction_has_negative_answer(t);
if (r < 0) if (r < 0)
return r; return r;
if (r > 0) { if (r > 0) {
const char *name; const char *name, *signed_status;
uint16_t type = 0; uint16_t type = 0;
name = dns_resource_key_name(dns_transaction_key(t)); name = dns_resource_key_name(dns_transaction_key(t));
signed_status = dns_answer_contains_nsec_or_nsec3(t->answer) ? "signed" : "unsigned";
/* If this was a SOA or NS request, then check if there's a DS RR for the same domain. Note that this /* If this was a SOA or NS request, then check if there's a DS RR for the same domain. Note that this
* could also be used as indication that we are not at a zone apex, but in real world setups there are * could also be used as indication that we are not at a zone apex, but in real world setups there are
@ -2583,21 +2577,21 @@ int dns_transaction_request_dnssec_keys(DnsTransaction *t) {
r = dns_name_parent(&name); r = dns_name_parent(&name);
if (r > 0) { if (r > 0) {
type = DNS_TYPE_SOA; type = DNS_TYPE_SOA;
log_debug("Requesting parent SOA (→ %s) to validate transaction %" PRIu16 " (%s, unsigned empty DS response).", log_debug("Requesting parent SOA (→ %s) to validate transaction %" PRIu16 " (%s, %s empty DS response).",
name, t->id, dns_resource_key_name(dns_transaction_key(t))); name, t->id, dns_resource_key_name(dns_transaction_key(t)), signed_status);
} else } else
name = NULL; name = NULL;
} else if (IN_SET(dns_transaction_key(t)->type, DNS_TYPE_SOA, DNS_TYPE_NS)) { } else if (IN_SET(dns_transaction_key(t)->type, DNS_TYPE_SOA, DNS_TYPE_NS)) {
type = DNS_TYPE_DS; type = DNS_TYPE_DS;
log_debug("Requesting DS (→ %s) to validate transaction %" PRIu16 " (%s, unsigned empty SOA/NS response).", log_debug("Requesting DS (→ %s) to validate transaction %" PRIu16 " (%s, %s empty SOA/NS response).",
name, t->id, name); name, t->id, name, signed_status);
} else { } else {
type = DNS_TYPE_SOA; type = DNS_TYPE_SOA;
log_debug("Requesting SOA (→ %s) to validate transaction %" PRIu16 " (%s, unsigned empty non-SOA/NS/DS response).", log_debug("Requesting SOA (→ %s) to validate transaction %" PRIu16 " (%s, %s empty non-SOA/NS/DS response).",
name, t->id, name); name, t->id, name, signed_status);
} }
if (name) { if (name) {