blob: d73536ef695a16d619068404ab98dcfd8609a438 [file] [log] [blame]
// Copyright 2014 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 "storage/common/file_system/file_system_util.h"
#include <stddef.h>
#include "base/files/file_path.h"
#include "base/macros.h"
#include "base/stl_util.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"
using storage::CrackIsolatedFileSystemName;
using storage::GetExternalFileSystemRootURIString;
using storage::GetIsolatedFileSystemName;
using storage::GetIsolatedFileSystemRootURIString;
using storage::ValidateIsolatedFileSystemId;
using storage::VirtualPath;
namespace content {
namespace {
class FileSystemUtilTest : public testing::Test {};
TEST_F(FileSystemUtilTest, ParseFileSystemSchemeURL) {
GURL uri("filesystem:http://chromium.org/temporary/foo/bar");
GURL origin_url;
storage::FileSystemType type;
base::FilePath virtual_path;
ParseFileSystemSchemeURL(uri, &origin_url, &type, &virtual_path);
EXPECT_EQ(GURL("http://chromium.org"), origin_url);
EXPECT_EQ(storage::kFileSystemTypeTemporary, type);
#if defined(FILE_PATH_USES_WIN_SEPARATORS)
base::FilePath expected_path(FILE_PATH_LITERAL("foo\\bar"));
#else
base::FilePath expected_path(FILE_PATH_LITERAL("foo/bar"));
#endif
EXPECT_EQ(expected_path, virtual_path);
}
TEST_F(FileSystemUtilTest, GetTempFileSystemRootURI) {
GURL origin_url("http://chromium.org");
storage::FileSystemType type = storage::kFileSystemTypeTemporary;
GURL uri = GURL("filesystem:http://chromium.org/temporary/");
EXPECT_EQ(uri, GetFileSystemRootURI(origin_url, type));
}
TEST_F(FileSystemUtilTest, GetPersistentFileSystemRootURI) {
GURL origin_url("http://chromium.org");
storage::FileSystemType type = storage::kFileSystemTypePersistent;
GURL uri = GURL("filesystem:http://chromium.org/persistent/");
EXPECT_EQ(uri, GetFileSystemRootURI(origin_url, type));
}
TEST_F(FileSystemUtilTest, VirtualPathBaseName) {
struct test_data {
const base::FilePath::StringType path;
const base::FilePath::StringType base_name;
} test_cases[] = {
{FILE_PATH_LITERAL("foo/bar"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("foo/b:bar"), FILE_PATH_LITERAL("b:bar")},
{FILE_PATH_LITERAL(""), FILE_PATH_LITERAL("")},
{FILE_PATH_LITERAL("/"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("foo//////bar"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("foo/bar/"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("foo/bar/////"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("/bar/////"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("bar/////"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("bar/"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("/bar"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("////bar"), FILE_PATH_LITERAL("bar")},
{FILE_PATH_LITERAL("bar"), FILE_PATH_LITERAL("bar")}};
for (const auto& test_case : test_cases) {
base::FilePath input = base::FilePath(test_case.path);
base::FilePath base_name = VirtualPath::BaseName(input);
EXPECT_EQ(test_case.base_name, base_name.value());
}
}
TEST_F(FileSystemUtilTest, VirtualPathDirName) {
struct test_data {
const base::FilePath::StringType path;
const base::FilePath::StringType dir_name;
} test_cases[] = {
{FILE_PATH_LITERAL("foo/bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo/b:bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL(""), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("/"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("foo//////bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo/bar/"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo/bar/////"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("/bar/////"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("bar/////"), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("bar/"), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("/bar"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("////bar"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("bar"), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("c:bar"), FILE_PATH_LITERAL(".")},
#ifdef FILE_PATH_USES_WIN_SEPARATORS
{FILE_PATH_LITERAL("foo\\bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo\\b:bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("\\"), FILE_PATH_LITERAL("\\")},
{FILE_PATH_LITERAL("foo\\\\\\\\\\\\bar"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo\\bar\\"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("foo\\bar\\\\\\\\\\"), FILE_PATH_LITERAL("foo")},
{FILE_PATH_LITERAL("\\bar\\\\\\\\\\"), FILE_PATH_LITERAL("\\")},
{FILE_PATH_LITERAL("bar\\\\\\\\\\"), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("bar\\"), FILE_PATH_LITERAL(".")},
{FILE_PATH_LITERAL("\\bar"), FILE_PATH_LITERAL("\\")},
{FILE_PATH_LITERAL("\\\\\\\\bar"), FILE_PATH_LITERAL("\\")},
#endif
};
for (const auto& test_case : test_cases) {
base::FilePath input = base::FilePath(test_case.path);
base::FilePath dir_name = VirtualPath::DirName(input);
EXPECT_EQ(test_case.dir_name, dir_name.value());
}
}
TEST_F(FileSystemUtilTest, GetNormalizedFilePath) {
struct test_data {
const base::FilePath::StringType path;
const base::FilePath::StringType normalized_path;
} test_cases[] = {
{FILE_PATH_LITERAL(""), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("/"), FILE_PATH_LITERAL("/")},
{FILE_PATH_LITERAL("foo/bar"), FILE_PATH_LITERAL("/foo/bar")},
{FILE_PATH_LITERAL("/foo/bar"), FILE_PATH_LITERAL("/foo/bar")},
#if defined(FILE_PATH_USES_WIN_SEPARATORS)
{FILE_PATH_LITERAL("\\foo"), FILE_PATH_LITERAL("/foo")},
#endif
};
for (const auto& test_case : test_cases) {
base::FilePath input = base::FilePath(test_case.path);
base::FilePath::StringType normalized_path_string =
VirtualPath::GetNormalizedFilePath(input);
EXPECT_EQ(test_case.normalized_path, normalized_path_string);
}
}
TEST_F(FileSystemUtilTest, IsAbsolutePath) {
EXPECT_TRUE(VirtualPath::IsAbsolute(FILE_PATH_LITERAL("/")));
EXPECT_TRUE(VirtualPath::IsAbsolute(FILE_PATH_LITERAL("/foo/bar")));
EXPECT_FALSE(VirtualPath::IsAbsolute(base::FilePath::StringType()));
EXPECT_FALSE(VirtualPath::IsAbsolute(FILE_PATH_LITERAL("foo/bar")));
}
TEST_F(FileSystemUtilTest, IsRootPath) {
EXPECT_TRUE(VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL(""))));
EXPECT_TRUE(VirtualPath::IsRootPath(base::FilePath()));
EXPECT_TRUE(VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL("/"))));
EXPECT_TRUE(VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL("//"))));
EXPECT_FALSE(
VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL("c:/"))));
#if defined(FILE_PATH_USES_WIN_SEPARATORS)
EXPECT_TRUE(VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL("\\"))));
EXPECT_FALSE(
VirtualPath::IsRootPath(base::FilePath(FILE_PATH_LITERAL("c:\\"))));
#endif
}
TEST_F(FileSystemUtilTest, VirtualPathGetComponents) {
struct test_data {
const base::FilePath::StringType path;
size_t count;
const base::FilePath::StringType components[2];
} test_cases[] = {
{FILE_PATH_LITERAL("foo/bar"),
2,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("bar")}},
{FILE_PATH_LITERAL("foo"),
1,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("")}},
{FILE_PATH_LITERAL("foo////bar"),
2,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("bar")}},
{FILE_PATH_LITERAL("foo/c:bar"),
2,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("c:bar")}},
{FILE_PATH_LITERAL("c:foo/bar"),
2,
{FILE_PATH_LITERAL("c:foo"), FILE_PATH_LITERAL("bar")}},
{FILE_PATH_LITERAL("foo/bar"),
2,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("bar")}},
{FILE_PATH_LITERAL("/foo/bar"),
2,
{FILE_PATH_LITERAL("foo"), FILE_PATH_LITERAL("bar")}},
{FILE_PATH_LITERAL("c:/bar"),
2,
{FILE_PATH_LITERAL("c:"), FILE_PATH_LITERAL("bar")}},
#ifdef FILE_PATH_USES_WIN_SEPARATORS
{FILE_PATH_LITERAL("c:\\bar"),
2,
{FILE_PATH_LITERAL("c:"), FILE_PATH_LITERAL("bar")}},
#endif
};
for (const auto& test_case : test_cases) {
base::FilePath input = base::FilePath(test_case.path);
std::vector<base::FilePath::StringType> components =
VirtualPath::GetComponents(input);
EXPECT_EQ(test_case.count, components.size());
for (size_t j = 0; j < components.size(); ++j)
EXPECT_EQ(test_case.components[j], components[j]);
}
for (const auto& test_case : test_cases) {
base::FilePath input = base::FilePath(test_case.path);
std::vector<std::string> components =
VirtualPath::GetComponentsUTF8Unsafe(input);
EXPECT_EQ(test_case.count, components.size());
for (size_t j = 0; j < components.size(); ++j) {
EXPECT_EQ(base::FilePath(test_case.components[j]).AsUTF8Unsafe(),
components[j]);
}
}
}
TEST_F(FileSystemUtilTest, GetIsolatedFileSystemName) {
GURL origin_url("http://foo");
std::string fsname1 = GetIsolatedFileSystemName(origin_url, "bar");
EXPECT_EQ("http_foo_0:Isolated_bar", fsname1);
}
TEST_F(FileSystemUtilTest, CrackIsolatedFileSystemName) {
std::string fsid;
EXPECT_TRUE(CrackIsolatedFileSystemName("foo:Isolated_bar", &fsid));
EXPECT_EQ("bar", fsid);
EXPECT_TRUE(CrackIsolatedFileSystemName("foo:isolated_bar", &fsid));
EXPECT_EQ("bar", fsid);
EXPECT_TRUE(CrackIsolatedFileSystemName("foo:Isolated__bar", &fsid));
EXPECT_EQ("_bar", fsid);
EXPECT_TRUE(CrackIsolatedFileSystemName("foo::Isolated_bar", &fsid));
EXPECT_EQ("bar", fsid);
}
TEST_F(FileSystemUtilTest, RejectBadIsolatedFileSystemName) {
std::string fsid;
EXPECT_FALSE(CrackIsolatedFileSystemName("foobar", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:_bar", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:Isolatedbar", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("fooIsolatedbar", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:Persistent", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:Temporary", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:External", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName(":Isolated_bar", &fsid));
EXPECT_FALSE(CrackIsolatedFileSystemName("foo:Isolated_", &fsid));
}
TEST_F(FileSystemUtilTest, ValidateIsolatedFileSystemId) {
EXPECT_TRUE(ValidateIsolatedFileSystemId("ABCDEF0123456789ABCDEF0123456789"));
EXPECT_TRUE(ValidateIsolatedFileSystemId("ABCDEFABCDEFABCDEFABCDEFABCDEFAB"));
EXPECT_TRUE(ValidateIsolatedFileSystemId("01234567890123456789012345678901"));
const size_t kExpectedFileSystemIdSize = 32;
// Should not contain lowercase characters.
const std::string kLowercaseId = "abcdef0123456789abcdef0123456789";
EXPECT_EQ(kExpectedFileSystemIdSize, kLowercaseId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kLowercaseId));
// Should not be shorter/longer than expected.
EXPECT_FALSE(ValidateIsolatedFileSystemId(std::string()));
const std::string kShorterId = "ABCDEF0123456789ABCDEF";
EXPECT_GT(kExpectedFileSystemIdSize, kShorterId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kShorterId));
const std::string kLongerId = "ABCDEF0123456789ABCDEF0123456789ABCDEF";
EXPECT_LT(kExpectedFileSystemIdSize, kLongerId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kLongerId));
// Should not contain not alphabetical nor numerical characters.
const std::string kSlashId = "ABCD/EFGH/IJKL/MNOP/QRST/UVWX/YZ";
EXPECT_EQ(kExpectedFileSystemIdSize, kSlashId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kSlashId));
const std::string kBackslashId = "ABCD\\EFGH\\IJKL\\MNOP\\QRST\\UVWX\\YZ";
EXPECT_EQ(kExpectedFileSystemIdSize, kBackslashId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kBackslashId));
const std::string kSpaceId = "ABCD EFGH IJKL MNOP QRST UVWX YZ";
EXPECT_EQ(kExpectedFileSystemIdSize, kSpaceId.size());
EXPECT_FALSE(ValidateIsolatedFileSystemId(kSpaceId));
}
TEST_F(FileSystemUtilTest, GetIsolatedFileSystemRootURIString) {
const GURL kOriginURL("http://foo");
// Percents must be escaped, otherwise they will be unintentionally unescaped.
const std::string kFileSystemId = "A%20B";
const std::string kRootName = "C%20D";
const std::string url_string =
GetIsolatedFileSystemRootURIString(kOriginURL, kFileSystemId, kRootName);
EXPECT_EQ("filesystem:http://foo/isolated/A%2520B/C%2520D/", url_string);
}
TEST_F(FileSystemUtilTest, GetExternalFileSystemRootURIString) {
const GURL kOriginURL("http://foo");
// Percents must be escaped, otherwise they will be unintentionally unescaped.
const std::string kMountName = "X%20Y";
const std::string url_string =
GetExternalFileSystemRootURIString(kOriginURL, kMountName);
EXPECT_EQ("filesystem:http://foo/external/X%2520Y/", url_string);
}
} // namespace
} // namespace content