summaryrefslogtreecommitdiffstats
path: root/net/base/net_util_unittest.cc
diff options
context:
space:
mode:
Diffstat (limited to 'net/base/net_util_unittest.cc')
-rw-r--r--net/base/net_util_unittest.cc671
1 files changed, 671 insertions, 0 deletions
diff --git a/net/base/net_util_unittest.cc b/net/base/net_util_unittest.cc
new file mode 100644
index 0000000..c0b1f7c
--- /dev/null
+++ b/net/base/net_util_unittest.cc
@@ -0,0 +1,671 @@
+// Copyright 2008, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#include "base/basictypes.h"
+#include "base/logging.h"
+#include "base/string_util.h"
+#include "googleurl/src/gurl.h"
+#include "net/base/net_util.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+namespace {
+ class NetUtilTest : public testing::Test {
+ };
+}
+
+TEST(NetUtilTest, FileURLConversion) {
+ // a list of test file names and the corresponding URLs
+ const struct FileCase {
+ const wchar_t* file;
+ const wchar_t* url;
+ } round_trip_cases[] = {
+ {L"C:\\foo\\bar.txt", L"file:///C:/foo/bar.txt"},
+ {L"\\\\some computer\\foo\\bar.txt", L"file://some%20computer/foo/bar.txt"}, // UNC
+ {L"D:\\Name;with%some symbols*#", L"file:///D:/Name%3Bwith%25some%20symbols*%23"},
+ {L"D:\\Chinese\\\x6240\x6709\x4e2d\x6587\x7f51\x9875.doc", L"file:///D:/Chinese/%E6%89%80%E6%9C%89%E4%B8%AD%E6%96%87%E7%BD%91%E9%A1%B5.doc"},
+ };
+
+ // First, we'll test that we can round-trip all of the above cases of URLs
+ std::wstring output;
+ for (int i = 0; i < arraysize(round_trip_cases); i++) {
+ // convert to the file URL
+ GURL file_url(net_util::FilePathToFileURL(round_trip_cases[i].file));
+ EXPECT_EQ(std::wstring(round_trip_cases[i].url),
+ UTF8ToWide(file_url.spec()));
+
+ // Back to the filename.
+ EXPECT_TRUE(net_util::FileURLToFilePath(file_url, &output));
+ EXPECT_EQ(std::wstring(round_trip_cases[i].file), output);
+ }
+
+ // Test that various file: URLs get decoded into the correct file type
+ FileCase url_cases[] = {
+ {L"C:\\foo\\bar.txt", L"file:c|/foo\\bar.txt"},
+ {L"C:\\foo\\bar.txt", L"file:/c:/foo/bar.txt"},
+ {L"\\\\foo\\bar.txt", L"file://foo\\bar.txt"},
+ {L"C:\\foo\\bar.txt", L"file:///c:/foo/bar.txt"},
+ {L"\\\\foo\\bar.txt", L"file:////foo\\bar.txt"},
+ {L"\\\\foo\\bar.txt", L"file:/foo/bar.txt"},
+ {L"\\\\foo\\bar.txt", L"file://foo\\bar.txt"},
+ {L"C:\\foo\\bar.txt", L"file:\\\\\\c:/foo/bar.txt"},
+ };
+ for (int i = 0; i < arraysize(url_cases); i++) {
+ net_util::FileURLToFilePath(GURL(url_cases[i].url), &output);
+ EXPECT_EQ(std::wstring(url_cases[i].file), output);
+ }
+
+ // Here, we test that UTF-8 encoded strings get decoded properly, even when
+ // they might be stored with wide characters
+ const wchar_t utf8[] = L"file:///d:/Chinese/\xe6\x89\x80\xe6\x9c\x89\xe4\xb8\xad\xe6\x96\x87\xe7\xbd\x91\xe9\xa1\xb5.doc";
+ const wchar_t wide[] = L"D:\\Chinese\\\x6240\x6709\x4e2d\x6587\x7f51\x9875.doc";
+ EXPECT_TRUE(net_util::FileURLToFilePath(GURL(utf8), &output));
+ EXPECT_EQ(std::wstring(wide), output);
+
+ // Unfortunately, UTF8ToWide discards invalid UTF8 input.
+#ifdef BUG_878908_IS_FIXED
+ // Test that no conversion happens if the UTF-8 input is invalid, and that
+ // the input is preserved in UTF-8
+ const char invalid_utf8[] = "file:///d:/Blah/\xff.doc";
+ const wchar_t invalid_wide[] = L"D:\\Blah\\\xff.doc";
+ EXPECT_TRUE(net_util::FileURLToFilePath(
+ GURL(std::string(invalid_utf8)), &output));
+ EXPECT_EQ(std::wstring(invalid_wide), output);
+#endif
+
+ // Test that if a file URL is malformed, we get a failure
+ EXPECT_FALSE(net_util::FileURLToFilePath(GURL("filefoobar"), &output));
+}
+
+// Just a bunch of fake headers.
+const wchar_t* google_headers =
+ L"HTTP/1.1 200 OK\n"
+ L"Content-TYPE: text/html; charset=utf-8\n"
+ L"Content-disposition: attachment; filename=\"download.pdf\"\n"
+ L"Content-Length: 378557\n"
+ L"X-Google-Google1: 314159265\n"
+ L"X-Google-Google2: aaaa2:7783,bbb21:9441\n"
+ L"X-Google-Google4: home\n"
+ L"Transfer-Encoding: chunked\n"
+ L"Set-Cookie: HEHE_AT=6666x66beef666x6-66xx6666x66; Path=/mail\n"
+ L"Set-Cookie: HEHE_HELP=owned:0;Path=/\n"
+ L"Set-Cookie: S=gmail=Xxx-beefbeefbeef_beefb:gmail_yj=beefbeef000beefbeefbee:gmproxy=bee-fbeefbe; Domain=.google.com; Path=/\n"
+ L"X-Google-Google2: /one/two/three/four/five/six/seven-height/nine:9411\n"
+ L"Server: GFE/1.3\n"
+ L"Transfer-Encoding: chunked\n"
+ L"Date: Mon, 13 Nov 2006 21:38:09 GMT\n"
+ L"Expires: Tue, 14 Nov 2006 19:23:58 GMT\n"
+ L"X-Malformed: bla; arg=test\"\n"
+ L"X-Malformed2: bla; arg=\n"
+ L"X-Test: bla; arg1=val1; arg2=val2";
+
+TEST(NetUtilTest, GetSpecificHeader) {
+ const struct {
+ const wchar_t* header_name;
+ const wchar_t* expected;
+ } tests[] = {
+ {L"content-type", L"text/html; charset=utf-8"},
+ {L"CONTENT-LENGTH", L"378557"},
+ {L"Date", L"Mon, 13 Nov 2006 21:38:09 GMT"},
+ {L"Bad-Header", L""},
+ {L"", L""},
+ };
+
+ // Test first with google_headers.
+ for (size_t i = 0; i < arraysize(tests); ++i) {
+ std::wstring result = net_util::GetSpecificHeader(google_headers,
+ tests[i].header_name);
+ EXPECT_EQ(result, tests[i].expected);
+ }
+
+ // Test again with empty headers.
+ for (size_t i = 0; i < arraysize(tests); ++i) {
+ std::wstring result = net_util::GetSpecificHeader(L"", tests[i].header_name);
+ EXPECT_EQ(result, std::wstring());
+ }
+}
+
+TEST(NetUtilTest, GetHeaderParamValue) {
+ const struct {
+ const wchar_t* header_name;
+ const wchar_t* param_name;
+ const wchar_t* expected;
+ } tests[] = {
+ {L"Content-type", L"charset", L"utf-8"},
+ {L"content-disposition", L"filename", L"download.pdf"},
+ {L"Content-Type", L"badparam", L""},
+ {L"X-Malformed", L"arg", L"test\""},
+ {L"X-Malformed2", L"arg", L""},
+ {L"X-Test", L"arg1", L"val1"},
+ {L"X-Test", L"arg2", L"val2"},
+ {L"Bad-Header", L"badparam", L""},
+ {L"Bad-Header", L"", L""},
+ {L"", L"badparam", L""},
+ {L"", L"", L""},
+ };
+ // TODO(mpcomplete): add tests for other formats of headers.
+
+ for (size_t i = 0; i < arraysize(tests); ++i) {
+ std::wstring header_value = net_util::GetSpecificHeader(google_headers,
+ tests[i].header_name);
+ std::wstring result = net_util::GetHeaderParamValue(header_value,
+ tests[i].param_name);
+ EXPECT_EQ(result, tests[i].expected);
+ }
+
+ for (size_t i = 0; i < arraysize(tests); ++i) {
+ std::wstring header_value = net_util::GetSpecificHeader(L"",
+ tests[i].header_name);
+ std::wstring result = net_util::GetHeaderParamValue(header_value,
+ tests[i].param_name);
+ EXPECT_EQ(result, std::wstring());
+ }
+}
+
+TEST(NetUtilTest, GetFileNameFromCD) {
+ const struct {
+ const char* header_field;
+ const wchar_t* expected;
+ } tests[] = {
+ // Test various forms of C-D header fields emitted by web servers.
+ {"content-disposition: inline; filename=\"abcde.pdf\"", L"abcde.pdf"},
+ {"content-disposition: inline; name=\"abcde.pdf\"", L"abcde.pdf"},
+ {"content-disposition: attachment; filename=abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: attachment; name=abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: attachment; filename=abc,de.pdf", L"abc,de.pdf"},
+ {"content-disposition: filename=abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: filename= abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: filename =abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: filename = abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: filename\t=abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: filename \t\t =abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: name=abcde.pdf", L"abcde.pdf"},
+ {"content-disposition: inline; filename=\"abc%20de.pdf\"", L"abc de.pdf"},
+ // Whitespaces are converted to a space.
+ {"content-disposition: inline; filename=\"abc \t\nde.pdf\"", L"abc de.pdf"},
+ // %-escaped UTF-8
+ {"Content-Disposition: attachment; filename=\"%EC%98%88%EC%88%A0%20"
+ "%EC%98%88%EC%88%A0.jpg\"", L"\xc608\xc220 \xc608\xc220.jpg"},
+ {"Content-Disposition: attachment; filename=\"%F0%90%8C%B0%F0%90%8C%B1"
+ "abc.jpg\"", L"\U00010330\U00010331abc.jpg"},
+ {"Content-Disposition: attachment; filename=\"%EC%98%88%EC%88%A0 \n"
+ "%EC%98%88%EC%88%A0.jpg\"", L"\xc608\xc220 \xc608\xc220.jpg"},
+ // RFC 2047 with various charsets and Q/B encodings
+ {"Content-Disposition: attachment; filename=\"=?EUC-JP?Q?=B7=DD=BD="
+ "D13=2Epng?=\"", L"\x82b8\x8853" L"3.png"},
+ {"Content-Disposition: attachment; filename==?eUc-Kr?b?v7m8+iAzLnBuZw==?=",
+ L"\xc608\xc220 3.png"},
+ {"Content-Disposition: attachment; filename==?utf-8?Q?=E8=8A=B8=E8"
+ "=A1=93_3=2Epng?=", L"\x82b8\x8853 3.png"},
+ {"Content-Disposition: attachment; filename==?utf-8?Q?=F0=90=8C=B0"
+ "_3=2Epng?=", L"\U00010330 3.png"},
+ {"Content-Disposition: inline; filename=\"=?iso88591?Q?caf=e3_=2epng?=\"",
+ L"caf\x00e3 .png"},
+ // Space after an encode word should be removed.
+ {"Content-Disposition: inline; filename=\"=?iso88591?Q?caf=E3_?= .png\"",
+ L"caf\x00e3 .png"},
+ // Two encoded words with different charsets (not very likely to be emitted
+ // by web servers in the wild). Spaces between them are removed.
+ {"Content-Disposition: inline; filename=\"=?euc-kr?b?v7m8+iAz?="
+ " =?ksc5601?q?=BF=B9=BC=FA=2Epng?=\"", L"\xc608\xc220 3\xc608\xc220.png"},
+ {"Content-Disposition: attachment; filename=\"=?windows-1252?Q?caf=E3?="
+ " =?iso-8859-7?b?4eI=?= .png\"", L"caf\x00e3\x03b1\x03b2.png"},
+ // Non-ASCII string is passed through (and treated as UTF-8).
+ {"Content-Disposition: attachment; filename=caf\xc3\xa3.png",
+ L"caf\x00e3.png"},
+ // Failure cases
+ // Invalid hex-digit "G"
+ {"Content-Disposition: attachment; filename==?iiso88591?Q?caf=EG?=", L""},
+ // Incomplete RFC 2047 encoded-word (missing '='' at the end)
+ {"Content-Disposition: attachment; filename==?iso88591?Q?caf=E3?", L""},
+ // Extra character at the end of an encoded word
+ {"Content-Disposition: attachment; filename==?iso88591?Q?caf=E3?==", L""},
+ // Extra token at the end of an encoded word
+ {"Content-Disposition: attachment; filename==?iso88591?Q?caf=E3?=?", L""},
+ {"Content-Disposition: attachment; filename==?iso88591?Q?caf=E3?=?=", L""},
+ // Incomplete hex-escaped chars
+ {"Content-Disposition: attachment; filename==?windows-1252?Q?=63=61=E?=",
+ L""},
+ {"Content-Disposition: attachment; filename=%EC%98%88%EC%88%A", L""},
+ // %-escaped non-UTF-8 encoding is an "error"
+ {"Content-Disposition: attachment; filename=%B7%DD%BD%D1.png", L""},
+ // Two RFC 2047 encoded words in a row without a space is an error.
+ {"Content-Disposition: attachment; filename==?windows-1252?Q?caf=E3?="
+ "=?iso-8859-7?b?4eIucG5nCg==?=", L""},
+ };
+ for (size_t i = 0; i < arraysize(tests); ++i) {
+ EXPECT_EQ(tests[i].expected,
+ net_util::GetFileNameFromCD(tests[i].header_field));
+ }
+}
+
+TEST(NetUtilTest, IDNToUnicode) {
+ // TODO(jungshik) This is just a random sample of languages and is far
+ // from exhaustive. We may have to generate all the combinations
+ // of languages (powerset of a set of all the languages).
+ const wchar_t* languages[] = {
+ L"", L"en", L"zh-CN", L"ja", L"ko",
+ L"he", L"ar", L"ru", L"el", L"fr",
+ L"de", L"pt", L"se", L"th", L"hi",
+ L"de,en", L"el,en", L"zh,zh-TW,en", L"ko,ja", L"he,ru,en",
+ L"zh,ru,en"};
+ struct IDNTest {
+ const char* input;
+ const wchar_t* unicode_output;
+ const bool unicode_allowed[arraysize(languages)];
+ } idn_cases[] = {
+ // No IDN
+ {"www.google.com", L"www.google.com",
+ {true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true}},
+ {"www.google.com.", L"www.google.com.",
+ {true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true}},
+ {".", L".",
+ {true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true}},
+ {"", L"",
+ {true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true}},
+ // IDN
+ // Hanzi (Chinese)
+ {"xn--1lq90i.cn", L"\x5317\x4eac.cn",
+ {true, false, true, true, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, true, true, false,
+ true}},
+ // Hanzi + '123'
+ {"www.xn--123-p18d.com", L"www.\x4e00" L"123.com",
+ {true, false, true, true, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, true, true, false,
+ true}},
+ // Hanzi + Latin
+ {"www.xn--hello-9n1hm04c.com", L"www.hello\x4e2d\x56fd.com",
+ {false, false, true, true, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, true, true, false,
+ true}},
+ // Kanji + Kana (Japanese)
+ {"xn--l8jvb1ey91xtjb.jp", L"\x671d\x65e5\x3042\x3055\x3072.jp",
+ {true, false, false, true, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, true, false,
+ false}},
+ #if 0
+ // U+30FC is not a part of the Japanese exemplar set.
+ // Enable this after 'fixing' ICU data or locally working around it.
+ // Katakana + Latin (Japanese)
+ {"xn--e-efusa1mzf.jp", L"e\x30b3\x30de\x30fc\x30b9.jp",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ }},
+ #endif
+ // Hangul (Korean)
+ {"www.xn--or3b17p6jjc.kr", L"www.\xc804\xc790\xc815\xbd80.kr",
+ {true, false, false, false, true,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, true, false,
+ false}},
+ // b<u-umlaut>cher (German)
+ {"xn--bcher-kva.de", L"b\x00fc" L"cher.de",
+ {true, false, false, false, false,
+ false, false, false, false, true,
+ true, true, false, false, false,
+ true, false, false, false, false,
+ false}},
+ // a with diaeresis
+ {"www.xn--frgbolaget-q5a.se", L"www.f\x00e4rgbolaget.se",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ true, false, false, false, false,
+ true, false, false, false, false,
+ false}},
+ // c-cedilla (French)
+ {"www.xn--alliancefranaise-npb.fr", L"www.alliancefran\x00e7" L"aise.fr",
+ {true, false, false, false, false,
+ false, false, false, false, true,
+ false, true, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // caf'e with acute accent' (French)
+ {"xn--caf-dma.fr", L"caf\x00e9.fr",
+ {true, false, false, false, false,
+ false, false, false, false, true,
+ false, true, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // c-cedillla and a with tilde (Portuguese)
+ {"xn--poema-9qae5a.com.br", L"p\x00e3oema\x00e7\x00e3.com.br",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ false, true, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // s with caron
+ {"xn--achy-f6a.com", L"\x0161" L"achy.com",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // TODO(jungshik) : Add examples with Cyrillic letters
+ // only used in some languages written in Cyrillic.
+ // Eutopia (Greek)
+ {"xn--kxae4bafwg.gr", L"\x03bf\x03c5\x03c4\x03bf\x03c0\x03af\x03b1.gr",
+ {true, false, false, false, false,
+ false, false, false, true, false,
+ false, false, false, false, false,
+ false, true, false, false, false,
+ false}},
+ // Eutopia + 123 (Greek)
+ {"xn---123-pldm0haj2bk.gr",
+ L"\x03bf\x03c5\x03c4\x03bf\x03c0\x03af\x03b1-123.gr",
+ {true, false, false, false, false,
+ false, false, false, true, false,
+ false, false, false, false, false,
+ false, true, false, false, false,
+ false}},
+ // Cyrillic (Russian)
+ {"xn--n1aeec9b.ru", L"\x0442\x043e\x0440\x0442\x044b.ru",
+ {true, false, false, false, false,
+ false, false, true, false, false,
+ false, false, false, false, false,
+ false, false, false, false, true,
+ true}},
+ // Cyrillic + 123 (Russian)
+ {"xn---123-45dmmc5f.ru", L"\x0442\x043e\x0440\x0442\x044b-123.ru",
+ {true, false, false, false, false,
+ false, false, true, false, false,
+ false, false, false, false, false,
+ false, false, false, false, true,
+ true}},
+ // Arabic
+ {"xn--mgba1fmg.ar", L"\x0627\x0641\x0644\x0627\x0645.ar",
+ {true, false, false, false, false,
+ false, true, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // Hebrew
+ {"xn--4dbib.he", L"\x05d5\x05d0\x05d4.he",
+ {true, false, false, false, false,
+ true, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, true,
+ false}},
+ // Thai
+ {"xn--12c2cc4ag3b4ccu.th",
+ L"\x0e2a\x0e32\x0e22\x0e01\x0e32\x0e23\x0e1a\x0e34\x0e19.th",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, true, false,
+ false, false, false, false, false,
+ false}},
+ // Devangari (Hindi)
+ {"www.xn--l1b6a9e1b7c.in", L"www.\x0905\x0915\x094b\x0932\x093e.in",
+ {true, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, true,
+ false, false, false, false, false,
+ false}},
+ // Invalid IDN
+ {"xn--hello?world.com", NULL,
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // Unsafe IDNs
+ // "payp<alpha>l.com"
+ {"www.xn--paypl-g9d.com", L"payp\x03b1l.com",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // google.gr with Greek omicron and epsilon
+ {"xn--ggl-6xc1ca.gr", L"g\x03bf\x03bfgl\x03b5.gr",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // google.ru with Cyrillic o
+ {"xn--ggl-tdd6ba.ru", L"g\x043e\x043egl\x0435.ru",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // h<e with acute>llo<China in Han>.cn
+ {"xn--hllo-bpa7979ih5m.cn", L"h\x00e9llo\x4e2d\x56fd.cn",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // <Greek rho><Cyrillic a><Cyrillic u>.ru
+ {"xn--2xa6t2b.ru", L"\x03c1\x0430\x0443.ru",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ // One that's really long that will force a buffer realloc
+ {"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa",
+ L"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa",
+ {true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true, true, true, true, true,
+ true}},
+ // Test cases for characters we blacklisted although allowed in IDN.
+ // Embedded spaces will be turned to %20 in the display.
+ // TODO(jungshik): We need to have more cases. This is a typical
+ // data-driven trap. The following test cases need to be separated
+ // and tested only for a couple of languages.
+ {"xn--osd3820f24c.kr", L"\xac00\xb098\x115f.kr",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false}},
+ {"www.xn--google-ho0coa.com", L"www.\x2039google\x203a.com",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ }},
+ {"google.xn--comabc-k8d", L"google.com\x0338" L"abc",
+ {false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ }},
+#if 0
+ // These two cases are special. We need a separate test.
+ // U+3000 and U+3002 are normalized to ASCII space and dot.
+ {"xn-- -kq6ay5z.cn", L"\x4e2d\x56fd\x3000.cn",
+ {false, false, true, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, true, false, false,
+ true}},
+ {"xn--fiqs8s.cn", L"\x4e2d\x56fd\x3002" L"cn",
+ {false, false, true, false, false,
+ false, false, false, false, false,
+ false, false, false, false, false,
+ false, false, true, false, false,
+ true}},
+#endif
+ };
+
+ for (int i = 0; i < arraysize(idn_cases); i++) {
+ for (int j = 0; j < arraysize(languages); j++) {
+ std::wstring output;
+ net_util::IDNToUnicode(idn_cases[i].input,
+ static_cast<int>(strlen(idn_cases[i].input)),
+ languages[j],
+ &output);
+ std::wstring expected(idn_cases[i].unicode_allowed[j] ?
+ idn_cases[i].unicode_output :
+ ASCIIToWide(idn_cases[i].input));
+ EXPECT_EQ(expected, output);
+ }
+ }
+}
+
+TEST(NetUtilTest, StripWWW) {
+ EXPECT_EQ(L"", net_util::StripWWW(L""));
+ EXPECT_EQ(L"", net_util::StripWWW(L"www."));
+ EXPECT_EQ(L"blah", net_util::StripWWW(L"www.blah"));
+ EXPECT_EQ(L"blah", net_util::StripWWW(L"blah"));
+}
+
+TEST(NetUtilTest, GetSuggestedFilename) {
+ struct FilenameTest {
+ const char* url;
+ const wchar_t* content_disp_header;
+ const wchar_t* default_filename;
+ const wchar_t* expected_filename;
+ } test_cases[] = {
+ {"http://www.google.com/",
+ L"Content-disposition: attachment; filename=test.html",
+ L"",
+ L"test.html"},
+ {"http://www.google.com/",
+ L"Content-disposition: attachment; filename=\"test.html\"",
+ L"",
+ L"test.html"},
+ {"http://www.google.com/path/test.html",
+ L"Content-disposition: attachment",
+ L"",
+ L"test.html"},
+ {"http://www.google.com/path/test.html",
+ L"Content-disposition: attachment;",
+ L"",
+ L"test.html"},
+ {"http://www.google.com/",
+ L"",
+ L"",
+ L"www.google.com"},
+ {"http://www.google.com/test.html",
+ L"",
+ L"",
+ L"test.html"},
+ // Now that we use googleurl's ExtractFileName, this case falls back
+ // to the hostname. If this behavior is not desirable, we'd better
+ // change ExtractFileName (in url_parse).
+ {"http://www.google.com/path/",
+ L"",
+ L"",
+ L"www.google.com"},
+ {"http://www.google.com/path",
+ L"",
+ L"",
+ L"path"},
+ {"file:///",
+ L"",
+ L"",
+ L"download"},
+ {"view-cache:",
+ L"",
+ L"",
+ L"download"},
+ {"http://www.google.com/",
+ L"Content-disposition: attachment; filename =\"test.html\"",
+ L"download",
+ L"test.html"},
+ {"http://www.google.com/",
+ L"",
+ L"download",
+ L"download"},
+ {"http://www.google.com/",
+ L"Content-disposition: attachment; filename=\"../test.html\"",
+ L"",
+ L"test.html"},
+ {"http://www.google.com/",
+ L"Content-disposition: attachment; filename=\"..\"",
+ L"download",
+ L"download"},
+ {"http://www.google.com/test.html",
+ L"Content-disposition: attachment; filename=\"..\"",
+ L"download",
+ L"test.html"},
+ // Below is a small subset of cases taken from GetFileNameFromCD test above.
+ {"http://www.google.com/",
+ L"Content-Disposition: attachment; filename=\"%EC%98%88%EC%88%A0%20"
+ L"%EC%98%88%EC%88%A0.jpg\"",
+ L"",
+ L"\uc608\uc220 \uc608\uc220.jpg"},
+ {"http://www.google.com/%EC%98%88%EC%88%A0%20%EC%98%88%EC%88%A0.jpg",
+ L"",
+ L"download",
+ L"\uc608\uc220 \uc608\uc220.jpg"},
+ {"http://www.google.com/",
+ L"Content-disposition: attachment;",
+ L"\uB2E4\uC6B4\uB85C\uB4DC",
+ L"\uB2E4\uC6B4\uB85C\uB4DC"},
+ {"http://www.google.com/",
+ L"Content-Disposition: attachment; filename=\"=?EUC-JP?Q?=B7=DD=BD="
+ L"D13=2Epng?=\"",
+ L"download",
+ L"\u82b8\u88533.png"},
+ // Invalid C-D header. Extracts filename from url.
+ {"http://www.google.com/test.html",
+ L"Content-Disposition: attachment; filename==?iiso88591?Q?caf=EG?=",
+ L"",
+ L"test.html"},
+ };
+ for (int i = 0; i < arraysize(test_cases); ++i) {
+ std::wstring filename = net_util::GetSuggestedFilename(
+ GURL(test_cases[i].url), test_cases[i].content_disp_header,
+ test_cases[i].default_filename);
+ EXPECT_EQ(std::wstring(test_cases[i].expected_filename), filename);
+ }
+}