1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
|
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/extensions/api/file_handlers/directory_util.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/thread_task_runner_handle.h"
#include "chrome/browser/profiles/profile.h"
#include "content/public/browser/browser_thread.h"
#include "net/base/filename_util.h"
#include "storage/browser/fileapi/file_system_url.h"
#if defined(OS_CHROMEOS)
#include "chrome/browser/chromeos/file_manager/filesystem_api_util.h"
#endif
namespace extensions {
namespace app_file_handler_util {
namespace {
void GetIsDirectoryFromFileInfo(const base::FilePath& path,
bool* is_directory) {
base::File::Info file_info;
*is_directory = GetFileInfo(path, &file_info) && file_info.is_directory;
}
void OnGetIsDirectoryFromFileInfoCompleted(
scoped_ptr<bool> is_directory,
const base::Callback<void(bool)>& callback) {
callback.Run(*is_directory);
}
// The callback parameter contains the result and is required to support
// both native local directories to avoid UI thread and non native local
// path directories for the IsNonNativeLocalPathDirectory API.
void EntryIsDirectory(Profile* profile,
const base::FilePath& path,
const base::Callback<void(bool)>& callback) {
#if defined(OS_CHROMEOS)
if (file_manager::util::IsUnderNonNativeLocalPath(profile, path)) {
file_manager::util::IsNonNativeLocalPathDirectory(profile, path, callback);
return;
}
#endif
scoped_ptr<bool> is_directory(new bool);
bool* const is_directory_ptr = is_directory.get();
content::BrowserThread::PostBlockingPoolTaskAndReply(
FROM_HERE,
base::Bind(&GetIsDirectoryFromFileInfo, path, is_directory_ptr),
base::Bind(&OnGetIsDirectoryFromFileInfoCompleted,
base::Passed(&is_directory), callback));
}
} // namespace
IsDirectoryCollector::IsDirectoryCollector(Profile* profile)
: profile_(profile), left_(0), weak_ptr_factory_(this) {}
IsDirectoryCollector::~IsDirectoryCollector() {}
void IsDirectoryCollector::CollectForEntriesPaths(
const std::vector<base::FilePath>& paths,
const CompletionCallback& callback) {
DCHECK(!callback.is_null());
paths_ = paths;
callback_ = callback;
DCHECK(!result_.get());
result_.reset(new std::set<base::FilePath>());
left_ = paths.size();
if (!left_) {
// Nothing to process.
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::Bind(callback_, base::Passed(&result_)));
callback_ = CompletionCallback();
return;
}
for (size_t i = 0; i < paths.size(); ++i) {
EntryIsDirectory(profile_, paths[i],
base::Bind(&IsDirectoryCollector::OnIsDirectoryCollected,
weak_ptr_factory_.GetWeakPtr(), i));
}
}
void IsDirectoryCollector::OnIsDirectoryCollected(size_t index,
bool is_directory) {
if (is_directory)
result_->insert(paths_[index]);
if (!--left_) {
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::Bind(callback_, base::Passed(&result_)));
// Release the callback to avoid a circullar reference in case an instance
// of this class is a member of a ref counted class, which instance is bound
// to this callback.
callback_ = CompletionCallback();
}
}
} // namespace app_file_handler_util
} // namespace extensions
|