[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "net/dns/dns_test_util.h" |
| 6 | |
[email protected] | d9806a97 | 2014-02-26 18:14:57 | [diff] [blame] | 7 | #include "base/big_endian.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 8 | #include "base/bind.h" |
skyostil | 4891b25b | 2015-06-11 11:43:45 | [diff] [blame] | 9 | #include "base/location.h" |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 10 | #include "base/numerics/safe_conversions.h" |
skyostil | 4891b25b | 2015-06-11 11:43:45 | [diff] [blame] | 11 | #include "base/single_thread_task_runner.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 12 | #include "base/sys_byteorder.h" |
gab | f767595f | 2016-05-11 18:50:35 | [diff] [blame] | 13 | #include "base/threading/thread_task_runner_handle.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 14 | #include "net/base/io_buffer.h" |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 15 | #include "net/base/ip_address.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 16 | #include "net/base/net_errors.h" |
[email protected] | 0adcb2b | 2012-08-15 21:30:46 | [diff] [blame] | 17 | #include "net/dns/address_sorter.h" |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 18 | #include "net/dns/dns_hosts.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 19 | #include "net/dns/dns_query.h" |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 20 | #include "net/dns/dns_session.h" |
Eric Orth | 9300cc6 | 2020-08-21 00:29:34 | [diff] [blame^] | 21 | #include "net/dns/dns_socket_allocator.h" |
tfarina | 77021d6 | 2015-10-11 20:19:03 | [diff] [blame] | 22 | #include "net/dns/dns_util.h" |
Eric Orth | 069a3c6e | 2020-01-31 23:14:24 | [diff] [blame] | 23 | #include "net/dns/resolve_context.h" |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 24 | #include "testing/gtest/include/gtest/gtest.h" |
| 25 | |
| 26 | namespace net { |
| 27 | namespace { |
| 28 | |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 29 | const uint8_t kMalformedResponseHeader[] = { |
| 30 | // Header |
| 31 | 0x00, 0x14, // Arbitrary ID |
| 32 | 0x81, 0x80, // Standard query response, RA, no error |
| 33 | 0x00, 0x01, // 1 question |
| 34 | 0x00, 0x01, // 1 RR (answers) |
| 35 | 0x00, 0x00, // 0 authority RRs |
| 36 | 0x00, 0x00, // 0 additional RRs |
| 37 | }; |
| 38 | |
| 39 | // Create a response containing a valid question (as would normally be validated |
| 40 | // in DnsTransaction) but completely missing a header-declared answer. |
| 41 | std::unique_ptr<DnsResponse> CreateMalformedResponse(std::string hostname, |
| 42 | uint16_t type) { |
| 43 | std::string dns_name; |
| 44 | CHECK(DNSDomainFromDot(hostname, &dns_name)); |
| 45 | DnsQuery query(0x14 /* id */, dns_name, type); |
| 46 | |
| 47 | // Build response to simulate the barebones validation DnsResponse applies to |
| 48 | // responses received from the network. |
| 49 | auto buffer = base::MakeRefCounted<IOBufferWithSize>( |
| 50 | sizeof(kMalformedResponseHeader) + query.question().size()); |
| 51 | memcpy(buffer->data(), kMalformedResponseHeader, |
| 52 | sizeof(kMalformedResponseHeader)); |
| 53 | memcpy(buffer->data() + sizeof(kMalformedResponseHeader), |
| 54 | query.question().data(), query.question().size()); |
| 55 | |
| 56 | auto response = std::make_unique<DnsResponse>(buffer, buffer->size()); |
| 57 | CHECK(response->InitParseWithoutQuery(buffer->size())); |
| 58 | |
| 59 | return response; |
| 60 | } |
| 61 | |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 62 | class MockAddressSorter : public AddressSorter { |
| 63 | public: |
Chris Watkins | 68b1503 | 2017-12-01 03:07:13 | [diff] [blame] | 64 | ~MockAddressSorter() override = default; |
Bence Béky | d5e474c | 2018-08-02 18:45:08 | [diff] [blame] | 65 | void Sort(const AddressList& list, CallbackType callback) const override { |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 66 | // Do nothing. |
Bence Béky | d5e474c | 2018-08-02 18:45:08 | [diff] [blame] | 67 | std::move(callback).Run(true, list); |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 68 | } |
| 69 | }; |
| 70 | |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 71 | DnsResourceRecord BuildCannonnameRecord(std::string name, |
| 72 | std::string cannonname) { |
| 73 | DCHECK(!name.empty()); |
| 74 | DCHECK(!cannonname.empty()); |
| 75 | |
| 76 | DnsResourceRecord record; |
| 77 | record.name = std::move(name); |
| 78 | record.type = dns_protocol::kTypeCNAME; |
| 79 | record.klass = dns_protocol::kClassIN; |
| 80 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 81 | CHECK(DNSDomainFromDot(cannonname, &record.owned_rdata)); |
| 82 | record.rdata = record.owned_rdata; |
| 83 | |
| 84 | return record; |
| 85 | } |
| 86 | |
| 87 | // Note: This is not a fully compliant SOA record, merely the bare amount needed |
| 88 | // in DnsRecord::ParseToAddressList() processessing. This record will not pass |
| 89 | // RecordParsed validation. |
| 90 | DnsResourceRecord BuildSoaRecord(std::string name) { |
| 91 | DCHECK(!name.empty()); |
| 92 | |
| 93 | DnsResourceRecord record; |
| 94 | record.name = std::move(name); |
| 95 | record.type = dns_protocol::kTypeSOA; |
| 96 | record.klass = dns_protocol::kClassIN; |
| 97 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 98 | record.SetOwnedRdata("fake_rdata"); |
| 99 | |
| 100 | return record; |
| 101 | } |
| 102 | |
| 103 | DnsResourceRecord BuildTextRecord(std::string name, |
| 104 | std::vector<std::string> text_strings) { |
| 105 | DCHECK(!name.empty()); |
| 106 | DCHECK(!text_strings.empty()); |
| 107 | |
| 108 | DnsResourceRecord record; |
| 109 | record.name = std::move(name); |
| 110 | record.type = dns_protocol::kTypeTXT; |
| 111 | record.klass = dns_protocol::kClassIN; |
| 112 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 113 | |
| 114 | std::string rdata; |
| 115 | for (std::string text_string : text_strings) { |
| 116 | DCHECK(!text_string.empty()); |
| 117 | |
dalyk | ad3f6c3 | 2019-03-06 13:38:33 | [diff] [blame] | 118 | rdata += base::checked_cast<unsigned char>(text_string.size()); |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 119 | rdata += std::move(text_string); |
| 120 | } |
| 121 | record.SetOwnedRdata(std::move(rdata)); |
| 122 | |
| 123 | return record; |
| 124 | } |
| 125 | |
Eric Orth | e9db8d23 | 2019-01-14 21:24:45 | [diff] [blame] | 126 | DnsResourceRecord BuildPointerRecord(std::string name, |
| 127 | std::string pointer_name) { |
| 128 | DCHECK(!name.empty()); |
| 129 | DCHECK(!pointer_name.empty()); |
| 130 | |
| 131 | DnsResourceRecord record; |
| 132 | record.name = std::move(name); |
| 133 | record.type = dns_protocol::kTypePTR; |
| 134 | record.klass = dns_protocol::kClassIN; |
| 135 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 136 | CHECK(DNSDomainFromDot(pointer_name, &record.owned_rdata)); |
| 137 | record.rdata = record.owned_rdata; |
| 138 | |
| 139 | return record; |
| 140 | } |
| 141 | |
Eric Orth | a625b04 | 2019-01-16 01:14:45 | [diff] [blame] | 142 | DnsResourceRecord BuildServiceRecord(std::string name, |
| 143 | TestServiceRecord service) { |
| 144 | DCHECK(!name.empty()); |
| 145 | DCHECK(!service.target.empty()); |
| 146 | |
| 147 | DnsResourceRecord record; |
| 148 | record.name = std::move(name); |
| 149 | record.type = dns_protocol::kTypeSRV; |
| 150 | record.klass = dns_protocol::kClassIN; |
| 151 | record.ttl = base::TimeDelta::FromHours(5).InSeconds(); |
| 152 | |
| 153 | std::string rdata; |
| 154 | char num_buffer[2]; |
| 155 | base::WriteBigEndian(num_buffer, service.priority); |
| 156 | rdata.append(num_buffer, 2); |
| 157 | base::WriteBigEndian(num_buffer, service.weight); |
| 158 | rdata.append(num_buffer, 2); |
| 159 | base::WriteBigEndian(num_buffer, service.port); |
| 160 | rdata.append(num_buffer, 2); |
| 161 | std::string dns_name; |
| 162 | CHECK(DNSDomainFromDot(service.target, &dns_name)); |
| 163 | rdata += dns_name; |
| 164 | |
| 165 | record.SetOwnedRdata(std::move(rdata)); |
| 166 | |
| 167 | return record; |
| 168 | } |
| 169 | |
David Van Cleve | c719c88 | 2019-11-04 15:47:40 | [diff] [blame] | 170 | |
David Van Cleve | c719c88 | 2019-11-04 15:47:40 | [diff] [blame] | 171 | |
Daniel McArdle | 2daa3ec | 2020-06-03 16:05:48 | [diff] [blame] | 172 | DnsResourceRecord BuildIntegrityRecord( |
| 173 | std::string name, |
| 174 | const std::vector<uint8_t>& serialized_rdata) { |
| 175 | CHECK(!name.empty()); |
| 176 | |
| 177 | DnsResourceRecord record; |
| 178 | record.name = std::move(name); |
| 179 | record.type = dns_protocol::kExperimentalTypeIntegrity; |
| 180 | record.klass = dns_protocol::kClassIN; |
| 181 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 182 | |
| 183 | std::string serialized_rdata_str(serialized_rdata.begin(), |
| 184 | serialized_rdata.end()); |
| 185 | record.SetOwnedRdata(std::move(serialized_rdata_str)); |
| 186 | |
| 187 | CHECK_EQ(record.rdata.data(), record.owned_rdata.data()); |
| 188 | |
| 189 | return record; |
| 190 | } |
| 191 | |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 192 | } // namespace |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 193 | |
David Van Cleve | 34c74c7 | 2019-10-31 19:58:01 | [diff] [blame] | 194 | DnsResourceRecord BuildTestAddressRecord(std::string name, |
| 195 | const IPAddress& ip) { |
| 196 | DCHECK(!name.empty()); |
| 197 | DCHECK(ip.IsValid()); |
| 198 | |
| 199 | DnsResourceRecord record; |
| 200 | record.name = std::move(name); |
| 201 | record.type = ip.IsIPv4() ? dns_protocol::kTypeA : dns_protocol::kTypeAAAA; |
| 202 | record.klass = dns_protocol::kClassIN; |
| 203 | record.ttl = base::TimeDelta::FromDays(1).InSeconds(); |
| 204 | record.SetOwnedRdata(net::IPAddressToPackedString(ip)); |
| 205 | |
| 206 | return record; |
| 207 | } |
| 208 | |
David Van Cleve | c719c88 | 2019-11-04 15:47:40 | [diff] [blame] | 209 | |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 210 | std::unique_ptr<DnsResponse> BuildTestDnsResponse(std::string name, |
| 211 | const IPAddress& ip) { |
| 212 | DCHECK(ip.IsValid()); |
| 213 | |
David Van Cleve | 34c74c7 | 2019-10-31 19:58:01 | [diff] [blame] | 214 | std::vector<DnsResourceRecord> answers = {BuildTestAddressRecord(name, ip)}; |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 215 | std::string dns_name; |
| 216 | CHECK(DNSDomainFromDot(name, &dns_name)); |
| 217 | base::Optional<DnsQuery> query( |
| 218 | base::in_place, 0, dns_name, |
| 219 | ip.IsIPv4() ? dns_protocol::kTypeA : dns_protocol::kTypeAAAA); |
| 220 | return std::make_unique<DnsResponse>( |
| 221 | 0, false, std::move(answers), |
| 222 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 223 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 224 | } |
| 225 | |
dalyk | c0ab072 | 2019-03-14 17:00:58 | [diff] [blame] | 226 | std::unique_ptr<DnsResponse> BuildTestDnsResponseWithCname( |
| 227 | std::string name, |
| 228 | const IPAddress& ip, |
| 229 | std::string cannonname) { |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 230 | DCHECK(ip.IsValid()); |
| 231 | DCHECK(!cannonname.empty()); |
| 232 | |
| 233 | std::vector<DnsResourceRecord> answers = { |
| 234 | BuildCannonnameRecord(name, cannonname), |
David Van Cleve | 34c74c7 | 2019-10-31 19:58:01 | [diff] [blame] | 235 | BuildTestAddressRecord(cannonname, ip)}; |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 236 | std::string dns_name; |
| 237 | CHECK(DNSDomainFromDot(name, &dns_name)); |
| 238 | base::Optional<DnsQuery> query( |
| 239 | base::in_place, 0, dns_name, |
| 240 | ip.IsIPv4() ? dns_protocol::kTypeA : dns_protocol::kTypeAAAA); |
| 241 | return std::make_unique<DnsResponse>( |
| 242 | 0, false, std::move(answers), |
| 243 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 244 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 245 | } |
| 246 | |
dalyk | c0ab072 | 2019-03-14 17:00:58 | [diff] [blame] | 247 | std::unique_ptr<DnsResponse> BuildTestDnsTextResponse( |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 248 | std::string name, |
| 249 | std::vector<std::vector<std::string>> text_records, |
| 250 | std::string answer_name) { |
| 251 | if (answer_name.empty()) |
| 252 | answer_name = name; |
| 253 | |
| 254 | std::vector<DnsResourceRecord> answers; |
Eric Orth | a625b04 | 2019-01-16 01:14:45 | [diff] [blame] | 255 | for (std::vector<std::string>& text_record : text_records) { |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 256 | answers.push_back(BuildTextRecord(answer_name, std::move(text_record))); |
| 257 | } |
| 258 | |
| 259 | std::string dns_name; |
| 260 | CHECK(DNSDomainFromDot(name, &dns_name)); |
| 261 | base::Optional<DnsQuery> query(base::in_place, 0, dns_name, |
| 262 | dns_protocol::kTypeTXT); |
| 263 | |
| 264 | return std::make_unique<DnsResponse>( |
| 265 | 0, false, std::move(answers), |
| 266 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 267 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 268 | } |
| 269 | |
Eric Orth | e9db8d23 | 2019-01-14 21:24:45 | [diff] [blame] | 270 | std::unique_ptr<DnsResponse> BuildTestDnsPointerResponse( |
| 271 | std::string name, |
| 272 | std::vector<std::string> pointer_names, |
| 273 | std::string answer_name) { |
| 274 | if (answer_name.empty()) |
| 275 | answer_name = name; |
| 276 | |
| 277 | std::vector<DnsResourceRecord> answers; |
Eric Orth | a625b04 | 2019-01-16 01:14:45 | [diff] [blame] | 278 | for (std::string& pointer_name : pointer_names) { |
Eric Orth | e9db8d23 | 2019-01-14 21:24:45 | [diff] [blame] | 279 | answers.push_back(BuildPointerRecord(answer_name, std::move(pointer_name))); |
| 280 | } |
| 281 | |
| 282 | std::string dns_name; |
| 283 | CHECK(DNSDomainFromDot(name, &dns_name)); |
| 284 | base::Optional<DnsQuery> query(base::in_place, 0, dns_name, |
| 285 | dns_protocol::kTypePTR); |
| 286 | |
| 287 | return std::make_unique<DnsResponse>( |
| 288 | 0, false, std::move(answers), |
| 289 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 290 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 291 | } |
| 292 | |
dalyk | c0ab072 | 2019-03-14 17:00:58 | [diff] [blame] | 293 | std::unique_ptr<DnsResponse> BuildTestDnsServiceResponse( |
Eric Orth | a625b04 | 2019-01-16 01:14:45 | [diff] [blame] | 294 | std::string name, |
| 295 | std::vector<TestServiceRecord> service_records, |
| 296 | std::string answer_name) { |
| 297 | if (answer_name.empty()) |
| 298 | answer_name = name; |
| 299 | |
| 300 | std::vector<DnsResourceRecord> answers; |
| 301 | for (TestServiceRecord& service_record : service_records) { |
| 302 | answers.push_back( |
| 303 | BuildServiceRecord(answer_name, std::move(service_record))); |
| 304 | } |
| 305 | |
| 306 | std::string dns_name; |
| 307 | CHECK(DNSDomainFromDot(name, &dns_name)); |
| 308 | base::Optional<DnsQuery> query(base::in_place, 0, dns_name, |
| 309 | dns_protocol::kTypeSRV); |
| 310 | |
| 311 | return std::make_unique<DnsResponse>( |
| 312 | 0, false, std::move(answers), |
| 313 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 314 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 315 | } |
| 316 | |
Daniel McArdle | 2daa3ec | 2020-06-03 16:05:48 | [diff] [blame] | 317 | std::unique_ptr<DnsResponse> BuildTestDnsIntegrityResponse( |
| 318 | std::string hostname, |
| 319 | const std::vector<uint8_t>& serialized_rdata) { |
| 320 | CHECK(!hostname.empty()); |
| 321 | |
| 322 | std::vector<DnsResourceRecord> answers{ |
| 323 | BuildIntegrityRecord(hostname, serialized_rdata)}; |
| 324 | |
| 325 | std::string dns_name; |
| 326 | CHECK(DNSDomainFromDot(hostname, &dns_name)); |
| 327 | base::Optional<DnsQuery> query(base::in_place, 0, dns_name, |
| 328 | dns_protocol::kExperimentalTypeIntegrity); |
| 329 | |
| 330 | return std::make_unique<DnsResponse>( |
| 331 | 0, false, std::move(answers), |
| 332 | std::vector<DnsResourceRecord>() /* authority_records */, |
| 333 | std::vector<DnsResourceRecord>() /* additional_records */, query); |
| 334 | } |
| 335 | |
Eric Orth | 828bd3ae | 2018-12-12 17:30:36 | [diff] [blame] | 336 | MockDnsClientRule::Result::Result(ResultType type) : type(type) {} |
| 337 | |
| 338 | MockDnsClientRule::Result::Result(std::unique_ptr<DnsResponse> response) |
| 339 | : type(OK), response(std::move(response)) {} |
| 340 | |
| 341 | MockDnsClientRule::Result::Result(Result&& result) = default; |
| 342 | |
| 343 | MockDnsClientRule::Result::~Result() = default; |
| 344 | |
| 345 | MockDnsClientRule::Result& MockDnsClientRule::Result::operator=( |
| 346 | Result&& result) = default; |
| 347 | |
Eric Orth | 502d568 | 2019-04-12 00:21:33 | [diff] [blame] | 348 | MockDnsClientRule::MockDnsClientRule(const std::string& prefix, |
| 349 | uint16_t qtype, |
dalyk | 4f4ac71 | 2019-05-31 16:33:13 | [diff] [blame] | 350 | bool secure, |
Eric Orth | 502d568 | 2019-04-12 00:21:33 | [diff] [blame] | 351 | Result result, |
| 352 | bool delay, |
| 353 | URLRequestContext* context) |
| 354 | : result(std::move(result)), |
| 355 | prefix(prefix), |
| 356 | qtype(qtype), |
dalyk | 4f4ac71 | 2019-05-31 16:33:13 | [diff] [blame] | 357 | secure(secure), |
Eric Orth | 502d568 | 2019-04-12 00:21:33 | [diff] [blame] | 358 | delay(delay), |
| 359 | context(context) {} |
| 360 | |
| 361 | MockDnsClientRule::MockDnsClientRule(MockDnsClientRule&& rule) = default; |
| 362 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 363 | // A DnsTransaction which uses MockDnsClientRuleList to determine the response. |
| 364 | class MockDnsTransactionFactory::MockTransaction |
| 365 | : public DnsTransaction, |
| 366 | public base::SupportsWeakPtr<MockTransaction> { |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 367 | public: |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 368 | MockTransaction(const MockDnsClientRuleList& rules, |
| 369 | const std::string& hostname, |
| 370 | uint16_t qtype, |
| 371 | bool secure, |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 372 | bool force_doh_server_available, |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 373 | DnsConfig::SecureDnsMode secure_dns_mode, |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 374 | ResolveContext* resolve_context, |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 375 | DnsTransactionFactory::CallbackType callback) |
| 376 | : result_(MockDnsClientRule::FAIL), |
| 377 | hostname_(hostname), |
| 378 | qtype_(qtype), |
| 379 | callback_(std::move(callback)), |
| 380 | started_(false), |
| 381 | delayed_(false) { |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 382 | // Do not allow matching any rules if transaction is secure and no DoH |
| 383 | // servers are available. |
| 384 | if (!secure || force_doh_server_available || |
| 385 | resolve_context->NumAvailableDohServers( |
| 386 | resolve_context->current_session_for_testing()) > 0) { |
| 387 | // Find the relevant rule which matches |qtype|, |secure|, prefix of |
| 388 | // |hostname|, and |url_request_context| (iff the rule context is not |
| 389 | // null). |
| 390 | for (size_t i = 0; i < rules.size(); ++i) { |
| 391 | const std::string& prefix = rules[i].prefix; |
| 392 | if ((rules[i].qtype == qtype) && (rules[i].secure == secure) && |
| 393 | (hostname.size() >= prefix.size()) && |
| 394 | (hostname.compare(0, prefix.size(), prefix) == 0) && |
| 395 | (!rules[i].context || |
| 396 | rules[i].context == resolve_context->url_request_context())) { |
| 397 | const MockDnsClientRule::Result* result = &rules[i].result; |
| 398 | result_ = MockDnsClientRule::Result(result->type); |
| 399 | delayed_ = rules[i].delay; |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 400 | |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 401 | // Generate a DnsResponse when not provided with the rule. |
| 402 | std::vector<DnsResourceRecord> authority_records; |
| 403 | std::string dns_name; |
| 404 | CHECK(DNSDomainFromDot(hostname_, &dns_name)); |
| 405 | base::Optional<DnsQuery> query(base::in_place, 22 /* id */, dns_name, |
| 406 | qtype_); |
| 407 | switch (result->type) { |
| 408 | case MockDnsClientRule::NODOMAIN: |
| 409 | case MockDnsClientRule::EMPTY: |
| 410 | DCHECK(!result->response); // Not expected to be provided. |
| 411 | authority_records = {BuildSoaRecord(hostname_)}; |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 412 | result_.response = std::make_unique<DnsResponse>( |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 413 | 22 /* id */, false /* is_authoritative */, |
| 414 | std::vector<DnsResourceRecord>() /* answers */, |
| 415 | authority_records, |
| 416 | std::vector<DnsResourceRecord>() /* additional_records */, |
| 417 | query, |
| 418 | result->type == MockDnsClientRule::NODOMAIN |
| 419 | ? dns_protocol::kRcodeNXDOMAIN |
| 420 | : 0); |
| 421 | break; |
| 422 | case MockDnsClientRule::FAIL: |
| 423 | case MockDnsClientRule::TIMEOUT: |
| 424 | DCHECK(!result->response); // Not expected to be provided. |
| 425 | break; |
| 426 | case MockDnsClientRule::OK: |
| 427 | if (result->response) { |
| 428 | // Copy response in case |rules| are destroyed before the |
| 429 | // transaction completes. |
| 430 | result_.response = std::make_unique<DnsResponse>( |
| 431 | result->response->io_buffer(), |
| 432 | result->response->io_buffer_size()); |
| 433 | CHECK(result_.response->InitParseWithoutQuery( |
| 434 | result->response->io_buffer_size())); |
| 435 | } else { |
| 436 | // Generated response only available for address types. |
| 437 | DCHECK(qtype_ == dns_protocol::kTypeA || |
| 438 | qtype_ == dns_protocol::kTypeAAAA); |
| 439 | result_.response = BuildTestDnsResponse( |
| 440 | hostname_, qtype_ == dns_protocol::kTypeA |
| 441 | ? IPAddress::IPv4Localhost() |
| 442 | : IPAddress::IPv6Localhost()); |
| 443 | } |
| 444 | break; |
| 445 | case MockDnsClientRule::MALFORMED: |
| 446 | DCHECK(!result->response); // Not expected to be provided. |
| 447 | result_.response = CreateMalformedResponse(hostname_, qtype_); |
| 448 | break; |
| 449 | } |
Eric Orth | 2a61be8 | 2020-02-12 17:56:48 | [diff] [blame] | 450 | |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 451 | break; |
| 452 | } |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 453 | } |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 454 | } |
[email protected] | 0adcb2b | 2012-08-15 21:30:46 | [diff] [blame] | 455 | } |
| 456 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 457 | const std::string& GetHostname() const override { return hostname_; } |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 458 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 459 | uint16_t GetType() const override { return qtype_; } |
| 460 | |
| 461 | void Start() override { |
| 462 | EXPECT_FALSE(started_); |
| 463 | started_ = true; |
| 464 | if (delayed_) |
| 465 | return; |
| 466 | // Using WeakPtr to cleanly cancel when transaction is destroyed. |
| 467 | base::ThreadTaskRunnerHandle::Get()->PostTask( |
| 468 | FROM_HERE, base::BindOnce(&MockTransaction::Finish, AsWeakPtr())); |
| 469 | } |
| 470 | |
| 471 | void FinishDelayedTransaction() { |
| 472 | EXPECT_TRUE(delayed_); |
| 473 | delayed_ = false; |
| 474 | Finish(); |
| 475 | } |
| 476 | |
| 477 | bool delayed() const { return delayed_; } |
| 478 | |
| 479 | private: |
| 480 | void Finish() { |
| 481 | switch (result_.type) { |
| 482 | case MockDnsClientRule::NODOMAIN: |
| 483 | case MockDnsClientRule::FAIL: |
| 484 | std::move(callback_).Run(this, ERR_NAME_NOT_RESOLVED, |
Daniel McArdle | 4e2037e | 2020-06-11 22:40:51 | [diff] [blame] | 485 | result_.response.get(), base::nullopt); |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 486 | break; |
| 487 | case MockDnsClientRule::EMPTY: |
| 488 | case MockDnsClientRule::OK: |
| 489 | case MockDnsClientRule::MALFORMED: |
Daniel McArdle | 4e2037e | 2020-06-11 22:40:51 | [diff] [blame] | 490 | std::move(callback_).Run(this, OK, result_.response.get(), |
| 491 | base::nullopt); |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 492 | break; |
| 493 | case MockDnsClientRule::TIMEOUT: |
Daniel McArdle | 4e2037e | 2020-06-11 22:40:51 | [diff] [blame] | 494 | std::move(callback_).Run(this, ERR_DNS_TIMED_OUT, nullptr, |
| 495 | base::nullopt); |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 496 | break; |
| 497 | } |
| 498 | } |
| 499 | |
| 500 | void SetRequestPriority(RequestPriority priority) override {} |
| 501 | |
| 502 | MockDnsClientRule::Result result_; |
| 503 | const std::string hostname_; |
| 504 | const uint16_t qtype_; |
| 505 | DnsTransactionFactory::CallbackType callback_; |
| 506 | bool started_; |
| 507 | bool delayed_; |
[email protected] | 0adcb2b | 2012-08-15 21:30:46 | [diff] [blame] | 508 | }; |
| 509 | |
Eric Orth | 41b249e0 | 2020-01-03 20:21:47 | [diff] [blame] | 510 | class MockDnsTransactionFactory::MockDohProbeRunner : public DnsProbeRunner { |
| 511 | public: |
| 512 | explicit MockDohProbeRunner(base::WeakPtr<MockDnsTransactionFactory> factory) |
| 513 | : factory_(std::move(factory)) {} |
| 514 | |
| 515 | ~MockDohProbeRunner() override { |
| 516 | if (factory_) |
| 517 | factory_->running_doh_probe_runners_.erase(this); |
| 518 | } |
| 519 | |
Eric Orth | 020e5fe | 2020-03-12 17:43:43 | [diff] [blame] | 520 | void Start(bool network_change) override { |
Eric Orth | 41b249e0 | 2020-01-03 20:21:47 | [diff] [blame] | 521 | DCHECK(factory_); |
| 522 | factory_->running_doh_probe_runners_.insert(this); |
| 523 | } |
| 524 | |
Eric Orth | 41b249e0 | 2020-01-03 20:21:47 | [diff] [blame] | 525 | base::TimeDelta GetDelayUntilNextProbeForTest( |
| 526 | size_t doh_server_index) const override { |
| 527 | NOTREACHED(); |
| 528 | return base::TimeDelta(); |
| 529 | } |
| 530 | |
| 531 | private: |
| 532 | base::WeakPtr<MockDnsTransactionFactory> factory_; |
| 533 | }; |
| 534 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 535 | MockDnsTransactionFactory::MockDnsTransactionFactory( |
| 536 | MockDnsClientRuleList rules) |
| 537 | : rules_(std::move(rules)) {} |
| 538 | |
| 539 | MockDnsTransactionFactory::~MockDnsTransactionFactory() = default; |
| 540 | |
| 541 | std::unique_ptr<DnsTransaction> MockDnsTransactionFactory::CreateTransaction( |
| 542 | const std::string& hostname, |
| 543 | uint16_t qtype, |
| 544 | DnsTransactionFactory::CallbackType callback, |
| 545 | const NetLogWithSource&, |
| 546 | bool secure, |
| 547 | DnsConfig::SecureDnsMode secure_dns_mode, |
Eric Orth | 069a3c6e | 2020-01-31 23:14:24 | [diff] [blame] | 548 | ResolveContext* resolve_context) { |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 549 | std::unique_ptr<MockTransaction> transaction = |
Eric Orth | 069a3c6e | 2020-01-31 23:14:24 | [diff] [blame] | 550 | std::make_unique<MockTransaction>( |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 551 | rules_, hostname, qtype, secure, force_doh_server_available_, |
| 552 | secure_dns_mode, resolve_context, std::move(callback)); |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 553 | if (transaction->delayed()) |
| 554 | delayed_transactions_.push_back(transaction->AsWeakPtr()); |
| 555 | return transaction; |
| 556 | } |
| 557 | |
Eric Orth | 41b249e0 | 2020-01-03 20:21:47 | [diff] [blame] | 558 | std::unique_ptr<DnsProbeRunner> MockDnsTransactionFactory::CreateDohProbeRunner( |
Eric Orth | 069a3c6e | 2020-01-31 23:14:24 | [diff] [blame] | 559 | ResolveContext* resolve_context) { |
Eric Orth | 41b249e0 | 2020-01-03 20:21:47 | [diff] [blame] | 560 | return std::make_unique<MockDohProbeRunner>(weak_ptr_factory_.GetWeakPtr()); |
| 561 | } |
| 562 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 563 | void MockDnsTransactionFactory::AddEDNSOption(const OptRecordRdata::Opt& opt) {} |
| 564 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 565 | DnsConfig::SecureDnsMode MockDnsTransactionFactory::GetSecureDnsModeForTest() { |
| 566 | return DnsConfig::SecureDnsMode::AUTOMATIC; |
| 567 | } |
| 568 | |
| 569 | void MockDnsTransactionFactory::CompleteDelayedTransactions() { |
| 570 | DelayedTransactionList old_delayed_transactions; |
| 571 | old_delayed_transactions.swap(delayed_transactions_); |
| 572 | for (auto it = old_delayed_transactions.begin(); |
| 573 | it != old_delayed_transactions.end(); ++it) { |
| 574 | if (it->get()) |
| 575 | (*it)->FinishDelayedTransaction(); |
| 576 | } |
| 577 | } |
| 578 | |
David Van Cleve | 1fb5e8c | 2019-11-04 16:45:16 | [diff] [blame] | 579 | bool MockDnsTransactionFactory::CompleteOneDelayedTransactionOfType( |
| 580 | DnsQueryType type) { |
| 581 | for (base::WeakPtr<MockTransaction>& t : delayed_transactions_) { |
| 582 | if (t && t->GetType() == DnsQueryTypeToQtype(type)) { |
| 583 | t->FinishDelayedTransaction(); |
| 584 | t.reset(); |
| 585 | return true; |
| 586 | } |
| 587 | } |
| 588 | return false; |
| 589 | } |
| 590 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 591 | MockDnsClient::MockDnsClient(DnsConfig config, MockDnsClientRuleList rules) |
| 592 | : config_(std::move(config)), |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 593 | factory_(new MockDnsTransactionFactory(std::move(rules))), |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 594 | address_sorter_(new MockAddressSorter()) { |
| 595 | effective_config_ = BuildEffectiveConfig(); |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 596 | session_ = BuildSession(); |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 597 | } |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 598 | |
Chris Watkins | 68b1503 | 2017-12-01 03:07:13 | [diff] [blame] | 599 | MockDnsClient::~MockDnsClient() = default; |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 600 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 601 | bool MockDnsClient::CanUseSecureDnsTransactions() const { |
| 602 | const DnsConfig* config = GetEffectiveConfig(); |
| 603 | return config && config->IsValid() && !config->dns_over_https_servers.empty(); |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 604 | } |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 605 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 606 | bool MockDnsClient::CanUseInsecureDnsTransactions() const { |
| 607 | const DnsConfig* config = GetEffectiveConfig(); |
| 608 | return config && config->IsValid() && insecure_enabled_ && |
| 609 | !config->dns_over_tls_active; |
| 610 | } |
| 611 | |
| 612 | void MockDnsClient::SetInsecureEnabled(bool enabled) { |
| 613 | insecure_enabled_ = enabled; |
| 614 | } |
| 615 | |
Eric Orth | 069a3c6e | 2020-01-31 23:14:24 | [diff] [blame] | 616 | bool MockDnsClient::FallbackFromSecureTransactionPreferred( |
| 617 | ResolveContext* context) const { |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 618 | bool doh_server_available = |
| 619 | force_doh_server_available_ || |
| 620 | context->NumAvailableDohServers(session_.get()) > 0; |
| 621 | return !CanUseSecureDnsTransactions() || !doh_server_available; |
dalyk | c2adf18 | 2019-09-02 14:31:22 | [diff] [blame] | 622 | } |
| 623 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 624 | bool MockDnsClient::FallbackFromInsecureTransactionPreferred() const { |
| 625 | return !CanUseInsecureDnsTransactions() || |
| 626 | fallback_failures_ >= max_fallback_failures_; |
| 627 | } |
| 628 | |
| 629 | bool MockDnsClient::SetSystemConfig(base::Optional<DnsConfig> system_config) { |
| 630 | if (ignore_system_config_changes_) |
| 631 | return false; |
| 632 | |
| 633 | base::Optional<DnsConfig> before = effective_config_; |
| 634 | config_ = std::move(system_config); |
| 635 | effective_config_ = BuildEffectiveConfig(); |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 636 | session_ = BuildSession(); |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 637 | return before != effective_config_; |
| 638 | } |
| 639 | |
| 640 | bool MockDnsClient::SetConfigOverrides(DnsConfigOverrides config_overrides) { |
| 641 | base::Optional<DnsConfig> before = effective_config_; |
| 642 | overrides_ = std::move(config_overrides); |
| 643 | effective_config_ = BuildEffectiveConfig(); |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 644 | session_ = BuildSession(); |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 645 | return before != effective_config_; |
| 646 | } |
| 647 | |
Eric Orth | 020e5fe | 2020-03-12 17:43:43 | [diff] [blame] | 648 | void MockDnsClient::ReplaceCurrentSession() { |
| 649 | // Noop if no current effective config. |
| 650 | session_ = BuildSession(); |
| 651 | } |
| 652 | |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 653 | DnsSession* MockDnsClient::GetCurrentSession() { |
| 654 | return session_.get(); |
| 655 | } |
| 656 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 657 | const DnsConfig* MockDnsClient::GetEffectiveConfig() const { |
| 658 | return effective_config_.has_value() ? &effective_config_.value() : nullptr; |
| 659 | } |
| 660 | |
| 661 | const DnsHosts* MockDnsClient::GetHosts() const { |
| 662 | const DnsConfig* config = GetEffectiveConfig(); |
| 663 | if (!config) |
| 664 | return nullptr; |
| 665 | |
| 666 | return &config->hosts; |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 667 | } |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 668 | |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 669 | DnsTransactionFactory* MockDnsClient::GetTransactionFactory() { |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 670 | return GetEffectiveConfig() ? factory_.get() : nullptr; |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 671 | } |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 672 | |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 673 | AddressSorter* MockDnsClient::GetAddressSorter() { |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 674 | return GetEffectiveConfig() ? address_sorter_.get() : nullptr; |
| 675 | } |
| 676 | |
| 677 | void MockDnsClient::IncrementInsecureFallbackFailures() { |
| 678 | ++fallback_failures_; |
| 679 | } |
| 680 | |
| 681 | void MockDnsClient::ClearInsecureFallbackFailures() { |
| 682 | fallback_failures_ = 0; |
| 683 | } |
| 684 | |
| 685 | base::Optional<DnsConfig> MockDnsClient::GetSystemConfigForTesting() const { |
| 686 | return config_; |
| 687 | } |
| 688 | |
| 689 | DnsConfigOverrides MockDnsClient::GetConfigOverridesForTesting() const { |
| 690 | return overrides_; |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 691 | } |
[email protected] | 0adcb2b | 2012-08-15 21:30:46 | [diff] [blame] | 692 | |
Eric Orth | 3421db38 | 2019-10-20 02:35:52 | [diff] [blame] | 693 | void MockDnsClient::SetTransactionFactoryForTesting( |
| 694 | std::unique_ptr<DnsTransactionFactory> factory) { |
| 695 | NOTREACHED(); |
| 696 | } |
| 697 | |
[email protected] | daae132 | 2013-09-05 18:26:50 | [diff] [blame] | 698 | void MockDnsClient::CompleteDelayedTransactions() { |
| 699 | factory_->CompleteDelayedTransactions(); |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 700 | } |
| 701 | |
David Van Cleve | 1fb5e8c | 2019-11-04 16:45:16 | [diff] [blame] | 702 | bool MockDnsClient::CompleteOneDelayedTransactionOfType(DnsQueryType type) { |
| 703 | return factory_->CompleteOneDelayedTransactionOfType(type); |
| 704 | } |
| 705 | |
Eric Orth | 7dc18e9 | 2020-02-13 20:27:56 | [diff] [blame] | 706 | void MockDnsClient::SetForceDohServerAvailable(bool available) { |
| 707 | force_doh_server_available_ = available; |
| 708 | factory_->set_force_doh_server_available(available); |
| 709 | } |
| 710 | |
Eric Orth | 394db173 | 2019-08-27 20:09:39 | [diff] [blame] | 711 | base::Optional<DnsConfig> MockDnsClient::BuildEffectiveConfig() { |
| 712 | if (overrides_.OverridesEverything()) |
| 713 | return overrides_.ApplyOverrides(DnsConfig()); |
| 714 | if (!config_ || !config_.value().IsValid()) |
| 715 | return base::nullopt; |
| 716 | |
| 717 | return overrides_.ApplyOverrides(config_.value()); |
| 718 | } |
| 719 | |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 720 | scoped_refptr<DnsSession> MockDnsClient::BuildSession() { |
| 721 | if (!effective_config_) |
| 722 | return nullptr; |
| 723 | |
| 724 | // Session not expected to be used for anything that will actually require |
| 725 | // random numbers. |
| 726 | auto null_random_callback = |
| 727 | base::BindRepeating([](int, int) -> int { IMMEDIATE_CRASH(); }); |
| 728 | |
Eric Orth | 9300cc6 | 2020-08-21 00:29:34 | [diff] [blame^] | 729 | auto socket_allocator = std::make_unique<DnsSocketAllocator>( |
Eric Orth | 990a74a9 | 2020-08-20 18:41:26 | [diff] [blame] | 730 | &socket_factory_, effective_config_.value().nameservers, |
| 731 | nullptr /* net_log */); |
| 732 | |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 733 | return base::MakeRefCounted<DnsSession>( |
Eric Orth | 9300cc6 | 2020-08-21 00:29:34 | [diff] [blame^] | 734 | effective_config_.value(), std::move(socket_allocator), |
| 735 | null_random_callback, nullptr /* net_log */); |
Eric Orth | af82b49a | 2020-02-01 01:48:50 | [diff] [blame] | 736 | } |
| 737 | |
[email protected] | 78eac2a | 2012-03-14 19:09:27 | [diff] [blame] | 738 | } // namespace net |