blob: fbb91ea638df555ed6bee52232f04edeb2d0f883 [file] [log] [blame]
[email protected]961745f2013-05-25 14:09:241// 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
avid0181f32015-12-10 19:41:477#include <stdint.h>
aviced8fb1c2016-12-27 23:55:308
[email protected]961745f2013-05-25 14:09:249#include <algorithm>
mgiucaf59c7a82015-06-25 09:11:5710#include <map>
Gyuyoung Kim637cefb2018-02-03 03:03:2111#include <memory>
aviced8fb1c2016-12-27 23:55:3012#include <unordered_map>
dchenge48600452015-12-28 02:24:5013#include <utility>
[email protected]961745f2013-05-25 14:09:2414
15#include "apps/saved_files_service_factory.h"
Nigel Taoabfa80e2020-06-09 10:33:3716#include "base/util/values/values_util.h"
michaelpg4d80e562017-04-04 01:48:1417#include "content/public/browser/browser_context.h"
[email protected]5a0613d32013-06-17 20:06:5318#include "content/public/browser/notification_service.h"
michaelpg868a94be2017-06-26 16:55:2519#include "extensions/browser/api/file_system/saved_file_entry.h"
[email protected]22401dc2014-03-21 01:38:5720#include "extensions/browser/extension_host.h"
[email protected]489db0842014-01-22 18:20:0321#include "extensions/browser/extension_prefs.h"
[email protected]59b0e602014-01-30 00:41:2422#include "extensions/browser/extension_system.h"
[email protected]adf5a102014-07-31 12:44:0623#include "extensions/browser/notification_types.h"
[email protected]793964a2013-10-08 00:47:1924#include "extensions/common/permissions/api_permission.h"
[email protected]5a55f3f2013-10-29 01:08:2925#include "extensions/common/permissions/permission_set.h"
[email protected]076ebeda2014-06-06 21:47:2626#include "extensions/common/permissions/permissions_data.h"
[email protected]961745f2013-05-25 14:09:2427
28namespace apps {
29
30using extensions::APIPermission;
31using extensions::Extension;
32using extensions::ExtensionHost;
33using extensions::ExtensionPrefs;
michaelpg868a94be2017-06-26 16:55:2534using extensions::SavedFileEntry;
[email protected]961745f2013-05-25 14:09:2435
36namespace {
37
38// Preference keys
39
40// The file entries that the app has permission to access.
41const char kFileEntries[] = "file_entries";
42
43// The path to a file entry that the app had permission to access.
44const char kFileEntryPath[] = "path";
45
[email protected]6b7ecdd2013-08-29 14:16:2446// Whether or not the the entry refers to a directory.
47const char kFileEntryIsDirectory[] = "is_directory";
48
[email protected]961745f2013-05-25 14:09:2449// The sequence number in the LRU of the file entry.
50const char kFileEntrySequenceNumber[] = "sequence_number";
51
52const size_t kMaxSavedFileEntries = 500;
avid0181f32015-12-10 19:41:4753const int kMaxSequenceNumber = INT32_MAX;
[email protected]961745f2013-05-25 14:09:2454
55// These might be different to the constant values in tests.
56size_t g_max_saved_file_entries = kMaxSavedFileEntries;
57int g_max_sequence_number = kMaxSequenceNumber;
58
59// Persists a SavedFileEntry in ExtensionPrefs.
60void AddSavedFileEntry(ExtensionPrefs* prefs,
61 const std::string& extension_id,
62 const SavedFileEntry& file_entry) {
63 ExtensionPrefs::ScopedDictionaryUpdate update(
64 prefs, extension_id, kFileEntries);
Sam McNally05e9e692017-05-24 08:07:1265 auto file_entries = update.Create();
[email protected]961745f2013-05-25 14:09:2466 DCHECK(!file_entries->GetDictionaryWithoutPathExpansion(file_entry.id, NULL));
67
corona1046702992016-10-10 03:27:1268 std::unique_ptr<base::DictionaryValue> file_entry_dict =
Gyuyoung Kim637cefb2018-02-03 03:03:2169 std::make_unique<base::DictionaryValue>();
Nigel Taoabfa80e2020-06-09 10:33:3770 file_entry_dict->SetKey(kFileEntryPath,
71 util::FilePathToValue(file_entry.path));
[email protected]6b7ecdd2013-08-29 14:16:2472 file_entry_dict->SetBoolean(kFileEntryIsDirectory, file_entry.is_directory);
[email protected]961745f2013-05-25 14:09:2473 file_entry_dict->SetInteger(kFileEntrySequenceNumber,
74 file_entry.sequence_number);
corona1046702992016-10-10 03:27:1275 file_entries->SetWithoutPathExpansion(file_entry.id,
76 std::move(file_entry_dict));
[email protected]961745f2013-05-25 14:09:2477}
78
79// Updates the sequence_number of a SavedFileEntry persisted in ExtensionPrefs.
80void UpdateSavedFileEntry(ExtensionPrefs* prefs,
81 const std::string& extension_id,
82 const SavedFileEntry& file_entry) {
83 ExtensionPrefs::ScopedDictionaryUpdate update(
84 prefs, extension_id, kFileEntries);
Sam McNally05e9e692017-05-24 08:07:1285 auto file_entries = update.Get();
[email protected]961745f2013-05-25 14:09:2486 DCHECK(file_entries);
Sam McNally05e9e692017-05-24 08:07:1287 std::unique_ptr<prefs::DictionaryValueUpdate> file_entry_dict;
[email protected]961745f2013-05-25 14:09:2488 file_entries->GetDictionaryWithoutPathExpansion(file_entry.id,
89 &file_entry_dict);
90 DCHECK(file_entry_dict);
91 file_entry_dict->SetInteger(kFileEntrySequenceNumber,
92 file_entry.sequence_number);
93}
94
95// Removes a SavedFileEntry from ExtensionPrefs.
96void RemoveSavedFileEntry(ExtensionPrefs* prefs,
97 const std::string& extension_id,
98 const std::string& file_entry_id) {
99 ExtensionPrefs::ScopedDictionaryUpdate update(
100 prefs, extension_id, kFileEntries);
Sam McNally05e9e692017-05-24 08:07:12101 auto file_entries = update.Create();
[email protected]961745f2013-05-25 14:09:24102 file_entries->RemoveWithoutPathExpansion(file_entry_id, NULL);
103}
104
105// Clears all SavedFileEntry for the app from ExtensionPrefs.
106void ClearSavedFileEntries(ExtensionPrefs* prefs,
107 const std::string& extension_id) {
vabrfb687dc2017-03-22 11:40:57108 prefs->UpdateExtensionPref(extension_id, kFileEntries, nullptr);
[email protected]961745f2013-05-25 14:09:24109}
110
111// Returns all SavedFileEntries for the app.
112std::vector<SavedFileEntry> GetSavedFileEntries(
113 ExtensionPrefs* prefs,
114 const std::string& extension_id) {
115 std::vector<SavedFileEntry> result;
[email protected]53b14ea2013-12-21 18:06:37116 const base::DictionaryValue* file_entries = NULL;
[email protected]961745f2013-05-25 14:09:24117 if (!prefs->ReadPrefAsDictionary(extension_id, kFileEntries, &file_entries))
118 return result;
119
[email protected]53b14ea2013-12-21 18:06:37120 for (base::DictionaryValue::Iterator it(*file_entries); !it.IsAtEnd();
[email protected]961745f2013-05-25 14:09:24121 it.Advance()) {
[email protected]53b14ea2013-12-21 18:06:37122 const base::DictionaryValue* file_entry = NULL;
[email protected]961745f2013-05-25 14:09:24123 if (!it.value().GetAsDictionary(&file_entry))
124 continue;
125 const base::Value* path_value;
126 if (!file_entry->Get(kFileEntryPath, &path_value))
127 continue;
Nigel Taoabfa80e2020-06-09 10:33:37128 base::Optional<base::FilePath> file_path =
129 util::ValueToFilePath(*path_value);
130 if (!file_path)
[email protected]961745f2013-05-25 14:09:24131 continue;
[email protected]6b7ecdd2013-08-29 14:16:24132 bool is_directory = false;
133 file_entry->GetBoolean(kFileEntryIsDirectory, &is_directory);
[email protected]961745f2013-05-25 14:09:24134 int sequence_number = 0;
135 if (!file_entry->GetInteger(kFileEntrySequenceNumber, &sequence_number))
136 continue;
137 if (!sequence_number)
138 continue;
[email protected]6b7ecdd2013-08-29 14:16:24139 result.push_back(
Nigel Taoabfa80e2020-06-09 10:33:37140 SavedFileEntry(it.key(), *file_path, is_directory, sequence_number));
[email protected]961745f2013-05-25 14:09:24141 }
142 return result;
143}
144
145} // namespace
146
[email protected]961745f2013-05-25 14:09:24147class SavedFilesService::SavedFiles {
148 public:
michaelpg4d80e562017-04-04 01:48:14149 SavedFiles(content::BrowserContext* context, const std::string& extension_id);
[email protected]961745f2013-05-25 14:09:24150 ~SavedFiles();
151
152 void RegisterFileEntry(const std::string& id,
[email protected]6b7ecdd2013-08-29 14:16:24153 const base::FilePath& file_path,
154 bool is_directory);
[email protected]961745f2013-05-25 14:09:24155 void EnqueueFileEntry(const std::string& id);
156 bool IsRegistered(const std::string& id) const;
157 const SavedFileEntry* GetFileEntry(const std::string& id) const;
158 std::vector<SavedFileEntry> GetAllFileEntries() const;
159
160 private:
161 // Compacts sequence numbers if the largest sequence number is
162 // g_max_sequence_number. Outside of testing, it is set to kint32max, so this
163 // will almost never do any real work.
164 void MaybeCompactSequenceNumbers();
165
166 void LoadSavedFileEntriesFromPreferences();
167
michaelpg4d80e562017-04-04 01:48:14168 content::BrowserContext* context_;
[email protected]961745f2013-05-25 14:09:24169 const std::string extension_id_;
170
aviced8fb1c2016-12-27 23:55:30171 // Contains all file entries that have been registered, keyed by ID.
172 std::unordered_map<std::string, std::unique_ptr<SavedFileEntry>>
mgiucaf59c7a82015-06-25 09:11:57173 registered_file_entries_;
[email protected]961745f2013-05-25 14:09:24174
175 // The queue of file entries that have been retained, keyed by
176 // sequence_number. Values are a subset of values in registered_file_entries_.
177 // This should be kept in sync with file entries stored in extension prefs.
178 std::map<int, SavedFileEntry*> saved_file_lru_;
179
180 DISALLOW_COPY_AND_ASSIGN(SavedFiles);
181};
182
183// static
michaelpg4d80e562017-04-04 01:48:14184SavedFilesService* SavedFilesService::Get(content::BrowserContext* context) {
185 return SavedFilesServiceFactory::GetForBrowserContext(context);
[email protected]961745f2013-05-25 14:09:24186}
187
michaelpg4d80e562017-04-04 01:48:14188SavedFilesService::SavedFilesService(content::BrowserContext* context)
189 : context_(context) {
[email protected]961745f2013-05-25 14:09:24190 registrar_.Add(this,
[email protected]adf5a102014-07-31 12:44:06191 extensions::NOTIFICATION_EXTENSION_HOST_DESTROYED,
[email protected]961745f2013-05-25 14:09:24192 content::NotificationService::AllSources());
[email protected]961745f2013-05-25 14:09:24193}
194
Chris Watkinsee8488b2017-11-27 04:06:56195SavedFilesService::~SavedFilesService() = default;
[email protected]961745f2013-05-25 14:09:24196
197void SavedFilesService::Observe(int type,
198 const content::NotificationSource& source,
199 const content::NotificationDetails& details) {
michaelpgdbdcdcc2017-04-06 01:40:56200 DCHECK_EQ(extensions::NOTIFICATION_EXTENSION_HOST_DESTROYED, type);
201 ExtensionHost* host = content::Details<ExtensionHost>(details).ptr();
202 const Extension* extension = host->extension();
203 if (extension) {
204 ClearQueueIfNoRetainPermission(extension);
205 Clear(extension->id());
[email protected]961745f2013-05-25 14:09:24206 }
207}
208
209void SavedFilesService::RegisterFileEntry(const std::string& extension_id,
210 const std::string& id,
[email protected]6b7ecdd2013-08-29 14:16:24211 const base::FilePath& file_path,
212 bool is_directory) {
213 GetOrInsert(extension_id)->RegisterFileEntry(id, file_path, is_directory);
[email protected]961745f2013-05-25 14:09:24214}
215
216void SavedFilesService::EnqueueFileEntry(const std::string& extension_id,
217 const std::string& id) {
218 GetOrInsert(extension_id)->EnqueueFileEntry(id);
219}
220
221std::vector<SavedFileEntry> SavedFilesService::GetAllFileEntries(
222 const std::string& extension_id) {
[email protected]a2886e8b2013-06-08 05:15:02223 SavedFiles* saved_files = Get(extension_id);
224 if (saved_files)
225 return saved_files->GetAllFileEntries();
michaelpg4d80e562017-04-04 01:48:14226 return GetSavedFileEntries(ExtensionPrefs::Get(context_), extension_id);
[email protected]961745f2013-05-25 14:09:24227}
228
229bool SavedFilesService::IsRegistered(const std::string& extension_id,
230 const std::string& id) {
231 return GetOrInsert(extension_id)->IsRegistered(id);
232}
233
234const SavedFileEntry* SavedFilesService::GetFileEntry(
235 const std::string& extension_id,
236 const std::string& id) {
237 return GetOrInsert(extension_id)->GetFileEntry(id);
238}
239
240void SavedFilesService::ClearQueueIfNoRetainPermission(
241 const Extension* extension) {
benwells1dd4acd2015-12-09 02:20:24242 if (!extension->permissions_data()->active_permissions().HasAPIPermission(
[email protected]73f73a02013-07-04 02:15:12243 APIPermission::kFileSystemRetainEntries)) {
[email protected]a2886e8b2013-06-08 05:15:02244 ClearQueue(extension);
[email protected]961745f2013-05-25 14:09:24245 }
246}
247
[email protected]a2886e8b2013-06-08 05:15:02248void SavedFilesService::ClearQueue(const extensions::Extension* extension) {
michaelpg4d80e562017-04-04 01:48:14249 ClearSavedFileEntries(ExtensionPrefs::Get(context_), extension->id());
[email protected]a2886e8b2013-06-08 05:15:02250 Clear(extension->id());
251}
252
michaelpgdbdcdcc2017-04-06 01:40:56253void SavedFilesService::OnApplicationTerminating() {
254 // Stop listening to NOTIFICATION_EXTENSION_HOST_DESTROYED in particular
255 // as all extension hosts will be destroyed as a result of shutdown.
256 registrar_.RemoveAll();
257}
258
[email protected]a2886e8b2013-06-08 05:15:02259SavedFilesService::SavedFiles* SavedFilesService::Get(
260 const std::string& extension_id) const {
limasdf918cc2e92015-11-20 04:28:57261 auto it = extension_id_to_saved_files_.find(extension_id);
[email protected]961745f2013-05-25 14:09:24262 if (it != extension_id_to_saved_files_.end())
limasdf918cc2e92015-11-20 04:28:57263 return it->second.get();
[email protected]961745f2013-05-25 14:09:24264
[email protected]a2886e8b2013-06-08 05:15:02265 return NULL;
266}
267
268SavedFilesService::SavedFiles* SavedFilesService::GetOrInsert(
269 const std::string& extension_id) {
270 SavedFiles* saved_files = Get(extension_id);
271 if (saved_files)
272 return saved_files;
273
mostynbecb4a22b2016-04-04 06:08:01274 std::unique_ptr<SavedFiles> scoped_saved_files(
michaelpg4d80e562017-04-04 01:48:14275 new SavedFiles(context_, extension_id));
mgiucaf59c7a82015-06-25 09:11:57276 saved_files = scoped_saved_files.get();
limasdf918cc2e92015-11-20 04:28:57277 extension_id_to_saved_files_.insert(
278 std::make_pair(extension_id, std::move(scoped_saved_files)));
[email protected]961745f2013-05-25 14:09:24279 return saved_files;
280}
281
282void SavedFilesService::Clear(const std::string& extension_id) {
mgiucaf59c7a82015-06-25 09:11:57283 extension_id_to_saved_files_.erase(extension_id);
[email protected]961745f2013-05-25 14:09:24284}
285
michaelpg4d80e562017-04-04 01:48:14286SavedFilesService::SavedFiles::SavedFiles(content::BrowserContext* context,
[email protected]961745f2013-05-25 14:09:24287 const std::string& extension_id)
michaelpg4d80e562017-04-04 01:48:14288 : context_(context), extension_id_(extension_id) {
[email protected]961745f2013-05-25 14:09:24289 LoadSavedFileEntriesFromPreferences();
290}
291
Chris Watkinsee8488b2017-11-27 04:06:56292SavedFilesService::SavedFiles::~SavedFiles() = default;
[email protected]961745f2013-05-25 14:09:24293
294void SavedFilesService::SavedFiles::RegisterFileEntry(
295 const std::string& id,
[email protected]6b7ecdd2013-08-29 14:16:24296 const base::FilePath& file_path,
297 bool is_directory) {
aviced8fb1c2016-12-27 23:55:30298 auto it = registered_file_entries_.find(id);
299 if (it != registered_file_entries_.end())
[email protected]961745f2013-05-25 14:09:24300 return;
301
aviced8fb1c2016-12-27 23:55:30302 registered_file_entries_[id] =
Gyuyoung Kim637cefb2018-02-03 03:03:21303 std::make_unique<SavedFileEntry>(id, file_path, is_directory, 0);
[email protected]961745f2013-05-25 14:09:24304}
305
306void SavedFilesService::SavedFiles::EnqueueFileEntry(const std::string& id) {
mgiucaf59c7a82015-06-25 09:11:57307 auto it = registered_file_entries_.find(id);
[email protected]961745f2013-05-25 14:09:24308 DCHECK(it != registered_file_entries_.end());
309
aviced8fb1c2016-12-27 23:55:30310 SavedFileEntry* file_entry = it->second.get();
[email protected]961745f2013-05-25 14:09:24311 int old_sequence_number = file_entry->sequence_number;
Luciano Pacheco186422e2019-08-12 03:46:22312
313#if defined(OS_CHROMEOS)
314 // crbug.com/983844 Convert path from legacy Download/ to MyFiles/Downloads/
315 // so entries saved before MyFiles don't fail. TODO(lucmult): Remove this
316 // after M-83.
317 const auto legacy_downloads = context_->GetPath().AppendASCII("Downloads");
318 auto to_myfiles =
319 context_->GetPath().AppendASCII("MyFiles").AppendASCII("Downloads");
320 if (legacy_downloads.AppendRelativePath(file_entry->path, &to_myfiles))
321 file_entry->path = to_myfiles;
322#endif
323
[email protected]961745f2013-05-25 14:09:24324 if (!saved_file_lru_.empty()) {
325 // Get the sequence number after the last file entry in the LRU.
326 std::map<int, SavedFileEntry*>::reverse_iterator it =
327 saved_file_lru_.rbegin();
328 if (it->second == file_entry)
329 return;
330
331 file_entry->sequence_number = it->first + 1;
332 } else {
333 // The first sequence number is 1, as 0 means the entry is not in the LRU.
334 file_entry->sequence_number = 1;
335 }
336 saved_file_lru_.insert(
337 std::make_pair(file_entry->sequence_number, file_entry));
michaelpg4d80e562017-04-04 01:48:14338 ExtensionPrefs* prefs = ExtensionPrefs::Get(context_);
[email protected]961745f2013-05-25 14:09:24339 if (old_sequence_number) {
340 saved_file_lru_.erase(old_sequence_number);
341 UpdateSavedFileEntry(prefs, extension_id_, *file_entry);
342 } else {
343 AddSavedFileEntry(prefs, extension_id_, *file_entry);
344 if (saved_file_lru_.size() > g_max_saved_file_entries) {
345 std::map<int, SavedFileEntry*>::iterator it = saved_file_lru_.begin();
346 it->second->sequence_number = 0;
347 RemoveSavedFileEntry(prefs, extension_id_, it->second->id);
348 saved_file_lru_.erase(it);
349 }
350 }
351 MaybeCompactSequenceNumbers();
352}
353
354bool SavedFilesService::SavedFiles::IsRegistered(const std::string& id) const {
aviced8fb1c2016-12-27 23:55:30355 auto it = registered_file_entries_.find(id);
356 return it != registered_file_entries_.end();
[email protected]961745f2013-05-25 14:09:24357}
358
359const SavedFileEntry* SavedFilesService::SavedFiles::GetFileEntry(
360 const std::string& id) const {
mgiucaf59c7a82015-06-25 09:11:57361 auto it = registered_file_entries_.find(id);
[email protected]961745f2013-05-25 14:09:24362 if (it == registered_file_entries_.end())
363 return NULL;
364
aviced8fb1c2016-12-27 23:55:30365 return it->second.get();
[email protected]961745f2013-05-25 14:09:24366}
367
368std::vector<SavedFileEntry> SavedFilesService::SavedFiles::GetAllFileEntries()
369 const {
370 std::vector<SavedFileEntry> result;
mgiucaf59c7a82015-06-25 09:11:57371 for (auto it = registered_file_entries_.begin();
372 it != registered_file_entries_.end(); ++it) {
aviced8fb1c2016-12-27 23:55:30373 result.push_back(*it->second.get());
[email protected]961745f2013-05-25 14:09:24374 }
375 return result;
376}
377
378void SavedFilesService::SavedFiles::MaybeCompactSequenceNumbers() {
379 DCHECK_GE(g_max_sequence_number, 0);
380 DCHECK_GE(static_cast<size_t>(g_max_sequence_number),
381 g_max_saved_file_entries);
382 std::map<int, SavedFileEntry*>::reverse_iterator it =
383 saved_file_lru_.rbegin();
384 if (it == saved_file_lru_.rend())
385 return;
386
387 // Only compact sequence numbers if the last entry's sequence number is the
388 // maximum value. This should almost never be the case.
389 if (it->first < g_max_sequence_number)
390 return;
391
392 int sequence_number = 0;
michaelpg4d80e562017-04-04 01:48:14393 ExtensionPrefs* prefs = ExtensionPrefs::Get(context_);
[email protected]961745f2013-05-25 14:09:24394 for (std::map<int, SavedFileEntry*>::iterator it = saved_file_lru_.begin();
395 it != saved_file_lru_.end();
396 ++it) {
397 sequence_number++;
398 if (it->second->sequence_number == sequence_number)
399 continue;
400
401 SavedFileEntry* file_entry = it->second;
402 file_entry->sequence_number = sequence_number;
403 UpdateSavedFileEntry(prefs, extension_id_, *file_entry);
404 saved_file_lru_.erase(it++);
405 // Provide the following element as an insert hint. While optimized
406 // insertion time with the following element as a hint is only supported by
407 // the spec in C++11, the implementations do support this.
408 it = saved_file_lru_.insert(
409 it, std::make_pair(file_entry->sequence_number, file_entry));
410 }
411}
412
413void SavedFilesService::SavedFiles::LoadSavedFileEntriesFromPreferences() {
michaelpg4d80e562017-04-04 01:48:14414 ExtensionPrefs* prefs = ExtensionPrefs::Get(context_);
[email protected]961745f2013-05-25 14:09:24415 std::vector<SavedFileEntry> saved_entries =
416 GetSavedFileEntries(prefs, extension_id_);
417 for (std::vector<SavedFileEntry>::iterator it = saved_entries.begin();
418 it != saved_entries.end();
419 ++it) {
mostynbecb4a22b2016-04-04 06:08:01420 std::unique_ptr<SavedFileEntry> file_entry(new SavedFileEntry(*it));
mgiucaf59c7a82015-06-25 09:11:57421 const std::string& id = file_entry->id;
[email protected]961745f2013-05-25 14:09:24422 saved_file_lru_.insert(
mgiucaf59c7a82015-06-25 09:11:57423 std::make_pair(file_entry->sequence_number, file_entry.get()));
aviced8fb1c2016-12-27 23:55:30424 registered_file_entries_[id] = std::move(file_entry);
[email protected]961745f2013-05-25 14:09:24425 }
426}
427
428// static
429void SavedFilesService::SetMaxSequenceNumberForTest(int max_value) {
430 g_max_sequence_number = max_value;
431}
432
433// static
434void SavedFilesService::ClearMaxSequenceNumberForTest() {
435 g_max_sequence_number = kMaxSequenceNumber;
436}
437
438// static
439void SavedFilesService::SetLruSizeForTest(int size) {
440 g_max_saved_file_entries = size;
441}
442
443// static
444void SavedFilesService::ClearLruSizeForTest() {
445 g_max_saved_file_entries = kMaxSavedFileEntries;
446}
447
448} // namespace apps