• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 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 "storage/browser/fileapi/file_system_context.h"
6 
7 #include "base/files/scoped_temp_dir.h"
8 #include "base/message_loop/message_loop.h"
9 #include "base/strings/stringprintf.h"
10 #include "content/browser/quota/mock_quota_manager.h"
11 #include "content/public/test/mock_special_storage_policy.h"
12 #include "content/public/test/test_file_system_options.h"
13 #include "storage/browser/fileapi/external_mount_points.h"
14 #include "storage/browser/fileapi/file_system_backend.h"
15 #include "storage/browser/fileapi/isolated_context.h"
16 #include "testing/gtest/include/gtest/gtest.h"
17 
18 #define FPL(x) FILE_PATH_LITERAL(x)
19 
20 #if defined(FILE_PATH_USES_DRIVE_LETTERS)
21 #define DRIVE FPL("C:")
22 #else
23 #define DRIVE
24 #endif
25 
26 using storage::ExternalMountPoints;
27 using storage::FileSystemBackend;
28 using storage::FileSystemContext;
29 using storage::FileSystemMountOption;
30 using storage::FileSystemURL;
31 using storage::IsolatedContext;
32 
33 namespace content {
34 
35 namespace {
36 
37 const char kTestOrigin[] = "http://chromium.org/";
38 
CreateRawFileSystemURL(const std::string & type_str,const std::string & fs_id)39 GURL CreateRawFileSystemURL(const std::string& type_str,
40                             const std::string& fs_id) {
41   std::string url_str = base::StringPrintf(
42       "filesystem:http://chromium.org/%s/%s/root/file",
43       type_str.c_str(),
44       fs_id.c_str());
45   return GURL(url_str);
46 }
47 
48 class FileSystemContextTest : public testing::Test {
49  public:
FileSystemContextTest()50   FileSystemContextTest() {}
51 
SetUp()52   virtual void SetUp() {
53     ASSERT_TRUE(data_dir_.CreateUniqueTempDir());
54 
55     storage_policy_ = new MockSpecialStoragePolicy();
56 
57     mock_quota_manager_ =
58         new MockQuotaManager(false /* is_incognito */,
59                                     data_dir_.path(),
60                                     base::MessageLoopProxy::current().get(),
61                                     base::MessageLoopProxy::current().get(),
62                                     storage_policy_.get());
63   }
64 
65  protected:
CreateFileSystemContextForTest(storage::ExternalMountPoints * external_mount_points)66   FileSystemContext* CreateFileSystemContextForTest(
67       storage::ExternalMountPoints* external_mount_points) {
68     return new FileSystemContext(
69         base::MessageLoopProxy::current().get(),
70         base::MessageLoopProxy::current().get(),
71         external_mount_points,
72         storage_policy_.get(),
73         mock_quota_manager_->proxy(),
74         ScopedVector<FileSystemBackend>(),
75         std::vector<storage::URLRequestAutoMountHandler>(),
76         data_dir_.path(),
77         CreateAllowFileAccessOptions());
78   }
79 
80   // Verifies a *valid* filesystem url has expected values.
ExpectFileSystemURLMatches(const FileSystemURL & url,const GURL & expect_origin,storage::FileSystemType expect_mount_type,storage::FileSystemType expect_type,const base::FilePath & expect_path,const base::FilePath & expect_virtual_path,const std::string & expect_filesystem_id)81   void ExpectFileSystemURLMatches(const FileSystemURL& url,
82                                   const GURL& expect_origin,
83                                   storage::FileSystemType expect_mount_type,
84                                   storage::FileSystemType expect_type,
85                                   const base::FilePath& expect_path,
86                                   const base::FilePath& expect_virtual_path,
87                                   const std::string& expect_filesystem_id) {
88     EXPECT_TRUE(url.is_valid());
89 
90     EXPECT_EQ(expect_origin, url.origin());
91     EXPECT_EQ(expect_mount_type, url.mount_type());
92     EXPECT_EQ(expect_type, url.type());
93     EXPECT_EQ(expect_path, url.path());
94     EXPECT_EQ(expect_virtual_path, url.virtual_path());
95     EXPECT_EQ(expect_filesystem_id, url.filesystem_id());
96   }
97 
98  private:
99   base::ScopedTempDir data_dir_;
100   base::MessageLoop message_loop_;
101   scoped_refptr<storage::SpecialStoragePolicy> storage_policy_;
102   scoped_refptr<MockQuotaManager> mock_quota_manager_;
103 };
104 
105 // It is not valid to pass NULL ExternalMountPoints to FileSystemContext on
106 // ChromeOS.
107 #if !defined(OS_CHROMEOS)
TEST_F(FileSystemContextTest,NullExternalMountPoints)108 TEST_F(FileSystemContextTest, NullExternalMountPoints) {
109   scoped_refptr<FileSystemContext> file_system_context(
110       CreateFileSystemContextForTest(NULL));
111 
112   // Cracking system external mount and isolated mount points should work.
113   std::string isolated_name = "root";
114   std::string isolated_id =
115       IsolatedContext::GetInstance()->RegisterFileSystemForPath(
116           storage::kFileSystemTypeNativeLocal,
117           std::string(),
118           base::FilePath(DRIVE FPL("/test/isolated/root")),
119           &isolated_name);
120   // Register system external mount point.
121   ASSERT_TRUE(ExternalMountPoints::GetSystemInstance()->RegisterFileSystem(
122       "system",
123       storage::kFileSystemTypeNativeLocal,
124       FileSystemMountOption(),
125       base::FilePath(DRIVE FPL("/test/sys/"))));
126 
127   FileSystemURL cracked_isolated = file_system_context->CrackURL(
128       CreateRawFileSystemURL("isolated", isolated_id));
129 
130   ExpectFileSystemURLMatches(
131       cracked_isolated,
132       GURL(kTestOrigin),
133       storage::kFileSystemTypeIsolated,
134       storage::kFileSystemTypeNativeLocal,
135       base::FilePath(DRIVE FPL("/test/isolated/root/file"))
136           .NormalizePathSeparators(),
137       base::FilePath::FromUTF8Unsafe(isolated_id)
138           .Append(FPL("root/file"))
139           .NormalizePathSeparators(),
140       isolated_id);
141 
142   FileSystemURL cracked_external = file_system_context->CrackURL(
143       CreateRawFileSystemURL("external", "system"));
144 
145   ExpectFileSystemURLMatches(
146       cracked_external,
147       GURL(kTestOrigin),
148       storage::kFileSystemTypeExternal,
149       storage::kFileSystemTypeNativeLocal,
150       base::FilePath(DRIVE FPL("/test/sys/root/file"))
151           .NormalizePathSeparators(),
152       base::FilePath(FPL("system/root/file")).NormalizePathSeparators(),
153       "system");
154 
155   IsolatedContext::GetInstance()->RevokeFileSystem(isolated_id);
156   ExternalMountPoints::GetSystemInstance()->RevokeFileSystem("system");
157 }
158 #endif  // !defiend(OS_CHROMEOS)
159 
TEST_F(FileSystemContextTest,FileSystemContextKeepsMountPointsAlive)160 TEST_F(FileSystemContextTest, FileSystemContextKeepsMountPointsAlive) {
161   scoped_refptr<ExternalMountPoints> mount_points =
162       ExternalMountPoints::CreateRefCounted();
163 
164   // Register system external mount point.
165   ASSERT_TRUE(mount_points->RegisterFileSystem(
166       "system",
167       storage::kFileSystemTypeNativeLocal,
168       FileSystemMountOption(),
169       base::FilePath(DRIVE FPL("/test/sys/"))));
170 
171   scoped_refptr<FileSystemContext> file_system_context(
172       CreateFileSystemContextForTest(mount_points.get()));
173 
174   // Release a MountPoints reference created in the test.
175   mount_points = NULL;
176 
177   // FileSystemContext should keep a reference to the |mount_points|, so it
178   // should be able to resolve the URL.
179   FileSystemURL cracked_external = file_system_context->CrackURL(
180       CreateRawFileSystemURL("external", "system"));
181 
182   ExpectFileSystemURLMatches(
183       cracked_external,
184       GURL(kTestOrigin),
185       storage::kFileSystemTypeExternal,
186       storage::kFileSystemTypeNativeLocal,
187       base::FilePath(DRIVE FPL("/test/sys/root/file"))
188           .NormalizePathSeparators(),
189       base::FilePath(FPL("system/root/file")).NormalizePathSeparators(),
190       "system");
191 
192   // No need to revoke the registered filesystem since |mount_points| lifetime
193   // is bound to this test.
194 }
195 
TEST_F(FileSystemContextTest,CrackFileSystemURL)196 TEST_F(FileSystemContextTest, CrackFileSystemURL) {
197   scoped_refptr<ExternalMountPoints> external_mount_points(
198       ExternalMountPoints::CreateRefCounted());
199   scoped_refptr<FileSystemContext> file_system_context(
200       CreateFileSystemContextForTest(external_mount_points.get()));
201 
202   // Register an isolated mount point.
203   std::string isolated_file_system_name = "root";
204   const std::string kIsolatedFileSystemID =
205       IsolatedContext::GetInstance()->RegisterFileSystemForPath(
206           storage::kFileSystemTypeNativeLocal,
207           std::string(),
208           base::FilePath(DRIVE FPL("/test/isolated/root")),
209           &isolated_file_system_name);
210   // Register system external mount point.
211   ASSERT_TRUE(ExternalMountPoints::GetSystemInstance()->RegisterFileSystem(
212       "system",
213       storage::kFileSystemTypeDrive,
214       FileSystemMountOption(),
215       base::FilePath(DRIVE FPL("/test/sys/"))));
216   ASSERT_TRUE(ExternalMountPoints::GetSystemInstance()->RegisterFileSystem(
217       "ext",
218       storage::kFileSystemTypeNativeLocal,
219       FileSystemMountOption(),
220       base::FilePath(DRIVE FPL("/test/ext"))));
221   // Register a system external mount point with the same name/id as the
222   // registered isolated mount point.
223   ASSERT_TRUE(ExternalMountPoints::GetSystemInstance()->RegisterFileSystem(
224       kIsolatedFileSystemID,
225       storage::kFileSystemTypeRestrictedNativeLocal,
226       FileSystemMountOption(),
227       base::FilePath(DRIVE FPL("/test/system/isolated"))));
228   // Add a mount points with the same name as a system mount point to
229   // FileSystemContext's external mount points.
230   ASSERT_TRUE(external_mount_points->RegisterFileSystem(
231       "ext",
232       storage::kFileSystemTypeNativeLocal,
233       FileSystemMountOption(),
234       base::FilePath(DRIVE FPL("/test/local/ext/"))));
235 
236   const GURL kTestOrigin = GURL("http://chromium.org/");
237   const base::FilePath kVirtualPathNoRoot = base::FilePath(FPL("root/file"));
238 
239   struct TestCase {
240     // Test case values.
241     std::string root;
242     std::string type_str;
243 
244     // Expected test results.
245     bool expect_is_valid;
246     storage::FileSystemType expect_mount_type;
247     storage::FileSystemType expect_type;
248     const base::FilePath::CharType* expect_path;
249     std::string expect_filesystem_id;
250   };
251 
252   const TestCase kTestCases[] = {
253       // Following should not be handled by the url crackers:
254       {
255        "pers_mount", "persistent", true /* is_valid */,
256        storage::kFileSystemTypePersistent, storage::kFileSystemTypePersistent,
257        FPL("pers_mount/root/file"), std::string() /* filesystem id */
258       },
259       {
260        "temp_mount", "temporary", true /* is_valid */,
261        storage::kFileSystemTypeTemporary, storage::kFileSystemTypeTemporary,
262        FPL("temp_mount/root/file"), std::string() /* filesystem id */
263       },
264       // Should be cracked by isolated mount points:
265       {kIsolatedFileSystemID, "isolated", true /* is_valid */,
266        storage::kFileSystemTypeIsolated, storage::kFileSystemTypeNativeLocal,
267        DRIVE FPL("/test/isolated/root/file"), kIsolatedFileSystemID},
268       // Should be cracked by system mount points:
269       {"system", "external", true /* is_valid */,
270        storage::kFileSystemTypeExternal, storage::kFileSystemTypeDrive,
271        DRIVE FPL("/test/sys/root/file"), "system"},
272       {kIsolatedFileSystemID, "external", true /* is_valid */,
273        storage::kFileSystemTypeExternal,
274        storage::kFileSystemTypeRestrictedNativeLocal,
275        DRIVE FPL("/test/system/isolated/root/file"), kIsolatedFileSystemID},
276       // Should be cracked by FileSystemContext's ExternalMountPoints.
277       {"ext", "external", true /* is_valid */, storage::kFileSystemTypeExternal,
278        storage::kFileSystemTypeNativeLocal,
279        DRIVE FPL("/test/local/ext/root/file"), "ext"},
280       // Test for invalid filesystem url (made invalid by adding invalid
281       // filesystem type).
282       {"sytem", "external", false /* is_valid */,
283        // The rest of values will be ignored.
284        storage::kFileSystemTypeUnknown, storage::kFileSystemTypeUnknown,
285        FPL(""), std::string()},
286       // Test for URL with non-existing filesystem id.
287       {"invalid", "external", false /* is_valid */,
288        // The rest of values will be ignored.
289        storage::kFileSystemTypeUnknown, storage::kFileSystemTypeUnknown,
290        FPL(""), std::string()},
291   };
292 
293   for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kTestCases); ++i) {
294     const base::FilePath virtual_path =
295         base::FilePath::FromUTF8Unsafe(
296             kTestCases[i].root).Append(kVirtualPathNoRoot);
297 
298     GURL raw_url =
299         CreateRawFileSystemURL(kTestCases[i].type_str, kTestCases[i].root);
300     FileSystemURL cracked_url = file_system_context->CrackURL(raw_url);
301 
302     SCOPED_TRACE(testing::Message() << "Test case " << i << ": "
303                                     << "Cracking URL: " << raw_url);
304 
305     EXPECT_EQ(kTestCases[i].expect_is_valid, cracked_url.is_valid());
306     if (!kTestCases[i].expect_is_valid)
307       continue;
308 
309     ExpectFileSystemURLMatches(
310         cracked_url,
311         GURL(kTestOrigin),
312         kTestCases[i].expect_mount_type,
313         kTestCases[i].expect_type,
314         base::FilePath(kTestCases[i].expect_path).NormalizePathSeparators(),
315         virtual_path.NormalizePathSeparators(),
316         kTestCases[i].expect_filesystem_id);
317   }
318 
319   IsolatedContext::GetInstance()->RevokeFileSystemByPath(
320       base::FilePath(DRIVE FPL("/test/isolated/root")));
321   ExternalMountPoints::GetSystemInstance()->RevokeFileSystem("system");
322   ExternalMountPoints::GetSystemInstance()->RevokeFileSystem("ext");
323   ExternalMountPoints::GetSystemInstance()->RevokeFileSystem(
324       kIsolatedFileSystemID);
325 }
326 
TEST_F(FileSystemContextTest,CanServeURLRequest)327 TEST_F(FileSystemContextTest, CanServeURLRequest) {
328   scoped_refptr<ExternalMountPoints> external_mount_points(
329       ExternalMountPoints::CreateRefCounted());
330   scoped_refptr<FileSystemContext> context(
331       CreateFileSystemContextForTest(external_mount_points.get()));
332 
333   // A request for a sandbox mount point should be served.
334   FileSystemURL cracked_url =
335       context->CrackURL(CreateRawFileSystemURL("persistent", "pers_mount"));
336   EXPECT_EQ(storage::kFileSystemTypePersistent, cracked_url.mount_type());
337   EXPECT_TRUE(context->CanServeURLRequest(cracked_url));
338 
339   // A request for an isolated mount point should NOT be served.
340   std::string isolated_fs_name = "root";
341   std::string isolated_fs_id =
342       IsolatedContext::GetInstance()->RegisterFileSystemForPath(
343           storage::kFileSystemTypeNativeLocal,
344           std::string(),
345           base::FilePath(DRIVE FPL("/test/isolated/root")),
346           &isolated_fs_name);
347   cracked_url = context->CrackURL(
348       CreateRawFileSystemURL("isolated", isolated_fs_id));
349   EXPECT_EQ(storage::kFileSystemTypeIsolated, cracked_url.mount_type());
350   EXPECT_FALSE(context->CanServeURLRequest(cracked_url));
351 
352   // A request for an external mount point should be served.
353   const std::string kExternalMountName = "ext_mount";
354   ASSERT_TRUE(ExternalMountPoints::GetSystemInstance()->RegisterFileSystem(
355       kExternalMountName,
356       storage::kFileSystemTypeDrive,
357       FileSystemMountOption(),
358       base::FilePath()));
359   cracked_url = context->CrackURL(
360       CreateRawFileSystemURL("external", kExternalMountName));
361   EXPECT_EQ(storage::kFileSystemTypeExternal, cracked_url.mount_type());
362   EXPECT_TRUE(context->CanServeURLRequest(cracked_url));
363 
364   ExternalMountPoints::GetSystemInstance()->RevokeFileSystem(
365       kExternalMountName);
366   IsolatedContext::GetInstance()->RevokeFileSystem(isolated_fs_id);
367 }
368 
369 }  // namespace
370 
371 }  // namespace content
372