2 # Copyright 2013 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
9 from appengine_wrappers
import GetAppVersion
10 from compiled_file_system
import CompiledFileSystem
11 from copy
import deepcopy
12 from file_system
import FileNotFoundError
13 from mock_file_system
import MockFileSystem
14 from object_store_creator
import ObjectStoreCreator
15 from test_file_system
import TestFileSystem
16 from test_object_store
import TestObjectStore
20 '404.html': '404.html contents',
22 'a11y.html': 'a11y.html contents',
23 'about_apps.html': 'about_apps.html contents',
25 'file.html': 'file.html contents'
28 'deepfile.html': 'deepfile.html contents',
30 'deepest.html': 'deepest.html contents',
35 'activeTab.html': 'activeTab.html contents',
36 'alarms.html': 'alarms.html contents'
40 identity
= lambda _
, x
: x
42 def _GetTestCompiledFsCreator():
43 '''Returns a function which creates CompiledFileSystem views of
44 TestFileSystems backed by _TEST_DATA.
46 return functools
.partial(
47 CompiledFileSystem
.Factory(
48 ObjectStoreCreator(start_empty
=False,
49 store_type
=TestObjectStore
,
50 disable_wrappers
=True),
52 TestFileSystem(deepcopy(_TEST_DATA
)))
54 class CompiledFileSystemTest(unittest
.TestCase
):
55 def testPopulateNamespace(self
):
56 def CheckNamespace(expected_file
, expected_list
, fs
):
57 self
.assertEqual(expected_file
, fs
._file
_object
_store
.namespace
)
58 self
.assertEqual(expected_list
, fs
._list
_object
_store
.namespace
)
59 compiled_fs_creator
= _GetTestCompiledFsCreator()
62 'class=CompiledFileSystem&'
63 'category=CompiledFileSystemTest/TestFileSystem/file&'
64 'app_version=%s' % GetAppVersion(),
65 'class=CompiledFileSystem&'
66 'category=CompiledFileSystemTest/TestFileSystem/list&'
67 'app_version=%s' % GetAppVersion(),
68 compiled_fs_creator(f
, CompiledFileSystemTest
))
70 'class=CompiledFileSystem&'
71 'category=CompiledFileSystemTest/TestFileSystem/foo/file&'
72 'app_version=%s' % GetAppVersion(),
73 'class=CompiledFileSystem&'
74 'category=CompiledFileSystemTest/TestFileSystem/foo/list&'
75 'app_version=%s' % GetAppVersion(),
76 compiled_fs_creator(f
, CompiledFileSystemTest
, category
='foo'))
78 def testPopulateFromFile(self
):
80 return '%s%s' % ('Z' * len(key
), 'z' * len(val
))
81 compiled_fs
= _GetTestCompiledFsCreator()(Sleepy
, CompiledFileSystemTest
)
82 self
.assertEqual('ZZZZZZZZzzzzzzzzzzzzzzzzz',
83 compiled_fs
.GetFromFile('404.html').Get())
84 self
.assertEqual('ZZZZZZZZZZZZZZzzzzzzzzzzzzzzzzzz',
85 compiled_fs
.GetFromFile('apps/a11y.html').Get())
86 self
.assertEqual('ZZZZZZZZZZZZZZZZZZZZZZzzzzzzzzzzzzzzzzzz',
87 compiled_fs
.GetFromFile('apps/fakedir/file.html').Get())
89 def testPopulateFromFileListing(self
):
90 def strip_ext(path
, files
):
91 return [os
.path
.splitext(f
)[0] for f
in files
]
92 compiled_fs
= _GetTestCompiledFsCreator()(strip_ext
, CompiledFileSystemTest
)
93 expected_top_listing
= [
97 'apps/deepdir/deeper/deepest',
98 'apps/deepdir/deepfile',
100 'extensions/activeTab',
103 self
.assertEqual(expected_top_listing
,
104 sorted(compiled_fs
.GetFromFileListing('').Get()))
105 expected_apps_listing
= [
108 'deepdir/deeper/deepest',
112 self
.assertEqual(expected_apps_listing
,
113 sorted(compiled_fs
.GetFromFileListing('apps/').Get()))
114 self
.assertEqual(['file',],
115 compiled_fs
.GetFromFileListing('apps/fakedir/').Get())
116 self
.assertEqual(['deeper/deepest', 'deepfile'],
117 sorted(compiled_fs
.GetFromFileListing(
118 'apps/deepdir/').Get()))
119 self
.assertEqual(['deepest'],
120 compiled_fs
.GetFromFileListing(
121 'apps/deepdir/deeper/').Get())
123 def testCaching(self
):
124 compiled_fs
= _GetTestCompiledFsCreator()(identity
, CompiledFileSystemTest
)
125 self
.assertEqual('404.html contents',
126 compiled_fs
.GetFromFile('404.html').Get())
127 self
.assertEqual(set(('file.html',)),
128 set(compiled_fs
.GetFromFileListing('apps/fakedir/').Get()))
130 compiled_fs
._file
_system
._path
_values
['404.html'] = 'boom'
131 compiled_fs
._file
_system
._path
_values
['apps/fakedir/'] = [
132 'file.html', 'boom.html']
133 self
.assertEqual('404.html contents',
134 compiled_fs
.GetFromFile('404.html').Get())
135 self
.assertEqual(set(('file.html',)),
136 set(compiled_fs
.GetFromFileListing('apps/fakedir/').Get()))
138 compiled_fs
._file
_system
.IncrementStat()
139 self
.assertEqual('boom', compiled_fs
.GetFromFile('404.html').Get())
140 self
.assertEqual(set(('file.html', 'boom.html')),
141 set(compiled_fs
.GetFromFileListing('apps/fakedir/').Get()))
143 def testFailures(self
):
144 compiled_fs
= _GetTestCompiledFsCreator()(identity
, CompiledFileSystemTest
)
145 self
.assertRaises(FileNotFoundError
,
146 compiled_fs
.GetFromFile('405.html').Get
)
147 # TODO(kalman): would be nice to test this fails since apps/ is a dir.
148 compiled_fs
.GetFromFile('apps')
149 #self.assertRaises(SomeError, compiled_fs.GetFromFile, 'apps/')
150 self
.assertRaises(FileNotFoundError
,
151 compiled_fs
.GetFromFileListing('nodir/').Get
)
152 # TODO(kalman): likewise, not a FileNotFoundError.
153 self
.assertRaises(FileNotFoundError
,
154 compiled_fs
.GetFromFileListing('404.html/').Get
)
156 def testCorrectFutureBehaviour(self
):
157 # Tests that the underlying FileSystem's Read Future has had Get() called
158 # on it before the Future is resolved, but the underlying Future isn't
159 # resolved until Get is.
160 mock_fs
= MockFileSystem(TestFileSystem(_TEST_DATA
))
161 compiled_fs
= CompiledFileSystem
.Factory(
162 ObjectStoreCreator
.ForTest()).Create(
163 mock_fs
, lambda path
, contents
: contents
, type(self
))
165 self
.assertTrue(*mock_fs
.CheckAndReset())
166 future
= compiled_fs
.GetFromFile('404.html')
167 self
.assertTrue(*mock_fs
.CheckAndReset(stat_count
=1, read_count
=1))
169 self
.assertTrue(*mock_fs
.CheckAndReset(read_resolve_count
=1))
171 future
= compiled_fs
.GetFromFileListing('apps/')
172 # Current behaviour is to have read=2 and read_resolve=1 because the first
173 # level is read eagerly, then all of the second is read (in parallel). If
174 # it weren't eager (and it may be worth experimenting with that) then it'd
175 # be read=1 and read_resolve=0.
176 self
.assertTrue(*mock_fs
.CheckAndReset(stat_count
=1,
178 read_resolve_count
=1))
180 # It's doing 1 more level 'deeper' (already read 'fakedir' and 'deepdir'
181 # though not resolved), so that's 1 more read/resolve + the resolve from
183 self
.assertTrue(*mock_fs
.CheckAndReset(read_count
=1, read_resolve_count
=2))
185 # Even though the directory is 1 layer deep the caller has no way of
186 # determining that ahead of time (though perhaps the API could give some
187 # kind of clue, if we really cared).
188 future
= compiled_fs
.GetFromFileListing('extensions/')
189 self
.assertTrue(*mock_fs
.CheckAndReset(stat_count
=1,
191 read_resolve_count
=1))
193 self
.assertTrue(*mock_fs
.CheckAndReset())
195 # Similar configuration to the 'apps/' case but deeper.
196 future
= compiled_fs
.GetFromFileListing('')
197 self
.assertTrue(*mock_fs
.CheckAndReset(stat_count
=1,
199 read_resolve_count
=1))
201 self
.assertTrue(*mock_fs
.CheckAndReset(read_count
=2, read_resolve_count
=3))
205 if __name__
== '__main__':