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

Side by Side Diff: apps/saved_files_service.cc

Issue 14607023: Add support for persistent file access in apps. (Closed) Base URL: http://git.chromium.org/chromium/src.git@master
Patch Set: rebase Created 7 years, 7 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 unified diff | Download patch
« no previous file with comments | « apps/saved_files_service.h ('k') | apps/saved_files_service_factory.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
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 "apps/saved_files_service.h"
6
7 #include <algorithm>
8
9 #include "apps/saved_files_service_factory.h"
10 #include "base/basictypes.h"
11 #include "base/hash_tables.h"
12 #include "base/value_conversions.h"
13 #include "chrome/browser/extensions/extension_host.h"
14 #include "chrome/browser/extensions/extension_prefs.h"
15 #include "chrome/browser/extensions/extension_service.h"
16 #include "chrome/browser/extensions/extension_system.h"
17 #include "chrome/common/extensions/permissions/api_permission.h"
18 #include "chrome/common/extensions/permissions/permission_set.h"
19
20 namespace apps {
21
22 using extensions::APIPermission;
23 using extensions::Extension;
24 using extensions::ExtensionHost;
25 using extensions::ExtensionPrefs;
26
27 namespace {
28
29 // Preference keys
30
31 // The file entries that the app has permission to access.
32 const char kFileEntries[] = "file_entries";
33
34 // The path to a file entry that the app had permission to access.
35 const char kFileEntryPath[] = "path";
36
37 // Whether or not the app had write access to a file entry.
38 const char kFileEntryWritable[] = "writable";
39
40 // The sequence number in the LRU of the file entry.
41 const char kFileEntrySequenceNumber[] = "sequence_number";
42
43 const size_t kMaxSavedFileEntries = 500;
44 const int kMaxSequenceNumber = kint32max;
45
46 // These might be different to the constant values in tests.
47 size_t g_max_saved_file_entries = kMaxSavedFileEntries;
48 int g_max_sequence_number = kMaxSequenceNumber;
49
50 // Persists a SavedFileEntry in ExtensionPrefs.
51 void AddSavedFileEntry(ExtensionPrefs* prefs,
52 const std::string& extension_id,
53 const SavedFileEntry& file_entry) {
54 ExtensionPrefs::ScopedDictionaryUpdate update(
55 prefs, extension_id, kFileEntries);
56 DictionaryValue* file_entries = update.Get();
57 if (!file_entries)
58 file_entries = update.Create();
59 DCHECK(!file_entries->GetDictionaryWithoutPathExpansion(file_entry.id, NULL));
60
61 DictionaryValue* file_entry_dict = new DictionaryValue();
62 file_entry_dict->Set(kFileEntryPath, CreateFilePathValue(file_entry.path));
63 file_entry_dict->SetBoolean(kFileEntryWritable, file_entry.writable);
64 file_entry_dict->SetInteger(kFileEntrySequenceNumber,
65 file_entry.sequence_number);
66 file_entries->SetWithoutPathExpansion(file_entry.id, file_entry_dict);
67 }
68
69 // Updates the sequence_number of a SavedFileEntry persisted in ExtensionPrefs.
70 void UpdateSavedFileEntry(ExtensionPrefs* prefs,
71 const std::string& extension_id,
72 const SavedFileEntry& file_entry) {
73 ExtensionPrefs::ScopedDictionaryUpdate update(
74 prefs, extension_id, kFileEntries);
75 DictionaryValue* file_entries = update.Get();
76 DCHECK(file_entries);
77 DictionaryValue* file_entry_dict = NULL;
78 file_entries->GetDictionaryWithoutPathExpansion(file_entry.id,
79 &file_entry_dict);
80 DCHECK(file_entry_dict);
81 file_entry_dict->SetInteger(kFileEntrySequenceNumber,
82 file_entry.sequence_number);
83 }
84
85 // Removes a SavedFileEntry from ExtensionPrefs.
86 void RemoveSavedFileEntry(ExtensionPrefs* prefs,
87 const std::string& extension_id,
88 const std::string& file_entry_id) {
89 ExtensionPrefs::ScopedDictionaryUpdate update(
90 prefs, extension_id, kFileEntries);
91 DictionaryValue* file_entries = update.Get();
92 if (!file_entries)
93 file_entries = update.Create();
94 file_entries->RemoveWithoutPathExpansion(file_entry_id, NULL);
95 }
96
97 // Clears all SavedFileEntry for the app from ExtensionPrefs.
98 void ClearSavedFileEntries(ExtensionPrefs* prefs,
99 const std::string& extension_id) {
100 prefs->UpdateExtensionPref(extension_id, kFileEntries, NULL);
101 }
102
103 // Returns all SavedFileEntries for the app.
104 std::vector<SavedFileEntry> GetSavedFileEntries(
105 ExtensionPrefs* prefs,
106 const std::string& extension_id) {
107 std::vector<SavedFileEntry> result;
108 const DictionaryValue* file_entries = NULL;
109 if (!prefs->ReadPrefAsDictionary(extension_id, kFileEntries, &file_entries))
110 return result;
111
112 for (DictionaryValue::Iterator it(*file_entries); !it.IsAtEnd();
113 it.Advance()) {
114 const DictionaryValue* file_entry = NULL;
115 if (!it.value().GetAsDictionary(&file_entry))
116 continue;
117 const base::Value* path_value;
118 if (!file_entry->Get(kFileEntryPath, &path_value))
119 continue;
120 base::FilePath file_path;
121 if (!GetValueAsFilePath(*path_value, &file_path))
122 continue;
123 bool writable = false;
124 if (!file_entry->GetBoolean(kFileEntryWritable, &writable))
125 continue;
126 int sequence_number = 0;
127 if (!file_entry->GetInteger(kFileEntrySequenceNumber, &sequence_number))
128 continue;
129 if (!sequence_number)
130 continue;
131 result.push_back(
132 SavedFileEntry(it.key(), file_path, writable, sequence_number));
133 }
134 return result;
135 }
136
137 } // namespace
138
139 SavedFileEntry::SavedFileEntry() : writable(false), sequence_number(0) {}
140
141 SavedFileEntry::SavedFileEntry(const std::string& id,
142 const base::FilePath& path,
143 bool writable,
144 int sequence_number)
145 : id(id),
146 path(path),
147 writable(writable),
148 sequence_number(sequence_number) {}
149
150 class SavedFilesService::SavedFiles {
151 public:
152 SavedFiles(Profile* profile, const std::string& extension_id);
153 ~SavedFiles();
154
155 void RegisterFileEntry(const std::string& id,
156 const base::FilePath& file_path,
157 bool writable);
158 void EnqueueFileEntry(const std::string& id);
159 bool IsRegistered(const std::string& id) const;
160 const SavedFileEntry* GetFileEntry(const std::string& id) const;
161 std::vector<SavedFileEntry> GetAllFileEntries() const;
162
163 private:
164 // Compacts sequence numbers if the largest sequence number is
165 // g_max_sequence_number. Outside of testing, it is set to kint32max, so this
166 // will almost never do any real work.
167 void MaybeCompactSequenceNumbers();
168
169 void LoadSavedFileEntriesFromPreferences();
170
171 Profile* profile_;
172 const std::string extension_id_;
173
174 // Contains all file entries that have been registered, keyed by ID. Owns
175 // values.
176 base::hash_map<std::string, SavedFileEntry*> registered_file_entries_;
177 STLValueDeleter<base::hash_map<std::string, SavedFileEntry*> >
178 registered_file_entries_deleter_;
179
180 // The queue of file entries that have been retained, keyed by
181 // sequence_number. Values are a subset of values in registered_file_entries_.
182 // This should be kept in sync with file entries stored in extension prefs.
183 std::map<int, SavedFileEntry*> saved_file_lru_;
184
185 DISALLOW_COPY_AND_ASSIGN(SavedFiles);
186 };
187
188 // static
189 SavedFilesService* SavedFilesService::Get(Profile* profile) {
190 return SavedFilesServiceFactory::GetForProfile(profile);
191 }
192
193 SavedFilesService::SavedFilesService(Profile* profile)
194 : extension_id_to_saved_files_deleter_(&extension_id_to_saved_files_),
195 profile_(profile) {
196 registrar_.Add(this,
197 chrome::NOTIFICATION_EXTENSION_HOST_DESTROYED,
198 content::NotificationService::AllSources());
199 registrar_.Add(this,
200 chrome::NOTIFICATION_APP_TERMINATING,
201 content::NotificationService::AllSources());
202 }
203
204 SavedFilesService::~SavedFilesService() {}
205
206 void SavedFilesService::Observe(int type,
207 const content::NotificationSource& source,
208 const content::NotificationDetails& details) {
209 switch (type) {
210 case chrome::NOTIFICATION_EXTENSION_HOST_DESTROYED: {
211 ExtensionHost* host = content::Details<ExtensionHost>(details).ptr();
212 const Extension* extension = host->extension();
213 if (extension) {
214 ClearQueueIfNoRetainPermission(extension);
215 Clear(extension->id());
216 }
217 break;
218 }
219
220 case chrome::NOTIFICATION_APP_TERMINATING: {
221 // Stop listening to NOTIFICATION_EXTENSION_HOST_DESTROYED in particular
222 // as all extension hosts will be destroyed as a result of shutdown.
223 registrar_.RemoveAll();
224 break;
225 }
226 }
227 }
228
229 void SavedFilesService::RegisterFileEntry(const std::string& extension_id,
230 const std::string& id,
231 const base::FilePath& file_path,
232 bool writable) {
233 GetOrInsert(extension_id)->RegisterFileEntry(id, file_path, writable);
234 }
235
236 void SavedFilesService::EnqueueFileEntry(const std::string& extension_id,
237 const std::string& id) {
238 GetOrInsert(extension_id)->EnqueueFileEntry(id);
239 }
240
241 std::vector<SavedFileEntry> SavedFilesService::GetAllFileEntries(
242 const std::string& extension_id) {
243 return GetOrInsert(extension_id)->GetAllFileEntries();
244 }
245
246 bool SavedFilesService::IsRegistered(const std::string& extension_id,
247 const std::string& id) {
248 return GetOrInsert(extension_id)->IsRegistered(id);
249 }
250
251 const SavedFileEntry* SavedFilesService::GetFileEntry(
252 const std::string& extension_id,
253 const std::string& id) {
254 return GetOrInsert(extension_id)->GetFileEntry(id);
255 }
256
257 void SavedFilesService::ClearQueueIfNoRetainPermission(
258 const Extension* extension) {
259 if (!extension->GetActivePermissions()->HasAPIPermission(
260 APIPermission::kFileSystemRetainFiles)) {
261 ClearSavedFileEntries(ExtensionPrefs::Get(profile_), extension->id());
262 Clear(extension->id());
263 }
264 }
265
266 SavedFilesService::SavedFiles* SavedFilesService::GetOrInsert(
267 const std::string& extension_id) {
268 std::map<std::string, SavedFiles*>::iterator it =
269 extension_id_to_saved_files_.find(extension_id);
270 if (it != extension_id_to_saved_files_.end())
271 return it->second;
272
273 SavedFiles* saved_files = new SavedFiles(profile_, extension_id);
274 extension_id_to_saved_files_.insert(
275 std::make_pair(extension_id, saved_files));
276 return saved_files;
277 }
278
279 void SavedFilesService::Clear(const std::string& extension_id) {
280 std::map<std::string, SavedFiles*>::iterator it =
281 extension_id_to_saved_files_.find(extension_id);
282 if (it != extension_id_to_saved_files_.end()) {
283 delete it->second;
284 extension_id_to_saved_files_.erase(it);
285 }
286 }
287
288 SavedFilesService::SavedFiles::SavedFiles(Profile* profile,
289 const std::string& extension_id)
290 : profile_(profile),
291 extension_id_(extension_id),
292 registered_file_entries_deleter_(&registered_file_entries_) {
293 LoadSavedFileEntriesFromPreferences();
294 }
295
296 SavedFilesService::SavedFiles::~SavedFiles() {}
297
298 void SavedFilesService::SavedFiles::RegisterFileEntry(
299 const std::string& id,
300 const base::FilePath& file_path,
301 bool writable) {
302 if (ContainsKey(registered_file_entries_, id))
303 return;
304
305 registered_file_entries_.insert(
306 std::make_pair(id, new SavedFileEntry(id, file_path, writable, 0)));
307 }
308
309 void SavedFilesService::SavedFiles::EnqueueFileEntry(const std::string& id) {
310 base::hash_map<std::string, SavedFileEntry*>::iterator it =
311 registered_file_entries_.find(id);
312 DCHECK(it != registered_file_entries_.end());
313
314 SavedFileEntry* file_entry = it->second;
315 int old_sequence_number = file_entry->sequence_number;
316 if (!saved_file_lru_.empty()) {
317 // Get the sequence number after the last file entry in the LRU.
318 std::map<int, SavedFileEntry*>::reverse_iterator it =
319 saved_file_lru_.rbegin();
320 if (it->second == file_entry)
321 return;
322
323 file_entry->sequence_number = it->first + 1;
324 } else {
325 // The first sequence number is 1, as 0 means the entry is not in the LRU.
326 file_entry->sequence_number = 1;
327 }
328 saved_file_lru_.insert(
329 std::make_pair(file_entry->sequence_number, file_entry));
330 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
331 if (old_sequence_number) {
332 saved_file_lru_.erase(old_sequence_number);
333 UpdateSavedFileEntry(prefs, extension_id_, *file_entry);
334 } else {
335 AddSavedFileEntry(prefs, extension_id_, *file_entry);
336 if (saved_file_lru_.size() > g_max_saved_file_entries) {
337 std::map<int, SavedFileEntry*>::iterator it = saved_file_lru_.begin();
338 it->second->sequence_number = 0;
339 RemoveSavedFileEntry(prefs, extension_id_, it->second->id);
340 saved_file_lru_.erase(it);
341 }
342 }
343 MaybeCompactSequenceNumbers();
344 }
345
346 bool SavedFilesService::SavedFiles::IsRegistered(const std::string& id) const {
347 return ContainsKey(registered_file_entries_, id);
348 }
349
350 const SavedFileEntry* SavedFilesService::SavedFiles::GetFileEntry(
351 const std::string& id) const {
352 base::hash_map<std::string, SavedFileEntry*>::const_iterator it =
353 registered_file_entries_.find(id);
354 if (it == registered_file_entries_.end())
355 return NULL;
356
357 return it->second;
358 }
359
360 std::vector<SavedFileEntry> SavedFilesService::SavedFiles::GetAllFileEntries()
361 const {
362 std::vector<SavedFileEntry> result;
363 for (base::hash_map<std::string, SavedFileEntry*>::const_iterator it =
364 registered_file_entries_.begin();
365 it != registered_file_entries_.end();
366 ++it) {
367 result.push_back(*it->second);
368 }
369 return result;
370 }
371
372 void SavedFilesService::SavedFiles::MaybeCompactSequenceNumbers() {
373 DCHECK_GE(g_max_sequence_number, 0);
374 DCHECK_GE(static_cast<size_t>(g_max_sequence_number),
375 g_max_saved_file_entries);
376 std::map<int, SavedFileEntry*>::reverse_iterator it =
377 saved_file_lru_.rbegin();
378 if (it == saved_file_lru_.rend())
379 return;
380
381 // Only compact sequence numbers if the last entry's sequence number is the
382 // maximum value. This should almost never be the case.
383 if (it->first < g_max_sequence_number)
384 return;
385
386 int sequence_number = 0;
387 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
388 for (std::map<int, SavedFileEntry*>::iterator it = saved_file_lru_.begin();
389 it != saved_file_lru_.end();
390 ++it) {
391 sequence_number++;
392 if (it->second->sequence_number == sequence_number)
393 continue;
394
395 SavedFileEntry* file_entry = it->second;
396 file_entry->sequence_number = sequence_number;
397 UpdateSavedFileEntry(prefs, extension_id_, *file_entry);
398 saved_file_lru_.erase(it++);
399 // Provide the following element as an insert hint. While optimized
400 // insertion time with the following element as a hint is only supported by
401 // the spec in C++11, the implementations do support this.
402 it = saved_file_lru_.insert(
403 it, std::make_pair(file_entry->sequence_number, file_entry));
404 }
405 }
406
407 void SavedFilesService::SavedFiles::LoadSavedFileEntriesFromPreferences() {
408 ExtensionPrefs* prefs = ExtensionPrefs::Get(profile_);
409 std::vector<SavedFileEntry> saved_entries =
410 GetSavedFileEntries(prefs, extension_id_);
411 for (std::vector<SavedFileEntry>::iterator it = saved_entries.begin();
412 it != saved_entries.end();
413 ++it) {
414 SavedFileEntry* file_entry = new SavedFileEntry(*it);
415 registered_file_entries_.insert(std::make_pair(file_entry->id, file_entry));
416 saved_file_lru_.insert(
417 std::make_pair(file_entry->sequence_number, file_entry));
418 }
419 }
420
421 // static
422 void SavedFilesService::SetMaxSequenceNumberForTest(int max_value) {
423 g_max_sequence_number = max_value;
424 }
425
426 // static
427 void SavedFilesService::ClearMaxSequenceNumberForTest() {
428 g_max_sequence_number = kMaxSequenceNumber;
429 }
430
431 // static
432 void SavedFilesService::SetLruSizeForTest(int size) {
433 g_max_saved_file_entries = size;
434 }
435
436 // static
437 void SavedFilesService::ClearLruSizeForTest() {
438 g_max_saved_file_entries = kMaxSavedFileEntries;
439 }
440
441 } // namespace apps
OLDNEW
« no previous file with comments | « apps/saved_files_service.h ('k') | apps/saved_files_service_factory.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698