blob: 237646bcbc9552c2a0f74ab65674e36e53f11643 [file] [log] [blame]
[email protected]78eac2a2012-03-14 19:09:271// 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
7#include <string>
8
9#include "base/bind.h"
10#include "base/memory/weak_ptr.h"
11#include "base/message_loop.h"
12#include "base/sys_byteorder.h"
13#include "net/base/big_endian.h"
14#include "net/base/dns_util.h"
15#include "net/base/io_buffer.h"
16#include "net/base/net_errors.h"
[email protected]0adcb2b2012-08-15 21:30:4617#include "net/dns/address_sorter.h"
[email protected]78eac2a2012-03-14 19:09:2718#include "net/dns/dns_client.h"
19#include "net/dns/dns_config_service.h"
20#include "net/dns/dns_protocol.h"
21#include "net/dns/dns_query.h"
22#include "net/dns/dns_response.h"
23#include "net/dns/dns_transaction.h"
24#include "testing/gtest/include/gtest/gtest.h"
25
26namespace net {
27namespace {
28
[email protected]0adcb2b2012-08-15 21:30:4629// A DnsTransaction which uses MockDnsClientRuleList to determine the response.
[email protected]78eac2a2012-03-14 19:09:2730class MockTransaction : public DnsTransaction,
31 public base::SupportsWeakPtr<MockTransaction> {
32 public:
[email protected]0adcb2b2012-08-15 21:30:4633 MockTransaction(const MockDnsClientRuleList& rules,
34 const std::string& hostname,
[email protected]78eac2a2012-03-14 19:09:2735 uint16 qtype,
36 const DnsTransactionFactory::CallbackType& callback)
[email protected]0adcb2b2012-08-15 21:30:4637 : result_(MockDnsClientRule::FAIL_SYNC),
38 hostname_(hostname),
[email protected]78eac2a2012-03-14 19:09:2739 qtype_(qtype),
40 callback_(callback),
41 started_(false) {
[email protected]0adcb2b2012-08-15 21:30:4642 // Find the relevant rule which matches |qtype| and prefix of |hostname|.
43 for (size_t i = 0; i < rules.size(); ++i) {
44 const std::string& prefix = rules[i].prefix;
45 if ((rules[i].qtype == qtype) &&
46 (hostname.size() >= prefix.size()) &&
47 (hostname.compare(0, prefix.size(), prefix) == 0)) {
48 result_ = rules[i].result;
49 break;
50 }
51 }
[email protected]78eac2a2012-03-14 19:09:2752 }
53
54 virtual const std::string& GetHostname() const OVERRIDE {
55 return hostname_;
56 }
57
58 virtual uint16 GetType() const OVERRIDE {
59 return qtype_;
60 }
61
62 virtual int Start() OVERRIDE {
63 EXPECT_FALSE(started_);
64 started_ = true;
[email protected]0adcb2b2012-08-15 21:30:4665 if (MockDnsClientRule::FAIL_SYNC == result_)
[email protected]78eac2a2012-03-14 19:09:2766 return ERR_NAME_NOT_RESOLVED;
67 // Using WeakPtr to cleanly cancel when transaction is destroyed.
68 MessageLoop::current()->PostTask(
69 FROM_HERE,
70 base::Bind(&MockTransaction::Finish, AsWeakPtr()));
71 return ERR_IO_PENDING;
72 }
73
74 private:
75 void Finish() {
[email protected]0adcb2b2012-08-15 21:30:4676 switch (result_) {
77 case MockDnsClientRule::EMPTY:
78 case MockDnsClientRule::OK: {
79 std::string qname;
80 DNSDomainFromDot(hostname_, &qname);
81 DnsQuery query(0, qname, qtype_);
[email protected]78eac2a2012-03-14 19:09:2782
[email protected]0adcb2b2012-08-15 21:30:4683 DnsResponse response;
84 char* buffer = response.io_buffer()->data();
85 int nbytes = query.io_buffer()->size();
86 memcpy(buffer, query.io_buffer()->data(), nbytes);
87 dns_protocol::Header* header =
88 reinterpret_cast<dns_protocol::Header*>(buffer);
89 header->flags |= dns_protocol::kFlagResponse;
[email protected]78eac2a2012-03-14 19:09:2790
[email protected]0adcb2b2012-08-15 21:30:4691 if (MockDnsClientRule::OK == result_) {
92 const uint16 kPointerToQueryName =
93 static_cast<uint16>(0xc000 | sizeof(*header));
[email protected]78eac2a2012-03-14 19:09:2794
[email protected]0adcb2b2012-08-15 21:30:4695 const uint32 kTTL = 86400; // One day.
[email protected]78eac2a2012-03-14 19:09:2796
[email protected]0adcb2b2012-08-15 21:30:4697 // Size of RDATA which is a IPv4 or IPv6 address.
98 size_t rdata_size = qtype_ == net::dns_protocol::kTypeA ?
99 net::kIPv4AddressSize : net::kIPv6AddressSize;
[email protected]78eac2a2012-03-14 19:09:27100
[email protected]0adcb2b2012-08-15 21:30:46101 // 12 is the sum of sizes of the compressed name reference, TYPE,
102 // CLASS, TTL and RDLENGTH.
103 size_t answer_size = 12 + rdata_size;
[email protected]78eac2a2012-03-14 19:09:27104
[email protected]0adcb2b2012-08-15 21:30:46105 // Write answer with loopback IP address.
106 header->ancount = base::HostToNet16(1);
107 BigEndianWriter writer(buffer + nbytes, answer_size);
108 writer.WriteU16(kPointerToQueryName);
109 writer.WriteU16(qtype_);
110 writer.WriteU16(net::dns_protocol::kClassIN);
111 writer.WriteU32(kTTL);
112 writer.WriteU16(rdata_size);
113 if (qtype_ == net::dns_protocol::kTypeA) {
114 char kIPv4Loopback[] = { 0x7f, 0, 0, 1 };
115 writer.WriteBytes(kIPv4Loopback, sizeof(kIPv4Loopback));
116 } else {
117 char kIPv6Loopback[] = { 0, 0, 0, 0, 0, 0, 0, 0,
118 0, 0, 0, 0, 0, 0, 0, 1 };
119 writer.WriteBytes(kIPv6Loopback, sizeof(kIPv6Loopback));
120 }
121 nbytes += answer_size;
122 }
123 EXPECT_TRUE(response.InitParse(nbytes, query));
124 callback_.Run(this, OK, &response);
125 } break;
126 case MockDnsClientRule::FAIL_ASYNC:
127 callback_.Run(this, ERR_NAME_NOT_RESOLVED, NULL);
128 break;
[email protected]4817d122012-12-13 02:21:59129 case MockDnsClientRule::TIMEOUT:
130 callback_.Run(this, ERR_DNS_TIMED_OUT, NULL);
131 break;
[email protected]0adcb2b2012-08-15 21:30:46132 default:
133 NOTREACHED();
134 break;
[email protected]78eac2a2012-03-14 19:09:27135 }
136 }
137
[email protected]0adcb2b2012-08-15 21:30:46138 MockDnsClientRule::Result result_;
[email protected]78eac2a2012-03-14 19:09:27139 const std::string hostname_;
140 const uint16 qtype_;
141 DnsTransactionFactory::CallbackType callback_;
142 bool started_;
143};
144
145
146// A DnsTransactionFactory which creates MockTransaction.
147class MockTransactionFactory : public DnsTransactionFactory {
148 public:
[email protected]0adcb2b2012-08-15 21:30:46149 explicit MockTransactionFactory(const MockDnsClientRuleList& rules)
150 : rules_(rules) {}
[email protected]78eac2a2012-03-14 19:09:27151 virtual ~MockTransactionFactory() {}
152
153 virtual scoped_ptr<DnsTransaction> CreateTransaction(
154 const std::string& hostname,
155 uint16 qtype,
156 const DnsTransactionFactory::CallbackType& callback,
157 const BoundNetLog&) OVERRIDE {
158 return scoped_ptr<DnsTransaction>(
[email protected]0adcb2b2012-08-15 21:30:46159 new MockTransaction(rules_, hostname, qtype, callback));
160 }
161
162 private:
163 MockDnsClientRuleList rules_;
164};
165
166class MockAddressSorter : public AddressSorter {
167 public:
168 virtual ~MockAddressSorter() {}
169 virtual void Sort(const AddressList& list,
170 const CallbackType& callback) const OVERRIDE {
171 // Do nothing.
172 callback.Run(true, list);
[email protected]78eac2a2012-03-14 19:09:27173 }
174};
175
176// MockDnsClient provides MockTransactionFactory.
177class MockDnsClient : public DnsClient {
178 public:
[email protected]0adcb2b2012-08-15 21:30:46179 MockDnsClient(const DnsConfig& config,
180 const MockDnsClientRuleList& rules)
181 : config_(config), factory_(rules) {}
[email protected]78eac2a2012-03-14 19:09:27182 virtual ~MockDnsClient() {}
183
184 virtual void SetConfig(const DnsConfig& config) OVERRIDE {
185 config_ = config;
186 }
187
188 virtual const DnsConfig* GetConfig() const OVERRIDE {
189 return config_.IsValid() ? &config_ : NULL;
190 }
191
192 virtual DnsTransactionFactory* GetTransactionFactory() OVERRIDE {
193 return config_.IsValid() ? &factory_ : NULL;
194 }
195
[email protected]0adcb2b2012-08-15 21:30:46196 virtual AddressSorter* GetAddressSorter() OVERRIDE {
197 return &address_sorter_;
198 }
199
[email protected]78eac2a2012-03-14 19:09:27200 private:
201 DnsConfig config_;
202 MockTransactionFactory factory_;
[email protected]0adcb2b2012-08-15 21:30:46203 MockAddressSorter address_sorter_;
[email protected]78eac2a2012-03-14 19:09:27204};
205
206} // namespace
207
208// static
[email protected]0adcb2b2012-08-15 21:30:46209scoped_ptr<DnsClient> CreateMockDnsClient(const DnsConfig& config,
210 const MockDnsClientRuleList& rules) {
211 return scoped_ptr<DnsClient>(new MockDnsClient(config, rules));
[email protected]78eac2a2012-03-14 19:09:27212}
213
[email protected]78eac2a2012-03-14 19:09:27214} // namespace net