Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1673)

Unified Diff: chrome/browser/download/download_util_unittest.cc

Issue 3043048: Clean up download code: (Closed)
Patch Set: Created 10 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « chrome/browser/download/download_util.cc ('k') | chrome/browser/download/save_package.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: chrome/browser/download/download_util_unittest.cc
diff --git a/chrome/browser/download/download_util_unittest.cc b/chrome/browser/download/download_util_unittest.cc
new file mode 100644
index 0000000000000000000000000000000000000000..d33ece9fed3dd83647709e429f38555f0e9ca268
--- /dev/null
+++ b/chrome/browser/download/download_util_unittest.cc
@@ -0,0 +1,588 @@
+// Copyright (c) 2010 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.
+
+#if defined(OS_POSIX) && !defined(OS_MACOSX)
+#include <locale.h>
+#endif
+
+#include "base/string_util.h"
+#include "chrome/browser/download/download_util.h"
+#include "googleurl/src/gurl.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+#if defined(OS_WIN)
+#define JPEG_EXT L".jpg"
+#define HTML_EXT L".htm"
+#define TXT_EXT L".txt"
+#define TAR_EXT L".tar"
+#elif defined(OS_MACOSX)
+#define JPEG_EXT L".jpeg"
+#define HTML_EXT L".html"
+#define TXT_EXT L".txt"
+#define TAR_EXT L".tar"
+#else
+#define JPEG_EXT L".jpg"
+#define HTML_EXT L".html"
+#define TXT_EXT L".txt"
+#define TAR_EXT L".tar"
+#endif
+
+namespace {
+
+const struct {
+ const char* disposition;
+ const char* url;
+ const char* mime_type;
+ const wchar_t* expected_name;
+} kGenerateFileNameTestCases[] = {
+ // No 'filename' keyword in the disposition, use the URL
+ {"a_file_name.txt",
+ "http://www.evil.com/my_download.txt",
+ "text/plain",
+ L"my_download.txt"},
+
+ // Disposition has relative paths, remove them
+ {"filename=../../../../././../a_file_name.txt",
+ "http://www.evil.com/my_download.txt",
+ "text/plain",
+ L"a_file_name.txt"},
+
+ // Disposition has parent directories, remove them
+ {"filename=dir1/dir2/a_file_name.txt",
+ "http://www.evil.com/my_download.txt",
+ "text/plain",
+ L"a_file_name.txt"},
+
+ // No useful information in disposition or URL, use default
+ {"", "http://www.truncated.com/path/", "text/plain",
+ L"download" TXT_EXT
+ },
+
+ // A normal avi should get .avi and not .avi.avi
+ {"", "https://blah.google.com/misc/2.avi", "video/x-msvideo", L"2.avi"},
+
+ // Spaces in the disposition file name
+ {"filename=My Downloaded File.exe",
+ "http://www.frontpagehacker.com/a_download.exe",
+ "application/octet-stream",
+ L"My Downloaded File.exe"},
+
+ // This block tests whether we append extensions based on MIME types;
+ // we don't do this on Linux, so we skip the tests rather than #ifdef
+ // them up.
+#if !defined(OS_POSIX) || defined(OS_MACOSX)
+ {"filename=my-cat",
+ "http://www.example.com/my-cat",
+ "image/jpeg",
+ L"my-cat" JPEG_EXT
+ },
+
+ {"filename=my-cat",
+ "http://www.example.com/my-cat",
+ "text/plain",
+ L"my-cat.txt"},
+
+ {"filename=my-cat",
+ "http://www.example.com/my-cat",
+ "text/html",
+ L"my-cat" HTML_EXT
+ },
+
+ {"filename=my-cat",
+ "http://www.example.com/my-cat",
+ "dance/party",
+ L"my-cat"},
+#endif // !defined(OS_POSIX) || defined(OS_MACOSX)
+
+ {"filename=my-cat.jpg",
+ "http://www.example.com/my-cat.jpg",
+ "text/plain",
+ L"my-cat.jpg"},
+
+ // .exe tests.
+#if defined(OS_WIN)
+ {"filename=evil.exe",
+ "http://www.goodguy.com/evil.exe",
+ "image/jpeg",
+ L"evil.jpg"},
+
+ {"filename=ok.exe",
+ "http://www.goodguy.com/ok.exe",
+ "binary/octet-stream",
+ L"ok.exe"},
+
+ {"filename=evil.exe.exe",
+ "http://www.goodguy.com/evil.exe.exe",
+ "dance/party",
+ L"evil.exe.download"},
+
+ {"filename=evil.exe",
+ "http://www.goodguy.com/evil.exe",
+ "application/xml",
+ L"evil.xml"},
+
+ {"filename=evil.exe",
+ "http://www.goodguy.com/evil.exe",
+ "application/html+xml",
+ L"evil.download"},
+
+ {"filename=evil.exe",
+ "http://www.goodguy.com/evil.exe",
+ "application/rss+xml",
+ L"evil.download"},
+
+ // Test truncation of trailing dots and spaces
+ {"filename=evil.exe ",
+ "http://www.goodguy.com/evil.exe ",
+ "binary/octet-stream",
+ L"evil.exe"},
+
+ {"filename=evil.exe.",
+ "http://www.goodguy.com/evil.exe.",
+ "binary/octet-stream",
+ L"evil.exe"},
+
+ {"filename=evil.exe. . .",
+ "http://www.goodguy.com/evil.exe. . .",
+ "binary/octet-stream",
+ L"evil.exe"},
+
+ {"filename=evil.",
+ "http://www.goodguy.com/evil.",
+ "binary/octet-stream",
+ L"evil"},
+
+ {"filename=. . . . .",
+ "http://www.goodguy.com/. . . . .",
+ "binary/octet-stream",
+ L"download"},
+
+#endif // OS_WIN
+
+ {"filename=utils.js",
+ "http://www.goodguy.com/utils.js",
+ "application/x-javascript",
+ L"utils.js"},
+
+ {"filename=contacts.js",
+ "http://www.goodguy.com/contacts.js",
+ "application/json",
+ L"contacts.js"},
+
+ {"filename=utils.js",
+ "http://www.goodguy.com/utils.js",
+ "text/javascript",
+ L"utils.js"},
+
+ {"filename=utils.js",
+ "http://www.goodguy.com/utils.js",
+ "text/javascript;version=2",
+ L"utils.js"},
+
+ {"filename=utils.js",
+ "http://www.goodguy.com/utils.js",
+ "application/ecmascript",
+ L"utils.js"},
+
+ {"filename=utils.js",
+ "http://www.goodguy.com/utils.js",
+ "application/ecmascript;version=4",
+ L"utils.js"},
+
+ {"filename=program.exe",
+ "http://www.goodguy.com/program.exe",
+ "application/foo-bar",
+ L"program.exe"},
+
+ {"filename=../foo.txt",
+ "http://www.evil.com/../foo.txt",
+ "text/plain",
+ L"foo.txt"},
+
+ {"filename=..\\foo.txt",
+ "http://www.evil.com/..\\foo.txt",
+ "text/plain",
+#if defined(OS_WIN)
+ L"foo.txt"
+#else
+ L"\\foo.txt"
+#endif
+ },
+
+ {"filename=.hidden",
+ "http://www.evil.com/.hidden",
+ "text/plain",
+ L"hidden" TXT_EXT
+ },
+
+ {"filename=trailing.",
+ "http://www.evil.com/trailing.",
+ "dance/party",
+ L"trailing"
+ },
+
+ {"filename=trailing.",
+ "http://www.evil.com/trailing.",
+ "text/plain",
+ L"trailing" TXT_EXT
+ },
+
+ {"filename=.",
+ "http://www.evil.com/.",
+ "dance/party",
+ L"download"},
+
+ {"filename=..",
+ "http://www.evil.com/..",
+ "dance/party",
+ L"download"},
+
+ {"filename=...",
+ "http://www.evil.com/...",
+ "dance/party",
+ L"download"},
+
+ // Note that this one doesn't have "filename=" on it.
+ {"a_file_name.txt",
+ "http://www.evil.com/",
+ "image/jpeg",
+ L"download" JPEG_EXT
+ },
+
+ {"filename=",
+ "http://www.evil.com/",
+ "image/jpeg",
+ L"download" JPEG_EXT
+ },
+
+ {"filename=simple",
+ "http://www.example.com/simple",
+ "application/octet-stream",
+ L"simple"},
+
+ {"filename=COM1",
+ "http://www.goodguy.com/COM1",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"_COM1"
+#else
+ L"COM1"
+#endif
+ },
+
+ {"filename=COM4.txt",
+ "http://www.goodguy.com/COM4.txt",
+ "text/plain",
+#if defined(OS_WIN)
+ L"_COM4.txt"
+#else
+ L"COM4.txt"
+#endif
+ },
+
+ {"filename=lpt1.TXT",
+ "http://www.goodguy.com/lpt1.TXT",
+ "text/plain",
+#if defined(OS_WIN)
+ L"_lpt1.TXT"
+#else
+ L"lpt1.TXT"
+#endif
+ },
+
+ {"filename=clock$.txt",
+ "http://www.goodguy.com/clock$.txt",
+ "text/plain",
+#if defined(OS_WIN)
+ L"_clock$.txt"
+#else
+ L"clock$.txt"
+#endif
+ },
+
+ {"filename=mycom1.foo",
+ "http://www.goodguy.com/mycom1.foo",
+ "text/plain",
+ L"mycom1.foo"},
+
+ {"filename=Setup.exe.local",
+ "http://www.badguy.com/Setup.exe.local",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"Setup.exe.download"
+#else
+ L"Setup.exe.local"
+#endif
+ },
+
+ {"filename=Setup.exe.local.local",
+ "http://www.badguy.com/Setup.exe.local",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"Setup.exe.local.download"
+#else
+ L"Setup.exe.local.local"
+#endif
+ },
+
+ {"filename=Setup.exe.lnk",
+ "http://www.badguy.com/Setup.exe.lnk",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"Setup.exe.download"
+#else
+ L"Setup.exe.lnk"
+#endif
+ },
+
+ {"filename=Desktop.ini",
+ "http://www.badguy.com/Desktop.ini",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"_Desktop.ini"
+#else
+ L"Desktop.ini"
+#endif
+ },
+
+ {"filename=Thumbs.db",
+ "http://www.badguy.com/Thumbs.db",
+ "application/foo-bar",
+#if defined(OS_WIN)
+ L"_Thumbs.db"
+#else
+ L"Thumbs.db"
+#endif
+ },
+
+ {"filename=source.srf",
+ "http://www.hotmail.com",
+ "image/jpeg",
+ L"source.srf" JPEG_EXT
+ },
+
+ {"filename=source.jpg",
+ "http://www.hotmail.com",
+ "application/x-javascript",
+#if defined(OS_WIN)
+ L"source.jpg"
+#elif defined(OS_MACOSX)
+ L"source.jpg.js"
+#else
+ L"source.jpg"
+#endif
+ },
+
+ // NetUtilTest.{GetSuggestedFilename, GetFileNameFromCD} test these
+ // more thoroughly. Tested below are a small set of samples.
+ {"attachment; filename=\"%EC%98%88%EC%88%A0%20%EC%98%88%EC%88%A0.jpg\"",
+ "http://www.examples.com/",
+ "image/jpeg",
+ L"\uc608\uc220 \uc608\uc220.jpg"},
+
+ {"attachment; name=abc de.pdf",
+ "http://www.examples.com/q.cgi?id=abc",
+ "application/octet-stream",
+ L"abc de.pdf"},
+
+ {"filename=\"=?EUC-JP?Q?=B7=DD=BD=D13=2Epng?=\"",
+ "http://www.example.com/path",
+ "image/png",
+ L"\x82b8\x8853" L"3.png"},
+
+ // The following two have invalid CD headers and filenames come
+ // from the URL.
+ {"attachment; filename==?iiso88591?Q?caf=EG?=",
+ "http://www.example.com/test%20123",
+ "image/jpeg",
+ L"test 123" JPEG_EXT
+ },
+
+ {"malformed_disposition",
+ "http://www.google.com/%EC%98%88%EC%88%A0%20%EC%98%88%EC%88%A0.jpg",
+ "image/jpeg",
+ L"\uc608\uc220 \uc608\uc220.jpg"},
+
+ // Invalid C-D. No filename from URL. Falls back to 'download'.
+ {"attachment; filename==?iso88591?Q?caf=E3?",
+ "http://www.google.com/path1/path2/",
+ "image/jpeg",
+ L"download" JPEG_EXT
+ },
+
+ // Issue=5772.
+ {"",
+ "http://www.example.com/foo.tar.gz",
+ "application/x-tar",
+ L"foo.tar.gz"},
+
+ // Issue=7337.
+ {"",
+ "http://maged.lordaeron.org/blank.reg",
+ "text/x-registry",
+ L"blank.reg"},
+
+ {"",
+ "http://www.example.com/bar.tar",
+ "application/x-tar",
+ L"bar.tar"},
+
+ {"",
+ "http://www.example.com/bar.bogus",
+ "application/x-tar",
+ L"bar.bogus" TAR_EXT
+ },
+
+ // http://code.google.com/p/chromium/issues/detail?id=20337
+ {"filename=.download.txt",
+ "http://www.example.com/.download.txt",
+ "text/plain",
+ L"download.txt"},
+};
+
+// Tests to ensure that the file names we generate from hints from the server
+// (content-disposition, URL name, etc) don't cause security holes.
+TEST(DownloadUtilTest, GenerateFileName) {
+#if defined(OS_POSIX) && !defined(OS_MACOSX)
+ // This test doesn't run when the locale is not UTF-8 becuase some of the
+ // string conversions fail. This is OK (we have the default value) but they
+ // don't match our expectations.
+ std::string locale = setlocale(LC_CTYPE, NULL);
+ StringToLowerASCII(&locale);
+ ASSERT_NE(std::string::npos, locale.find("utf-8"))
+ << "Your locale must be set to UTF-8 for this test to pass!";
+#endif
+
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kGenerateFileNameTestCases); ++i) {
+ FilePath generated_name;
+ download_util::GenerateFileName(GURL(kGenerateFileNameTestCases[i].url),
+ kGenerateFileNameTestCases[i].disposition,
+ "",
+ kGenerateFileNameTestCases[i].mime_type,
+ &generated_name);
+ EXPECT_EQ(kGenerateFileNameTestCases[i].expected_name,
+ generated_name.ToWStringHack());
+ }
+
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kGenerateFileNameTestCases); ++i) {
+ FilePath generated_name;
+ download_util::GenerateFileName(GURL(kGenerateFileNameTestCases[i].url),
+ kGenerateFileNameTestCases[i].disposition,
+ "GBK",
+ kGenerateFileNameTestCases[i].mime_type,
+ &generated_name);
+ EXPECT_EQ(kGenerateFileNameTestCases[i].expected_name,
+ generated_name.ToWStringHack());
+ }
+
+ // A couple of cases with raw 8bit characters in C-D.
+ {
+ FilePath generated_name;
+ download_util::GenerateFileName(GURL("http://www.example.com/images?id=3"),
+ "attachment; filename=caf\xc3\xa9.png",
+ "iso-8859-1",
+ "image/png",
+ &generated_name);
+ EXPECT_EQ(L"caf\u00e9.png", generated_name.ToWStringHack());
+ }
+
+ {
+ FilePath generated_name;
+ download_util::GenerateFileName(GURL("http://www.example.com/images?id=3"),
+ "attachment; filename=caf\xe5.png",
+ "windows-1253",
+ "image/png",
+ &generated_name);
+ EXPECT_EQ(L"caf\u03b5.png", generated_name.ToWStringHack());
+ }
+}
+
+const struct {
+ const FilePath::CharType* path;
+ const char* mime_type;
+ const FilePath::CharType* expected_path;
+} kSafeFilenameCases[] = {
+#if defined(OS_WIN)
+ { FILE_PATH_LITERAL("C:\\foo\\bar.htm"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\bar.htm") },
+ { FILE_PATH_LITERAL("C:\\foo\\bar.html"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\bar.html") },
+ { FILE_PATH_LITERAL("C:\\foo\\bar"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\bar.htm") },
+
+ { FILE_PATH_LITERAL("C:\\bar.html"),
+ "image/png",
+ FILE_PATH_LITERAL("C:\\bar.png") },
+ { FILE_PATH_LITERAL("C:\\bar"),
+ "image/png",
+ FILE_PATH_LITERAL("C:\\bar.png") },
+
+ { FILE_PATH_LITERAL("C:\\foo\\bar.exe"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\bar.htm") },
+ { FILE_PATH_LITERAL("C:\\foo\\bar.exe"),
+ "image/gif",
+ FILE_PATH_LITERAL("C:\\foo\\bar.gif") },
+
+ { FILE_PATH_LITERAL("C:\\foo\\google.com"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\google.htm") },
+
+ { FILE_PATH_LITERAL("C:\\foo\\con.htm"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\_con.htm") },
+ { FILE_PATH_LITERAL("C:\\foo\\con"),
+ "text/html",
+ FILE_PATH_LITERAL("C:\\foo\\_con.htm") },
+#else
+ { FILE_PATH_LITERAL("/foo/bar.htm"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/bar.htm") },
+ { FILE_PATH_LITERAL("/foo/bar.html"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/bar.html") },
+ { FILE_PATH_LITERAL("/foo/bar"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/bar.html") },
+
+ { FILE_PATH_LITERAL("/bar.html"),
+ "image/png",
+ FILE_PATH_LITERAL("/bar.html.png") },
+ { FILE_PATH_LITERAL("/bar"),
+ "image/png",
+ FILE_PATH_LITERAL("/bar.png") },
+
+ { FILE_PATH_LITERAL("/foo/bar.exe"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/bar.html") },
+ { FILE_PATH_LITERAL("/foo/bar.exe"),
+ "image/gif",
+ FILE_PATH_LITERAL("/foo/bar.gif") },
+
+ { FILE_PATH_LITERAL("/foo/google.com"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/google.com.html") },
+
+ { FILE_PATH_LITERAL("/foo/con.htm"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/con.htm") },
+ { FILE_PATH_LITERAL("/foo/con"),
+ "text/html",
+ FILE_PATH_LITERAL("/foo/con.html") },
+#endif // OS_WIN
+};
+
+TEST(DownloadUtilTest, GenerateSafeFileName) {
+ for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kSafeFilenameCases); ++i) {
+ FilePath path(kSafeFilenameCases[i].path);
+ download_util::GenerateSafeFileName(kSafeFilenameCases[i].mime_type, &path);
+ EXPECT_EQ(kSafeFilenameCases[i].expected_path, path.value());
+ }
+}
+
+} // namespace
+
« no previous file with comments | « chrome/browser/download/download_util.cc ('k') | chrome/browser/download/save_package.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698