Home | History | Annotate | Download | only in common
      1 // 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 <algorithm>
      6 
      7 #include "base/file_util.h"
      8 #include "base/files/scoped_temp_dir.h"
      9 #include "base/path_service.h"
     10 #include "extensions/common/constants.h"
     11 #include "extensions/common/extension_paths.h"
     12 #include "extensions/common/extension_resource.h"
     13 #include "extensions/common/id_util.h"
     14 #include "testing/gtest/include/gtest/gtest.h"
     15 #include "ui/base/l10n/l10n_util.h"
     16 
     17 namespace extensions {
     18 
     19 TEST(ExtensionResourceTest, CreateEmptyResource) {
     20   ExtensionResource resource;
     21 
     22   EXPECT_TRUE(resource.extension_root().empty());
     23   EXPECT_TRUE(resource.relative_path().empty());
     24   EXPECT_TRUE(resource.GetFilePath().empty());
     25 }
     26 
     27 const base::FilePath::StringType ToLower(
     28     const base::FilePath::StringType& in_str) {
     29   base::FilePath::StringType str(in_str);
     30   std::transform(str.begin(), str.end(), str.begin(), tolower);
     31   return str;
     32 }
     33 
     34 TEST(ExtensionResourceTest, CreateWithMissingResourceOnDisk) {
     35   base::FilePath root_path;
     36   ASSERT_TRUE(PathService::Get(DIR_TEST_DATA, &root_path));
     37   base::FilePath relative_path;
     38   relative_path = relative_path.AppendASCII("cira.js");
     39   std::string extension_id = id_util::GenerateId("test");
     40   ExtensionResource resource(extension_id, root_path, relative_path);
     41 
     42   // The path doesn't exist on disk, we will be returned an empty path.
     43   EXPECT_EQ(root_path.value(), resource.extension_root().value());
     44   EXPECT_EQ(relative_path.value(), resource.relative_path().value());
     45   EXPECT_TRUE(resource.GetFilePath().empty());
     46 }
     47 
     48 TEST(ExtensionResourceTest, ResourcesOutsideOfPath) {
     49   base::ScopedTempDir temp;
     50   ASSERT_TRUE(temp.CreateUniqueTempDir());
     51 
     52   base::FilePath inner_dir = temp.path().AppendASCII("directory");
     53   ASSERT_TRUE(file_util::CreateDirectory(inner_dir));
     54   base::FilePath sub_dir = inner_dir.AppendASCII("subdir");
     55   ASSERT_TRUE(file_util::CreateDirectory(sub_dir));
     56   base::FilePath inner_file = inner_dir.AppendASCII("inner");
     57   base::FilePath outer_file = temp.path().AppendASCII("outer");
     58   ASSERT_TRUE(file_util::WriteFile(outer_file, "X", 1));
     59   ASSERT_TRUE(file_util::WriteFile(inner_file, "X", 1));
     60   std::string extension_id = id_util::GenerateId("test");
     61 
     62 #if defined(OS_POSIX)
     63   base::FilePath symlink_file = inner_dir.AppendASCII("symlink");
     64   file_util::CreateSymbolicLink(
     65       base::FilePath().AppendASCII("..").AppendASCII("outer"),
     66       symlink_file);
     67 #endif
     68 
     69   // A non-packing extension should be able to access the file within the
     70   // directory.
     71   ExtensionResource r1(extension_id, inner_dir,
     72                        base::FilePath().AppendASCII("inner"));
     73   EXPECT_FALSE(r1.GetFilePath().empty());
     74 
     75   // ... but not a relative path that walks out of |inner_dir|.
     76   ExtensionResource r2(extension_id, inner_dir,
     77                        base::FilePath().AppendASCII("..").AppendASCII("outer"));
     78   EXPECT_TRUE(r2.GetFilePath().empty());
     79 
     80   // A packing extension should also be able to access the file within the
     81   // directory.
     82   ExtensionResource r3(extension_id, inner_dir,
     83                        base::FilePath().AppendASCII("inner"));
     84   r3.set_follow_symlinks_anywhere();
     85   EXPECT_FALSE(r3.GetFilePath().empty());
     86 
     87   // ... but, again, not a relative path that walks out of |inner_dir|.
     88   ExtensionResource r4(extension_id, inner_dir,
     89                        base::FilePath().AppendASCII("..").AppendASCII("outer"));
     90   r4.set_follow_symlinks_anywhere();
     91   EXPECT_TRUE(r4.GetFilePath().empty());
     92 
     93   // ... and not even when clever current-directory syntax is present. Note
     94   // that the path for this test case can't start with the current directory
     95   // component due to quirks in FilePath::Append(), and the path must exist.
     96   ExtensionResource r4a(
     97       extension_id, inner_dir,
     98       base::FilePath().AppendASCII("subdir").AppendASCII(".").AppendASCII("..").
     99       AppendASCII("..").AppendASCII("outer"));
    100   r4a.set_follow_symlinks_anywhere();
    101   EXPECT_TRUE(r4a.GetFilePath().empty());
    102 
    103 #if defined(OS_POSIX)
    104   // The non-packing extension should also not be able to access a resource that
    105   // symlinks out of the directory.
    106   ExtensionResource r5(extension_id, inner_dir,
    107                        base::FilePath().AppendASCII("symlink"));
    108   EXPECT_TRUE(r5.GetFilePath().empty());
    109 
    110   // ... but a packing extension can.
    111   ExtensionResource r6(extension_id, inner_dir,
    112                        base::FilePath().AppendASCII("symlink"));
    113   r6.set_follow_symlinks_anywhere();
    114   EXPECT_FALSE(r6.GetFilePath().empty());
    115 #endif
    116 }
    117 
    118 TEST(ExtensionResourceTest, CreateWithAllResourcesOnDisk) {
    119   base::ScopedTempDir temp;
    120   ASSERT_TRUE(temp.CreateUniqueTempDir());
    121 
    122   // Create resource in the extension root.
    123   const char* filename = "res.ico";
    124   base::FilePath root_resource = temp.path().AppendASCII(filename);
    125   std::string data = "some foo";
    126   ASSERT_TRUE(file_util::WriteFile(root_resource, data.c_str(), data.length()));
    127 
    128   // Create l10n resources (for current locale and its parents).
    129   base::FilePath l10n_path =
    130       temp.path().Append(kLocaleFolder);
    131   ASSERT_TRUE(file_util::CreateDirectory(l10n_path));
    132 
    133   std::vector<std::string> locales;
    134   l10n_util::GetParentLocales(l10n_util::GetApplicationLocale(std::string()),
    135                               &locales);
    136   ASSERT_FALSE(locales.empty());
    137   for (size_t i = 0; i < locales.size(); i++) {
    138     base::FilePath make_path;
    139     make_path = l10n_path.AppendASCII(locales[i]);
    140     ASSERT_TRUE(file_util::CreateDirectory(make_path));
    141     ASSERT_TRUE(file_util::WriteFile(make_path.AppendASCII(filename),
    142         data.c_str(), data.length()));
    143   }
    144 
    145   base::FilePath path;
    146   std::string extension_id = id_util::GenerateId("test");
    147   ExtensionResource resource(extension_id, temp.path(),
    148                              base::FilePath().AppendASCII(filename));
    149   base::FilePath resolved_path = resource.GetFilePath();
    150 
    151   base::FilePath expected_path;
    152   // Expect default path only, since fallback logic is disabled.
    153   // See http://crbug.com/27359.
    154   expected_path = base::MakeAbsoluteFilePath(root_resource);
    155   ASSERT_FALSE(expected_path.empty());
    156 
    157   EXPECT_EQ(ToLower(expected_path.value()), ToLower(resolved_path.value()));
    158   EXPECT_EQ(ToLower(temp.path().value()),
    159             ToLower(resource.extension_root().value()));
    160   EXPECT_EQ(ToLower(base::FilePath().AppendASCII(filename).value()),
    161             ToLower(resource.relative_path().value()));
    162 }
    163 
    164 }  // namespace extensions
    165