// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/dns/dns_response.h" #include "base/time.h" #include "net/base/address_list.h" #include "net/base/io_buffer.h" #include "net/base/net_util.h" #include "net/dns/dns_protocol.h" #include "net/dns/dns_query.h" #include "net/dns/dns_test_util.h" #include "testing/gtest/include/gtest/gtest.h" namespace net { namespace { TEST(DnsRecordParserTest, Constructor) { const char data[] = { 0 }; EXPECT_FALSE(DnsRecordParser().IsValid()); EXPECT_TRUE(DnsRecordParser(data, 1, 0).IsValid()); EXPECT_TRUE(DnsRecordParser(data, 1, 1).IsValid()); EXPECT_FALSE(DnsRecordParser(data, 1, 0).AtEnd()); EXPECT_TRUE(DnsRecordParser(data, 1, 1).AtEnd()); } TEST(DnsRecordParserTest, ReadName) { const uint8 data[] = { // all labels "foo.example.com" 0x03, 'f', 'o', 'o', 0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', 0x03, 'c', 'o', 'm', // byte 0x10 0x00, // byte 0x11 // part label, part pointer, "bar.example.com" 0x03, 'b', 'a', 'r', 0xc0, 0x04, // byte 0x17 // all pointer to "bar.example.com", 2 jumps 0xc0, 0x11, // byte 0x1a }; std::string out; DnsRecordParser parser(data, sizeof(data), 0); ASSERT_TRUE(parser.IsValid()); EXPECT_EQ(0x11u, parser.ReadName(data + 0x00, &out)); EXPECT_EQ("foo.example.com", out); // Check that the last "." is never stored. out.clear(); EXPECT_EQ(0x1u, parser.ReadName(data + 0x10, &out)); EXPECT_EQ("", out); out.clear(); EXPECT_EQ(0x6u, parser.ReadName(data + 0x11, &out)); EXPECT_EQ("bar.example.com", out); out.clear(); EXPECT_EQ(0x2u, parser.ReadName(data + 0x17, &out)); EXPECT_EQ("bar.example.com", out); // Parse name without storing it. EXPECT_EQ(0x11u, parser.ReadName(data + 0x00, NULL)); EXPECT_EQ(0x1u, parser.ReadName(data + 0x10, NULL)); EXPECT_EQ(0x6u, parser.ReadName(data + 0x11, NULL)); EXPECT_EQ(0x2u, parser.ReadName(data + 0x17, NULL)); // Check that it works even if initial position is different. parser = DnsRecordParser(data, sizeof(data), 0x12); EXPECT_EQ(0x6u, parser.ReadName(data + 0x11, NULL)); } TEST(DnsRecordParserTest, ReadNameFail) { const uint8 data[] = { // label length beyond packet 0x30, 'x', 'x', 0x00, // pointer offset beyond packet 0xc0, 0x20, // pointer loop 0xc0, 0x08, 0xc0, 0x06, // incorrect label type (currently supports only direct and pointer) 0x80, 0x00, // truncated name (missing root label) 0x02, 'x', 'x', }; DnsRecordParser parser(data, sizeof(data), 0); ASSERT_TRUE(parser.IsValid()); std::string out; EXPECT_EQ(0u, parser.ReadName(data + 0x00, &out)); EXPECT_EQ(0u, parser.ReadName(data + 0x04, &out)); EXPECT_EQ(0u, parser.ReadName(data + 0x08, &out)); EXPECT_EQ(0u, parser.ReadName(data + 0x0a, &out)); EXPECT_EQ(0u, parser.ReadName(data + 0x0c, &out)); EXPECT_EQ(0u, parser.ReadName(data + 0x0e, &out)); } TEST(DnsRecordParserTest, ReadRecord) { const uint8 data[] = { // Type CNAME record. 0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', 0x03, 'c', 'o', 'm', 0x00, 0x00, 0x05, // TYPE is CNAME. 0x00, 0x01, // CLASS is IN. 0x00, 0x01, 0x24, 0x74, // TTL is 0x00012474. 0x00, 0x06, // RDLENGTH is 6 bytes. 0x03, 'f', 'o', 'o', // compressed name in record 0xc0, 0x00, // Type A record. 0x03, 'b', 'a', 'r', // compressed owner name 0xc0, 0x00, 0x00, 0x01, // TYPE is A. 0x00, 0x01, // CLASS is IN. 0x00, 0x20, 0x13, 0x55, // TTL is 0x00201355. 0x00, 0x04, // RDLENGTH is 4 bytes. 0x7f, 0x02, 0x04, 0x01, // IP is 127.2.4.1 }; std::string out; DnsRecordParser parser(data, sizeof(data), 0); DnsResourceRecord record; EXPECT_TRUE(parser.ReadRecord(&record)); EXPECT_EQ("example.com", record.name); EXPECT_EQ(dns_protocol::kTypeCNAME, record.type); EXPECT_EQ(dns_protocol::kClassIN, record.klass); EXPECT_EQ(0x00012474u, record.ttl); EXPECT_EQ(6u, record.rdata.length()); EXPECT_EQ(6u, parser.ReadName(record.rdata.data(), &out)); EXPECT_EQ("foo.example.com", out); EXPECT_FALSE(parser.AtEnd()); EXPECT_TRUE(parser.ReadRecord(&record)); EXPECT_EQ("bar.example.com", record.name); EXPECT_EQ(dns_protocol::kTypeA, record.type); EXPECT_EQ(dns_protocol::kClassIN, record.klass); EXPECT_EQ(0x00201355u, record.ttl); EXPECT_EQ(4u, record.rdata.length()); EXPECT_EQ(base::StringPiece("\x7f\x02\x04\x01"), record.rdata); EXPECT_TRUE(parser.AtEnd()); // Test truncated record. parser = DnsRecordParser(data, sizeof(data) - 2, 0); EXPECT_TRUE(parser.ReadRecord(&record)); EXPECT_FALSE(parser.AtEnd()); EXPECT_FALSE(parser.ReadRecord(&record)); } TEST(DnsResponseTest, InitParse) { // This includes \0 at the end. const char qname_data[] = "\x0A""codereview""\x08""chromium""\x03""org"; const base::StringPiece qname(qname_data, sizeof(qname_data)); // Compilers want to copy when binding temporary to const &, so must use heap. scoped_ptr query(new DnsQuery(0xcafe, qname, dns_protocol::kTypeA)); const uint8 response_data[] = { // Header 0xca, 0xfe, // ID 0x81, 0x80, // Standard query response, RA, no error 0x00, 0x01, // 1 question 0x00, 0x02, // 2 RRs (answers) 0x00, 0x00, // 0 authority RRs 0x00, 0x00, // 0 additional RRs // Question // This part is echoed back from the respective query. 0x0a, 'c', 'o', 'd', 'e', 'r', 'e', 'v', 'i', 'e', 'w', 0x08, 'c', 'h', 'r', 'o', 'm', 'i', 'u', 'm', 0x03, 'o', 'r', 'g', 0x00, 0x00, 0x01, // TYPE is A. 0x00, 0x01, // CLASS is IN. // Answer 1 0xc0, 0x0c, // NAME is a pointer to name in Question section. 0x00, 0x05, // TYPE is CNAME. 0x00, 0x01, // CLASS is IN. 0x00, 0x01, // TTL (4 bytes) is 20 hours, 47 minutes, 48 seconds. 0x24, 0x74, 0x00, 0x12, // RDLENGTH is 18 bytes. // ghs.l.google.com in DNS format. 0x03, 'g', 'h', 's', 0x01, 'l', 0x06, 'g', 'o', 'o', 'g', 'l', 'e', 0x03, 'c', 'o', 'm', 0x00, // Answer 2 0xc0, 0x35, // NAME is a pointer to name in Answer 1. 0x00, 0x01, // TYPE is A. 0x00, 0x01, // CLASS is IN. 0x00, 0x00, // TTL (4 bytes) is 53 seconds. 0x00, 0x35, 0x00, 0x04, // RDLENGTH is 4 bytes. 0x4a, 0x7d, // RDATA is the IP: 74.125.95.121 0x5f, 0x79, }; DnsResponse resp; memcpy(resp.io_buffer()->data(), response_data, sizeof(response_data)); // Reject too short. EXPECT_FALSE(resp.InitParse(query->io_buffer()->size() - 1, *query)); EXPECT_FALSE(resp.IsValid()); // Reject wrong id. scoped_ptr other_query(query->CloneWithNewId(0xbeef)); EXPECT_FALSE(resp.InitParse(sizeof(response_data), *other_query)); EXPECT_FALSE(resp.IsValid()); // Reject wrong question. scoped_ptr wrong_query( new DnsQuery(0xcafe, qname, dns_protocol::kTypeCNAME)); EXPECT_FALSE(resp.InitParse(sizeof(response_data), *wrong_query)); EXPECT_FALSE(resp.IsValid()); // Accept matching question. EXPECT_TRUE(resp.InitParse(sizeof(response_data), *query)); EXPECT_TRUE(resp.IsValid()); // Check header access. EXPECT_EQ(0x8180, resp.flags()); EXPECT_EQ(0x0, resp.rcode()); EXPECT_EQ(2u, resp.answer_count()); // Check question access. EXPECT_EQ(query->qname(), resp.qname()); EXPECT_EQ(query->qtype(), resp.qtype()); EXPECT_EQ("codereview.chromium.org", resp.GetDottedName()); DnsResourceRecord record; DnsRecordParser parser = resp.Parser(); EXPECT_TRUE(parser.ReadRecord(&record)); EXPECT_FALSE(parser.AtEnd()); EXPECT_TRUE(parser.ReadRecord(&record)); EXPECT_TRUE(parser.AtEnd()); EXPECT_FALSE(parser.ReadRecord(&record)); } void VerifyAddressList(const std::vector& ip_addresses, const AddressList& addrlist) { ASSERT_EQ(ip_addresses.size(), addrlist.size()); for (size_t i = 0; i < addrlist.size(); ++i) { EXPECT_EQ(ip_addresses[i], addrlist[i].ToStringWithoutPort()); } } TEST(DnsResponseTest, ParseToAddressList) { const struct TestCase { size_t query_size; const uint8* response_data; size_t response_size; const char* const* expected_addresses; size_t num_expected_addresses; const char* expected_cname; int expected_ttl_sec; } cases[] = { { kT0QuerySize, kT0ResponseDatagram, arraysize(kT0ResponseDatagram), kT0IpAddresses, arraysize(kT0IpAddresses), kT0CanonName, kT0TTL, }, { kT1QuerySize, kT1ResponseDatagram, arraysize(kT1ResponseDatagram), kT1IpAddresses, arraysize(kT1IpAddresses), kT1CanonName, kT1TTL, }, { kT2QuerySize, kT2ResponseDatagram, arraysize(kT2ResponseDatagram), kT2IpAddresses, arraysize(kT2IpAddresses), kT2CanonName, kT2TTL, }, { kT3QuerySize, kT3ResponseDatagram, arraysize(kT3ResponseDatagram), kT3IpAddresses, arraysize(kT3IpAddresses), kT3CanonName, kT3TTL, }, }; for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) { const TestCase& t = cases[i]; DnsResponse response(t.response_data, t.response_size, t.query_size); AddressList addr_list; base::TimeDelta ttl; EXPECT_EQ(DnsResponse::DNS_SUCCESS, response.ParseToAddressList(&addr_list, &ttl)); std::vector expected_addresses( t.expected_addresses, t.expected_addresses + t.num_expected_addresses); VerifyAddressList(expected_addresses, addr_list); EXPECT_EQ(t.expected_cname, addr_list.canonical_name()); EXPECT_EQ(base::TimeDelta::FromSeconds(t.expected_ttl_sec), ttl); } } const uint8 kResponseTruncatedRecord[] = { // Header: 1 question, 1 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = A, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, // Truncated RDATA. }; const uint8 kResponseTruncatedCNAME[] = { // Header: 1 question, 1 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = CNAME, TTL = 0xFF, RDATA = 'foo' (truncated) 0x01, 'a', 0x00, 0x00, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x03, 0x03, 'f', 'o', // Truncated name. }; const uint8 kResponseNameMismatch[] = { // Header: 1 question, 1 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'b', type = A, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'b', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, }; const uint8 kResponseNameMismatchInChain[] = { // Header: 1 question, 3 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x03, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = CNAME, TTL = 0xFF, RDATA = 'b' 0x01, 'a', 0x00, 0x00, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x03, 0x01, 'b', 0x00, // Answer: name = 'b', type = A, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'b', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, // Answer: name = 'c', type = A, TTL = 0xFF, RDATA = 10.10.10.11 0x01, 'c', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0B, }; const uint8 kResponseSizeMismatch[] = { // Header: 1 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = AAAA (0x1c) 0x01, 'a', 0x00, 0x00, 0x1c, 0x00, 0x01, // Answer: name = 'a', type = AAAA, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'a', 0x00, 0x00, 0x1c, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, }; const uint8 kResponseCNAMEAfterAddress[] = { // Header: 2 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x02, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = A, TTL = 0xFF, RDATA = 10.10.10.10. 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, // Answer: name = 'a', type = CNAME, TTL = 0xFF, RDATA = 'b' 0x01, 'a', 0x00, 0x00, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x03, 0x01, 'b', 0x00, }; const uint8 kResponseTTLMismatch[] = { // Header: 1 question, 3 answer RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x03, 0x00, 0x00, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = CNAME, TTL = 0xFF, RDATA = 'b' 0x01, 'a', 0x00, 0x00, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x03, 0x01, 'b', 0x00, // Answer: name = 'b', type = A, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'b', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, // Answer: name = 'b', type = A, TTL = 0xBB, RDATA = 10.10.10.11 0x01, 'b', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xBB, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0B, }; const uint8 kResponseNoAddresses[] = { // Header: 1 question, 1 answer RR, 1 authority RR 0x00, 0x00, 0x81, 0x80, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, // Question: name = 'a', type = A (0x1) 0x01, 'a', 0x00, 0x00, 0x01, 0x00, 0x01, // Answer: name = 'a', type = CNAME, TTL = 0xFF, RDATA = 'b' 0x01, 'a', 0x00, 0x00, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x03, 0x01, 'b', 0x00, // Authority section // Answer: name = 'b', type = A, TTL = 0xFF, RDATA = 10.10.10.10 0x01, 'b', 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x04, 0x0A, 0x0A, 0x0A, 0x0A, }; TEST(DnsResponseTest, ParseToAddressListFail) { const struct TestCase { const uint8* data; size_t size; DnsResponse::Result expected_result; } cases[] = { { kResponseTruncatedRecord, arraysize(kResponseTruncatedRecord), DnsResponse::DNS_MALFORMED_RESPONSE }, { kResponseTruncatedCNAME, arraysize(kResponseTruncatedCNAME), DnsResponse::DNS_MALFORMED_CNAME }, { kResponseNameMismatch, arraysize(kResponseNameMismatch), DnsResponse::DNS_NAME_MISMATCH }, { kResponseNameMismatchInChain, arraysize(kResponseNameMismatchInChain), DnsResponse::DNS_NAME_MISMATCH }, { kResponseSizeMismatch, arraysize(kResponseSizeMismatch), DnsResponse::DNS_SIZE_MISMATCH }, { kResponseCNAMEAfterAddress, arraysize(kResponseCNAMEAfterAddress), DnsResponse::DNS_CNAME_AFTER_ADDRESS }, { kResponseTTLMismatch, arraysize(kResponseTTLMismatch), DnsResponse::DNS_ADDRESS_TTL_MISMATCH }, { kResponseNoAddresses, arraysize(kResponseNoAddresses), DnsResponse::DNS_NO_ADDRESSES }, }; const size_t kQuerySize = 12 + 7; for (size_t i = 0; i < ARRAYSIZE_UNSAFE(cases); ++i) { const TestCase& t = cases[i]; DnsResponse response(t.data, t.size, kQuerySize); AddressList addr_list; base::TimeDelta ttl; EXPECT_EQ(t.expected_result, response.ParseToAddressList(&addr_list, &ttl)); } } } // namespace } // namespace net