Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/chromeos/arc/arc_downloads_watcher_service.h" | 5 #include "chrome/browser/chromeos/arc/arc_downloads_watcher_service.h" |
| 6 | 6 |
| 7 #include <map> | 7 #include <map> |
| 8 #include <memory> | 8 #include <memory> |
| 9 #include <utility> | 9 #include <utility> |
| 10 | 10 |
| (...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 79 using Callback = base::Callback<void(mojo::Array<mojo::String> paths)>; | 79 using Callback = base::Callback<void(mojo::Array<mojo::String> paths)>; |
| 80 | 80 |
| 81 explicit DownloadsWatcher(const Callback& callback); | 81 explicit DownloadsWatcher(const Callback& callback); |
| 82 ~DownloadsWatcher(); | 82 ~DownloadsWatcher(); |
| 83 | 83 |
| 84 // Starts watching Downloads directory. | 84 // Starts watching Downloads directory. |
| 85 void Start(); | 85 void Start(); |
| 86 | 86 |
| 87 private: | 87 private: |
| 88 // Called by base::FilePathWatcher to notify file changes. | 88 // Called by base::FilePathWatcher to notify file changes. |
| 89 // Wrapper around HandleFilePathChanged to pass the notification timestamp. | |
| 89 void OnFilePathChanged(const base::FilePath& path, bool error); | 90 void OnFilePathChanged(const base::FilePath& path, bool error); |
| 90 | 91 |
| 92 // If there has not been a full scan of updated files since notify_time, | |
| 93 // update the last_timestamp_map_ and send any new/changed files to the | |
| 94 // android media scanner. | |
| 95 void HandleFilePathChanged(base::Time notify_time); | |
| 96 | |
| 91 // Scans files under |downloads_dir_| recursively and builds a map from file | 97 // Scans files under |downloads_dir_| recursively and builds a map from file |
| 92 // paths (in Android filesystem) to last modified timestamps. | 98 // paths (in Android filesystem) to last modified timestamps. |
| 93 TimestampMap BuildTimestampMap() const; | 99 TimestampMap BuildTimestampMap() const; |
| 94 | 100 |
| 95 Callback callback_; | 101 Callback callback_; |
| 96 base::FilePath downloads_dir_; | 102 base::FilePath downloads_dir_; |
| 97 std::unique_ptr<base::FilePathWatcher> watcher_; | 103 std::unique_ptr<base::FilePathWatcher> watcher_; |
| 98 TimestampMap last_timestamp_map_; | 104 TimestampMap last_timestamp_map_; |
| 105 base::Time last_update_time_; | |
| 99 | 106 |
| 100 // Note: This should remain the last member so it'll be destroyed and | 107 // Note: This should remain the last member so it'll be destroyed and |
| 101 // invalidate the weak pointers before any other members are destroyed. | 108 // invalidate the weak pointers before any other members are destroyed. |
| 102 base::WeakPtrFactory<DownloadsWatcher> weak_ptr_factory_; | 109 base::WeakPtrFactory<DownloadsWatcher> weak_ptr_factory_; |
| 103 | 110 |
| 104 DISALLOW_COPY_AND_ASSIGN(DownloadsWatcher); | 111 DISALLOW_COPY_AND_ASSIGN(DownloadsWatcher); |
| 105 }; | 112 }; |
| 106 | 113 |
| 107 ArcDownloadsWatcherService::DownloadsWatcher::DownloadsWatcher( | 114 ArcDownloadsWatcherService::DownloadsWatcher::DownloadsWatcher( |
| 108 const Callback& callback) | 115 const Callback& callback) |
| 109 : callback_(callback), weak_ptr_factory_(this) { | 116 : callback_(callback), |
| 117 last_update_time_(base::Time()), | |
| 118 weak_ptr_factory_(this) { | |
| 110 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 119 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 111 | 120 |
| 112 downloads_dir_ = DownloadPrefs(ProfileManager::GetActiveUserProfile()) | 121 downloads_dir_ = DownloadPrefs(ProfileManager::GetActiveUserProfile()) |
| 113 .GetDefaultDownloadDirectoryForProfile() | 122 .GetDefaultDownloadDirectoryForProfile() |
| 114 .StripTrailingSeparators(); | 123 .StripTrailingSeparators(); |
| 115 } | 124 } |
| 116 | 125 |
| 117 ArcDownloadsWatcherService::DownloadsWatcher::~DownloadsWatcher() { | 126 ArcDownloadsWatcherService::DownloadsWatcher::~DownloadsWatcher() { |
| 118 DCHECK_CURRENTLY_ON(BrowserThread::FILE); | 127 DCHECK_CURRENTLY_ON(BrowserThread::FILE); |
| 119 } | 128 } |
| (...skipping 11 matching lines...) Expand all Loading... | |
| 131 base::Bind(&DownloadsWatcher::OnFilePathChanged, | 140 base::Bind(&DownloadsWatcher::OnFilePathChanged, |
| 132 weak_ptr_factory_.GetWeakPtr())); | 141 weak_ptr_factory_.GetWeakPtr())); |
| 133 } | 142 } |
| 134 | 143 |
| 135 void ArcDownloadsWatcherService::DownloadsWatcher::OnFilePathChanged( | 144 void ArcDownloadsWatcherService::DownloadsWatcher::OnFilePathChanged( |
| 136 const base::FilePath& path, | 145 const base::FilePath& path, |
| 137 bool error) { | 146 bool error) { |
| 138 // On Linux, |error| is always false. Also, |path| is always the same path | 147 // On Linux, |error| is always false. Also, |path| is always the same path |
| 139 // as one given to FilePathWatcher::Watch(). | 148 // as one given to FilePathWatcher::Watch(). |
| 140 DCHECK_CURRENTLY_ON(BrowserThread::FILE); | 149 DCHECK_CURRENTLY_ON(BrowserThread::FILE); |
| 150 BrowserThread::PostTask( | |
| 151 FROM_HERE, base::Bind(&DownloadsWatcher::HandleFilePathChanged, | |
| 152 weak_ptr_factory_.GetWeakPtr(), base::Time::Now())); | |
| 153 } | |
| 154 | |
| 155 void ArcDownloadsWatcherService::DownloadsWatcher::HandleFilePathChanged( | |
| 156 base::Time notify_time) { | |
| 157 // Because this logic handles all updates and not just the update that | |
| 158 // triggered the callback we can return early if we have already processed | |
| 159 // updates to the directory after this callback was created. | |
| 160 if (notify_time < last_update_time_) { | |
| 161 return; | |
| 162 } | |
| 163 last_update_time_ = base::Time::Now(); | |
| 141 | 164 |
| 142 TimestampMap current_timestamp_map = BuildTimestampMap(); | 165 TimestampMap current_timestamp_map = BuildTimestampMap(); |
|
hidehiko
2016/09/07 01:44:32
For the record.
As we chatted offline, because th
| |
| 143 | 166 |
| 144 std::vector<base::FilePath> changed_paths = | 167 std::vector<base::FilePath> changed_paths = |
| 145 CollectChangedPaths(last_timestamp_map_, current_timestamp_map); | 168 CollectChangedPaths(last_timestamp_map_, current_timestamp_map); |
| 146 | 169 |
| 147 last_timestamp_map_ = std::move(current_timestamp_map); | 170 last_timestamp_map_ = std::move(current_timestamp_map); |
| 148 | 171 |
| 149 mojo::Array<mojo::String> mojo_paths(changed_paths.size()); | 172 mojo::Array<mojo::String> mojo_paths(changed_paths.size()); |
| 150 for (size_t i = 0; i < changed_paths.size(); ++i) { | 173 for (size_t i = 0; i < changed_paths.size(); ++i) { |
| 151 mojo_paths[i] = changed_paths[i].value(); | 174 mojo_paths[i] = changed_paths[i].value(); |
| 152 } | 175 } |
| (...skipping 72 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 225 mojo::Array<mojo::String> paths) { | 248 mojo::Array<mojo::String> paths) { |
| 226 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 249 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 227 | 250 |
| 228 auto* instance = arc_bridge_service()->file_system()->instance(); | 251 auto* instance = arc_bridge_service()->file_system()->instance(); |
| 229 if (!instance) | 252 if (!instance) |
| 230 return; | 253 return; |
| 231 instance->RequestMediaScan(std::move(paths)); | 254 instance->RequestMediaScan(std::move(paths)); |
| 232 } | 255 } |
| 233 | 256 |
| 234 } // namespace arc | 257 } // namespace arc |
| OLD | NEW |