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.
5 #include "apps/saved_files_service.h"
10 #include "apps/saved_files_service_factory.h"
11 #include "base/basictypes.h"
12 #include "base/containers/scoped_ptr_hash_map.h"
13 #include "base/value_conversions.h"
14 #include "chrome/browser/chrome_notification_types.h"
15 #include "chrome/browser/profiles/profile.h"
16 #include "content/public/browser/notification_service.h"
17 #include "extensions/browser/extension_host.h"
18 #include "extensions/browser/extension_prefs.h"
19 #include "extensions/browser/extension_system.h"
20 #include "extensions/browser/extension_util.h"
21 #include "extensions/browser/notification_types.h"
22 #include "extensions/common/permissions/api_permission.h"
23 #include "extensions/common/permissions/permission_set.h"
24 #include "extensions/common/permissions/permissions_data.h"
28 using extensions::APIPermission
;
29 using extensions::Extension
;
30 using extensions::ExtensionHost
;
31 using extensions::ExtensionPrefs
;
37 // The file entries that the app has permission to access.
38 const char kFileEntries
[] = "file_entries";
40 // The path to a file entry that the app had permission to access.
41 const char kFileEntryPath
[] = "path";
43 // Whether or not the the entry refers to a directory.
44 const char kFileEntryIsDirectory
[] = "is_directory";
46 // The sequence number in the LRU of the file entry.
47 const char kFileEntrySequenceNumber
[] = "sequence_number";
49 const size_t kMaxSavedFileEntries
= 500;
50 const int kMaxSequenceNumber
= kint32max
;
52 // These might be different to the constant values in tests.
53 size_t g_max_saved_file_entries
= kMaxSavedFileEntries
;
54 int g_max_sequence_number
= kMaxSequenceNumber
;
56 // Persists a SavedFileEntry in ExtensionPrefs.
57 void AddSavedFileEntry(ExtensionPrefs
* prefs
,
58 const std::string
& extension_id
,
59 const SavedFileEntry
& file_entry
) {
60 ExtensionPrefs::ScopedDictionaryUpdate
update(
61 prefs
, extension_id
, kFileEntries
);
62 base::DictionaryValue
* file_entries
= update
.Get();
64 file_entries
= update
.Create();
65 DCHECK(!file_entries
->GetDictionaryWithoutPathExpansion(file_entry
.id
, NULL
));
67 base::DictionaryValue
* file_entry_dict
= new base::DictionaryValue();
68 file_entry_dict
->Set(kFileEntryPath
, CreateFilePathValue(file_entry
.path
));
69 file_entry_dict
->SetBoolean(kFileEntryIsDirectory
, file_entry
.is_directory
);
70 file_entry_dict
->SetInteger(kFileEntrySequenceNumber
,
71 file_entry
.sequence_number
);
72 file_entries
->SetWithoutPathExpansion(file_entry
.id
, file_entry_dict
);
75 // Updates the sequence_number of a SavedFileEntry persisted in ExtensionPrefs.
76 void UpdateSavedFileEntry(ExtensionPrefs
* prefs
,
77 const std::string
& extension_id
,
78 const SavedFileEntry
& file_entry
) {
79 ExtensionPrefs::ScopedDictionaryUpdate
update(
80 prefs
, extension_id
, kFileEntries
);
81 base::DictionaryValue
* file_entries
= update
.Get();
83 base::DictionaryValue
* file_entry_dict
= NULL
;
84 file_entries
->GetDictionaryWithoutPathExpansion(file_entry
.id
,
86 DCHECK(file_entry_dict
);
87 file_entry_dict
->SetInteger(kFileEntrySequenceNumber
,
88 file_entry
.sequence_number
);
91 // Removes a SavedFileEntry from ExtensionPrefs.
92 void RemoveSavedFileEntry(ExtensionPrefs
* prefs
,
93 const std::string
& extension_id
,
94 const std::string
& file_entry_id
) {
95 ExtensionPrefs::ScopedDictionaryUpdate
update(
96 prefs
, extension_id
, kFileEntries
);
97 base::DictionaryValue
* file_entries
= update
.Get();
99 file_entries
= update
.Create();
100 file_entries
->RemoveWithoutPathExpansion(file_entry_id
, NULL
);
103 // Clears all SavedFileEntry for the app from ExtensionPrefs.
104 void ClearSavedFileEntries(ExtensionPrefs
* prefs
,
105 const std::string
& extension_id
) {
106 prefs
->UpdateExtensionPref(extension_id
, kFileEntries
, NULL
);
109 // Returns all SavedFileEntries for the app.
110 std::vector
<SavedFileEntry
> GetSavedFileEntries(
111 ExtensionPrefs
* prefs
,
112 const std::string
& extension_id
) {
113 std::vector
<SavedFileEntry
> result
;
114 const base::DictionaryValue
* file_entries
= NULL
;
115 if (!prefs
->ReadPrefAsDictionary(extension_id
, kFileEntries
, &file_entries
))
118 for (base::DictionaryValue::Iterator
it(*file_entries
); !it
.IsAtEnd();
120 const base::DictionaryValue
* file_entry
= NULL
;
121 if (!it
.value().GetAsDictionary(&file_entry
))
123 const base::Value
* path_value
;
124 if (!file_entry
->Get(kFileEntryPath
, &path_value
))
126 base::FilePath file_path
;
127 if (!GetValueAsFilePath(*path_value
, &file_path
))
129 bool is_directory
= false;
130 file_entry
->GetBoolean(kFileEntryIsDirectory
, &is_directory
);
131 int sequence_number
= 0;
132 if (!file_entry
->GetInteger(kFileEntrySequenceNumber
, &sequence_number
))
134 if (!sequence_number
)
137 SavedFileEntry(it
.key(), file_path
, is_directory
, sequence_number
));
144 SavedFileEntry::SavedFileEntry() : is_directory(false), sequence_number(0) {}
146 SavedFileEntry::SavedFileEntry(const std::string
& id
,
147 const base::FilePath
& path
,
152 is_directory(is_directory
),
153 sequence_number(sequence_number
) {}
155 class SavedFilesService::SavedFiles
{
157 SavedFiles(Profile
* profile
, const std::string
& extension_id
);
160 void RegisterFileEntry(const std::string
& id
,
161 const base::FilePath
& file_path
,
163 void EnqueueFileEntry(const std::string
& id
);
164 bool IsRegistered(const std::string
& id
) const;
165 const SavedFileEntry
* GetFileEntry(const std::string
& id
) const;
166 std::vector
<SavedFileEntry
> GetAllFileEntries() const;
169 // Compacts sequence numbers if the largest sequence number is
170 // g_max_sequence_number. Outside of testing, it is set to kint32max, so this
171 // will almost never do any real work.
172 void MaybeCompactSequenceNumbers();
174 void LoadSavedFileEntriesFromPreferences();
177 const std::string extension_id_
;
179 // Contains all file entries that have been registered, keyed by ID. Owns
181 base::ScopedPtrHashMap
<std::string
, scoped_ptr
<SavedFileEntry
>>
182 registered_file_entries_
;
184 // The queue of file entries that have been retained, keyed by
185 // sequence_number. Values are a subset of values in registered_file_entries_.
186 // This should be kept in sync with file entries stored in extension prefs.
187 std::map
<int, SavedFileEntry
*> saved_file_lru_
;
189 DISALLOW_COPY_AND_ASSIGN(SavedFiles
);
193 SavedFilesService
* SavedFilesService::Get(Profile
* profile
) {
194 return SavedFilesServiceFactory::GetForProfile(profile
);
197 SavedFilesService::SavedFilesService(Profile
* profile
) : profile_(profile
) {
199 extensions::NOTIFICATION_EXTENSION_HOST_DESTROYED
,
200 content::NotificationService::AllSources());
202 chrome::NOTIFICATION_APP_TERMINATING
,
203 content::NotificationService::AllSources());
206 SavedFilesService::~SavedFilesService() {}
208 void SavedFilesService::Observe(int type
,
209 const content::NotificationSource
& source
,
210 const content::NotificationDetails
& details
) {
212 case extensions::NOTIFICATION_EXTENSION_HOST_DESTROYED
: {
213 ExtensionHost
* host
= content::Details
<ExtensionHost
>(details
).ptr();
214 const Extension
* extension
= host
->extension();
216 ClearQueueIfNoRetainPermission(extension
);
217 Clear(extension
->id());
222 case chrome::NOTIFICATION_APP_TERMINATING
: {
223 // Stop listening to NOTIFICATION_EXTENSION_HOST_DESTROYED in particular
224 // as all extension hosts will be destroyed as a result of shutdown.
225 registrar_
.RemoveAll();
231 void SavedFilesService::RegisterFileEntry(const std::string
& extension_id
,
232 const std::string
& id
,
233 const base::FilePath
& file_path
,
235 GetOrInsert(extension_id
)->RegisterFileEntry(id
, file_path
, is_directory
);
238 void SavedFilesService::EnqueueFileEntry(const std::string
& extension_id
,
239 const std::string
& id
) {
240 GetOrInsert(extension_id
)->EnqueueFileEntry(id
);
243 std::vector
<SavedFileEntry
> SavedFilesService::GetAllFileEntries(
244 const std::string
& extension_id
) {
245 SavedFiles
* saved_files
= Get(extension_id
);
247 return saved_files
->GetAllFileEntries();
248 return GetSavedFileEntries(ExtensionPrefs::Get(profile_
), extension_id
);
251 bool SavedFilesService::IsRegistered(const std::string
& extension_id
,
252 const std::string
& id
) {
253 return GetOrInsert(extension_id
)->IsRegistered(id
);
256 const SavedFileEntry
* SavedFilesService::GetFileEntry(
257 const std::string
& extension_id
,
258 const std::string
& id
) {
259 return GetOrInsert(extension_id
)->GetFileEntry(id
);
262 void SavedFilesService::ClearQueueIfNoRetainPermission(
263 const Extension
* extension
) {
264 if (extensions::util::IsEphemeralApp(extension
->id(), profile_
) ||
265 !extension
->permissions_data()->active_permissions()->HasAPIPermission(
266 APIPermission::kFileSystemRetainEntries
)) {
267 ClearQueue(extension
);
271 void SavedFilesService::ClearQueue(const extensions::Extension
* extension
) {
272 ClearSavedFileEntries(ExtensionPrefs::Get(profile_
), extension
->id());
273 Clear(extension
->id());
276 SavedFilesService::SavedFiles
* SavedFilesService::Get(
277 const std::string
& extension_id
) const {
278 base::ScopedPtrMap
<std::string
, scoped_ptr
<SavedFiles
>>::const_iterator it
=
279 extension_id_to_saved_files_
.find(extension_id
);
280 if (it
!= extension_id_to_saved_files_
.end())
286 SavedFilesService::SavedFiles
* SavedFilesService::GetOrInsert(
287 const std::string
& extension_id
) {
288 SavedFiles
* saved_files
= Get(extension_id
);
292 scoped_ptr
<SavedFiles
> scoped_saved_files(
293 new SavedFiles(profile_
, extension_id
));
294 saved_files
= scoped_saved_files
.get();
295 extension_id_to_saved_files_
.insert(extension_id
, scoped_saved_files
.Pass());
299 void SavedFilesService::Clear(const std::string
& extension_id
) {
300 extension_id_to_saved_files_
.erase(extension_id
);
303 SavedFilesService::SavedFiles::SavedFiles(Profile
* profile
,
304 const std::string
& extension_id
)
305 : profile_(profile
), extension_id_(extension_id
) {
306 LoadSavedFileEntriesFromPreferences();
309 SavedFilesService::SavedFiles::~SavedFiles() {}
311 void SavedFilesService::SavedFiles::RegisterFileEntry(
312 const std::string
& id
,
313 const base::FilePath
& file_path
,
315 if (ContainsKey(registered_file_entries_
, id
))
318 registered_file_entries_
.add(
319 id
, make_scoped_ptr(new SavedFileEntry(id
, file_path
, is_directory
, 0)));
322 void SavedFilesService::SavedFiles::EnqueueFileEntry(const std::string
& id
) {
323 auto it
= registered_file_entries_
.find(id
);
324 DCHECK(it
!= registered_file_entries_
.end());
326 SavedFileEntry
* file_entry
= it
->second
;
327 int old_sequence_number
= file_entry
->sequence_number
;
328 if (!saved_file_lru_
.empty()) {
329 // Get the sequence number after the last file entry in the LRU.
330 std::map
<int, SavedFileEntry
*>::reverse_iterator it
=
331 saved_file_lru_
.rbegin();
332 if (it
->second
== file_entry
)
335 file_entry
->sequence_number
= it
->first
+ 1;
337 // The first sequence number is 1, as 0 means the entry is not in the LRU.
338 file_entry
->sequence_number
= 1;
340 saved_file_lru_
.insert(
341 std::make_pair(file_entry
->sequence_number
, file_entry
));
342 ExtensionPrefs
* prefs
= ExtensionPrefs::Get(profile_
);
343 if (old_sequence_number
) {
344 saved_file_lru_
.erase(old_sequence_number
);
345 UpdateSavedFileEntry(prefs
, extension_id_
, *file_entry
);
347 AddSavedFileEntry(prefs
, extension_id_
, *file_entry
);
348 if (saved_file_lru_
.size() > g_max_saved_file_entries
) {
349 std::map
<int, SavedFileEntry
*>::iterator it
= saved_file_lru_
.begin();
350 it
->second
->sequence_number
= 0;
351 RemoveSavedFileEntry(prefs
, extension_id_
, it
->second
->id
);
352 saved_file_lru_
.erase(it
);
355 MaybeCompactSequenceNumbers();
358 bool SavedFilesService::SavedFiles::IsRegistered(const std::string
& id
) const {
359 return ContainsKey(registered_file_entries_
, id
);
362 const SavedFileEntry
* SavedFilesService::SavedFiles::GetFileEntry(
363 const std::string
& id
) const {
364 auto it
= registered_file_entries_
.find(id
);
365 if (it
== registered_file_entries_
.end())
371 std::vector
<SavedFileEntry
> SavedFilesService::SavedFiles::GetAllFileEntries()
373 std::vector
<SavedFileEntry
> result
;
374 for (auto it
= registered_file_entries_
.begin();
375 it
!= registered_file_entries_
.end(); ++it
) {
376 result
.push_back(*it
->second
);
381 void SavedFilesService::SavedFiles::MaybeCompactSequenceNumbers() {
382 DCHECK_GE(g_max_sequence_number
, 0);
383 DCHECK_GE(static_cast<size_t>(g_max_sequence_number
),
384 g_max_saved_file_entries
);
385 std::map
<int, SavedFileEntry
*>::reverse_iterator it
=
386 saved_file_lru_
.rbegin();
387 if (it
== saved_file_lru_
.rend())
390 // Only compact sequence numbers if the last entry's sequence number is the
391 // maximum value. This should almost never be the case.
392 if (it
->first
< g_max_sequence_number
)
395 int sequence_number
= 0;
396 ExtensionPrefs
* prefs
= ExtensionPrefs::Get(profile_
);
397 for (std::map
<int, SavedFileEntry
*>::iterator it
= saved_file_lru_
.begin();
398 it
!= saved_file_lru_
.end();
401 if (it
->second
->sequence_number
== sequence_number
)
404 SavedFileEntry
* file_entry
= it
->second
;
405 file_entry
->sequence_number
= sequence_number
;
406 UpdateSavedFileEntry(prefs
, extension_id_
, *file_entry
);
407 saved_file_lru_
.erase(it
++);
408 // Provide the following element as an insert hint. While optimized
409 // insertion time with the following element as a hint is only supported by
410 // the spec in C++11, the implementations do support this.
411 it
= saved_file_lru_
.insert(
412 it
, std::make_pair(file_entry
->sequence_number
, file_entry
));
416 void SavedFilesService::SavedFiles::LoadSavedFileEntriesFromPreferences() {
417 ExtensionPrefs
* prefs
= ExtensionPrefs::Get(profile_
);
418 std::vector
<SavedFileEntry
> saved_entries
=
419 GetSavedFileEntries(prefs
, extension_id_
);
420 for (std::vector
<SavedFileEntry
>::iterator it
= saved_entries
.begin();
421 it
!= saved_entries
.end();
423 scoped_ptr
<SavedFileEntry
> file_entry(new SavedFileEntry(*it
));
424 const std::string
& id
= file_entry
->id
;
425 saved_file_lru_
.insert(
426 std::make_pair(file_entry
->sequence_number
, file_entry
.get()));
427 registered_file_entries_
.add(id
, file_entry
.Pass());
432 void SavedFilesService::SetMaxSequenceNumberForTest(int max_value
) {
433 g_max_sequence_number
= max_value
;
437 void SavedFilesService::ClearMaxSequenceNumberForTest() {
438 g_max_sequence_number
= kMaxSequenceNumber
;
442 void SavedFilesService::SetLruSizeForTest(int size
) {
443 g_max_saved_file_entries
= size
;
447 void SavedFilesService::ClearLruSizeForTest() {
448 g_max_saved_file_entries
= kMaxSavedFileEntries
;