address_list_unittest.cc revision 3345a6884c488ff3a535c2c9acdd33d74b37e311
1// Copyright (c) 2010 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/base/address_list.h"
6
7#include "base/string_util.h"
8#include "net/base/host_resolver_proc.h"
9#include "net/base/net_util.h"
10#include "net/base/sys_addrinfo.h"
11#if defined(OS_WIN)
12#include "net/base/winsock_init.h"
13#endif
14#include "testing/gtest/include/gtest/gtest.h"
15
16namespace {
17
18// Use getaddrinfo() to allocate an addrinfo structure.
19int CreateAddressList(const std::string& hostname, int port,
20                      net::AddressList* addrlist) {
21#if defined(OS_WIN)
22  net::EnsureWinsockInit();
23#endif
24  int rv = SystemHostResolverProc(hostname,
25                                  net::ADDRESS_FAMILY_UNSPECIFIED,
26                                  0,
27                                  addrlist, NULL);
28  if (rv == 0)
29    addrlist->SetPort(port);
30  return rv;
31}
32
33void CreateLongAddressList(net::AddressList* addrlist, int port) {
34  EXPECT_EQ(0, CreateAddressList("192.168.1.1", port, addrlist));
35  net::AddressList second_list;
36  EXPECT_EQ(0, CreateAddressList("192.168.1.2", port, &second_list));
37  addrlist->Append(second_list.head());
38}
39
40TEST(AddressListTest, GetPort) {
41  net::AddressList addrlist;
42  EXPECT_EQ(0, CreateAddressList("192.168.1.1", 81, &addrlist));
43  EXPECT_EQ(81, addrlist.GetPort());
44
45  addrlist.SetPort(83);
46  EXPECT_EQ(83, addrlist.GetPort());
47}
48
49TEST(AddressListTest, Assignment) {
50  net::AddressList addrlist1;
51  EXPECT_EQ(0, CreateAddressList("192.168.1.1", 85, &addrlist1));
52  EXPECT_EQ(85, addrlist1.GetPort());
53
54  // Should reference the same data as addrlist1 -- so when we change addrlist1
55  // both are changed.
56  net::AddressList addrlist2 = addrlist1;
57  EXPECT_EQ(85, addrlist2.GetPort());
58
59  addrlist1.SetPort(80);
60  EXPECT_EQ(80, addrlist1.GetPort());
61  EXPECT_EQ(80, addrlist2.GetPort());
62}
63
64TEST(AddressListTest, CopyRecursive) {
65  net::AddressList addrlist1;
66  CreateLongAddressList(&addrlist1, 85);
67  EXPECT_EQ(85, addrlist1.GetPort());
68
69  net::AddressList addrlist2;
70  addrlist2.Copy(addrlist1.head(), true);
71
72  ASSERT_TRUE(addrlist2.head()->ai_next != NULL);
73
74  // addrlist1 is the same as addrlist2 at this point.
75  EXPECT_EQ(85, addrlist1.GetPort());
76  EXPECT_EQ(85, addrlist2.GetPort());
77
78  // Changes to addrlist1 are not reflected in addrlist2.
79  addrlist1.SetPort(70);
80  addrlist2.SetPort(90);
81
82  EXPECT_EQ(70, addrlist1.GetPort());
83  EXPECT_EQ(90, addrlist2.GetPort());
84}
85
86TEST(AddressListTest, CopyNonRecursive) {
87  net::AddressList addrlist1;
88  CreateLongAddressList(&addrlist1, 85);
89  EXPECT_EQ(85, addrlist1.GetPort());
90
91  net::AddressList addrlist2;
92  addrlist2.Copy(addrlist1.head(), false);
93
94  ASSERT_TRUE(addrlist2.head()->ai_next == NULL);
95
96  // addrlist1 is the same as addrlist2 at this point.
97  EXPECT_EQ(85, addrlist1.GetPort());
98  EXPECT_EQ(85, addrlist2.GetPort());
99
100  // Changes to addrlist1 are not reflected in addrlist2.
101  addrlist1.SetPort(70);
102  addrlist2.SetPort(90);
103
104  EXPECT_EQ(70, addrlist1.GetPort());
105  EXPECT_EQ(90, addrlist2.GetPort());
106}
107
108TEST(AddressListTest, Append) {
109  net::AddressList addrlist1;
110  EXPECT_EQ(0, CreateAddressList("192.168.1.1", 11, &addrlist1));
111  EXPECT_EQ(11, addrlist1.GetPort());
112  net::AddressList addrlist2;
113  EXPECT_EQ(0, CreateAddressList("192.168.1.2", 12, &addrlist2));
114  EXPECT_EQ(12, addrlist2.GetPort());
115
116  ASSERT_TRUE(addrlist1.head()->ai_next == NULL);
117  addrlist1.Append(addrlist2.head());
118  ASSERT_TRUE(addrlist1.head()->ai_next != NULL);
119
120  net::AddressList addrlist3;
121  addrlist3.Copy(addrlist1.head()->ai_next, false);
122  EXPECT_EQ(12, addrlist3.GetPort());
123}
124
125static const char* kCanonicalHostname = "canonical.bar.com";
126
127TEST(AddressListTest, Canonical) {
128  // Create an addrinfo with a canonical name.
129  sockaddr_in address;
130  // The contents of address do not matter for this test,
131  // so just zero-ing them out for consistency.
132  memset(&address, 0x0, sizeof(address));
133  struct addrinfo ai;
134  memset(&ai, 0x0, sizeof(ai));
135  ai.ai_family = AF_INET;
136  ai.ai_socktype = SOCK_STREAM;
137  ai.ai_addrlen = sizeof(address);
138  ai.ai_addr = reinterpret_cast<sockaddr*>(&address);
139  ai.ai_canonname = const_cast<char *>(kCanonicalHostname);
140
141  // Copy the addrinfo struct into an AddressList object and
142  // make sure it seems correct.
143  net::AddressList addrlist1;
144  addrlist1.Copy(&ai, true);
145  const struct addrinfo* addrinfo1 = addrlist1.head();
146  EXPECT_TRUE(addrinfo1 != NULL);
147  EXPECT_TRUE(addrinfo1->ai_next == NULL);
148  std::string canon_name1;
149  EXPECT_TRUE(addrlist1.GetCanonicalName(&canon_name1));
150  EXPECT_EQ("canonical.bar.com", canon_name1);
151
152  // Copy the AddressList to another one.
153  net::AddressList addrlist2;
154  addrlist2.Copy(addrinfo1, true);
155  const struct addrinfo* addrinfo2 = addrlist2.head();
156  EXPECT_TRUE(addrinfo2 != NULL);
157  EXPECT_TRUE(addrinfo2->ai_next == NULL);
158  EXPECT_TRUE(addrinfo2->ai_canonname != NULL);
159  EXPECT_NE(addrinfo1, addrinfo2);
160  EXPECT_NE(addrinfo1->ai_canonname, addrinfo2->ai_canonname);
161  std::string canon_name2;
162  EXPECT_TRUE(addrlist2.GetCanonicalName(&canon_name2));
163  EXPECT_EQ("canonical.bar.com", canon_name2);
164
165  // Make sure that GetCanonicalName correctly returns false
166  // when ai_canonname is NULL.
167  ai.ai_canonname = NULL;
168  net::AddressList addrlist_no_canon;
169  addrlist_no_canon.Copy(&ai, true);
170  std::string canon_name3 = "blah";
171  EXPECT_FALSE(addrlist_no_canon.GetCanonicalName(&canon_name3));
172  EXPECT_EQ("blah", canon_name3);
173}
174
175TEST(AddressListTest, IPLiteralConstructor) {
176  struct TestData {
177    std::string ip_address;
178    std::string canonical_ip_address;
179    bool is_ipv6;
180  } tests[] = {
181    { "127.0.00.1", "127.0.0.1", false },
182    { "192.168.1.1", "192.168.1.1", false },
183    { "::1", "::1", true },
184    { "2001:db8:0::42", "2001:db8::42", true },
185  };
186  for (size_t i = 0; i < ARRAYSIZE_UNSAFE(tests); i++) {
187    net::AddressList expected_list;
188    int rv = CreateAddressList(tests[i].canonical_ip_address, 80,
189                               &expected_list);
190    if (tests[i].is_ipv6 && rv != 0) {
191      LOG(WARNING) << "Unable to resolve ip literal '" << tests[i].ip_address
192                   << "' test skipped.";
193      continue;
194    }
195    ASSERT_EQ(0, rv);
196    const struct addrinfo* good_ai = expected_list.head();
197
198    net::IPAddressNumber ip_number;
199    net::ParseIPLiteralToNumber(tests[i].ip_address, &ip_number);
200    net::AddressList test_list(ip_number, 80, true);
201    const struct addrinfo* test_ai = test_list.head();
202
203    EXPECT_EQ(good_ai->ai_family, test_ai->ai_family);
204    EXPECT_EQ(good_ai->ai_socktype, test_ai->ai_socktype);
205    EXPECT_EQ(good_ai->ai_addrlen, test_ai->ai_addrlen);
206    size_t sockaddr_size =
207        good_ai->ai_socktype == AF_INET ? sizeof(struct sockaddr_in) :
208        good_ai->ai_socktype == AF_INET6 ? sizeof(struct sockaddr_in6) : 0;
209    EXPECT_EQ(memcmp(good_ai->ai_addr, test_ai->ai_addr, sockaddr_size), 0);
210    EXPECT_EQ(good_ai->ai_next, test_ai->ai_next);
211    EXPECT_EQ(strcmp(tests[i].canonical_ip_address.c_str(),
212                     test_ai->ai_canonname), 0);
213  }
214}
215
216}  // namespace
217