OLD | NEW |
1 // Copyright (c) 2014 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2014 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/component_updater/sw_reporter_installer_win.h" | 5 #include "chrome/browser/component_updater/sw_reporter_installer_win.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 #include <vector> | 8 #include <vector> |
9 | 9 |
10 #include "base/base_paths.h" | 10 #include "base/base_paths.h" |
11 #include "base/bind.h" | 11 #include "base/bind.h" |
12 #include "base/bind_helpers.h" | 12 #include "base/bind_helpers.h" |
13 #include "base/command_line.h" | 13 #include "base/command_line.h" |
14 #include "base/file_util.h" | 14 #include "base/file_util.h" |
15 #include "base/files/file_path.h" | 15 #include "base/files/file_path.h" |
16 #include "base/logging.h" | 16 #include "base/logging.h" |
17 #include "base/metrics/histogram.h" | 17 #include "base/metrics/histogram.h" |
18 #include "base/metrics/sparse_histogram.h" | 18 #include "base/metrics/sparse_histogram.h" |
19 #include "base/path_service.h" | 19 #include "base/path_service.h" |
20 #include "base/prefs/pref_registry_simple.h" | 20 #include "base/prefs/pref_registry_simple.h" |
21 #include "base/prefs/pref_service.h" | 21 #include "base/prefs/pref_service.h" |
22 #include "base/process/kill.h" | 22 #include "base/process/kill.h" |
23 #include "base/process/launch.h" | 23 #include "base/process/launch.h" |
24 #include "base/task_runner_util.h" | 24 #include "base/task_runner_util.h" |
25 #include "base/threading/worker_pool.h" | 25 #include "base/threading/worker_pool.h" |
| 26 #include "base/time/time.h" |
26 #include "base/win/registry.h" | 27 #include "base/win/registry.h" |
27 #include "chrome/browser/browser_process.h" | 28 #include "chrome/browser/browser_process.h" |
| 29 #include "chrome/browser/metrics/chrome_metrics_service_accessor.h" |
| 30 #include "chrome/common/pref_names.h" |
28 #include "components/component_updater/component_updater_paths.h" | 31 #include "components/component_updater/component_updater_paths.h" |
29 #include "components/component_updater/component_updater_service.h" | 32 #include "components/component_updater/component_updater_service.h" |
30 #include "components/component_updater/component_updater_utils.h" | 33 #include "components/component_updater/component_updater_utils.h" |
31 #include "components/component_updater/default_component_installer.h" | 34 #include "components/component_updater/default_component_installer.h" |
32 #include "components/component_updater/pref_names.h" | 35 #include "components/component_updater/pref_names.h" |
33 #include "content/public/browser/browser_thread.h" | 36 #include "content/public/browser/browser_thread.h" |
34 | 37 |
35 using content::BrowserThread; | 38 using content::BrowserThread; |
36 | 39 |
37 namespace component_updater { | 40 namespace component_updater { |
38 | 41 |
39 namespace { | 42 namespace { |
40 | 43 |
41 // These values are used to send UMA information and are replicated in the | 44 // These values are used to send UMA information and are replicated in the |
42 // histograms.xml file, so the order MUST NOT CHANGE. | 45 // histograms.xml file, so the order MUST NOT CHANGE. |
43 enum SwReporterUmaValue { | 46 enum SwReporterUmaValue { |
44 SW_REPORTER_EXPLICIT_REQUEST = 0, | 47 SW_REPORTER_EXPLICIT_REQUEST = 0, |
45 SW_REPORTER_STARTUP_RETRY = 1, | 48 SW_REPORTER_STARTUP_RETRY = 1, |
46 SW_REPORTER_RETRIED_TOO_MANY_TIMES = 2, | 49 SW_REPORTER_RETRIED_TOO_MANY_TIMES = 2, |
47 SW_REPORTER_START_EXECUTION = 3, | 50 SW_REPORTER_START_EXECUTION = 3, |
48 SW_REPORTER_FAILED_TO_START = 4, | 51 SW_REPORTER_FAILED_TO_START = 4, |
49 SW_REPORTER_REGISTRY_EXIT_CODE = 5, | 52 SW_REPORTER_REGISTRY_EXIT_CODE = 5, |
50 SW_REPORTER_RESET_RETRIES = 6, | 53 SW_REPORTER_RESET_RETRIES = 6, |
51 SW_REPORTER_MAX, | 54 SW_REPORTER_MAX, |
52 }; | 55 }; |
53 | 56 |
54 // The maximum number of times to retry a download on startup. | 57 // The maximum number of times to retry a download on startup. |
55 const int kMaxRetry = 20; | 58 const int kMaxRetry = 20; |
56 | 59 |
| 60 // The number of days to wait before triggering another sw reporter run. |
| 61 const int kDaysBetweenSwReporterRuns = 7; |
| 62 |
57 // CRX hash. The extension id is: gkmgaooipdjhmangpemjhigmamcehddo. The hash was | 63 // CRX hash. The extension id is: gkmgaooipdjhmangpemjhigmamcehddo. The hash was |
58 // generated in Python with something like this: | 64 // generated in Python with something like this: |
59 // hashlib.sha256().update(open("<file>.crx").read()[16:16+294]).digest(). | 65 // hashlib.sha256().update(open("<file>.crx").read()[16:16+294]).digest(). |
60 const uint8 kSha256Hash[] = {0x6a, 0xc6, 0x0e, 0xe8, 0xf3, 0x97, 0xc0, 0xd6, | 66 const uint8 kSha256Hash[] = {0x6a, 0xc6, 0x0e, 0xe8, 0xf3, 0x97, 0xc0, 0xd6, |
61 0xf4, 0xc9, 0x78, 0x6c, 0x0c, 0x24, 0x73, 0x3e, | 67 0xf4, 0xc9, 0x78, 0x6c, 0x0c, 0x24, 0x73, 0x3e, |
62 0x05, 0xa5, 0x62, 0x4b, 0x2e, 0xc7, 0xb7, 0x1c, | 68 0x05, 0xa5, 0x62, 0x4b, 0x2e, 0xc7, 0xb7, 0x1c, |
63 0x5f, 0xea, 0xf0, 0x88, 0xf6, 0x97, 0x9b, 0xc7}; | 69 0x5f, 0xea, 0xf0, 0x88, 0xf6, 0x97, 0x9b, 0xc7}; |
64 | 70 |
65 const base::FilePath::CharType kSwReporterExeName[] = | 71 const base::FilePath::CharType kSwReporterExeName[] = |
66 FILE_PATH_LITERAL("software_reporter_tool.exe"); | 72 FILE_PATH_LITERAL("software_reporter_tool.exe"); |
(...skipping 146 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
213 void ExecuteSwReporter(ComponentUpdateService* cus, PrefService* prefs) { | 219 void ExecuteSwReporter(ComponentUpdateService* cus, PrefService* prefs) { |
214 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 220 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
215 // If we have a pending execution, send metrics about it so we can account for | 221 // If we have a pending execution, send metrics about it so we can account for |
216 // missing executions. | 222 // missing executions. |
217 if (prefs->GetInteger(prefs::kSwReporterExecuteTryCount) > 0) | 223 if (prefs->GetInteger(prefs::kSwReporterExecuteTryCount) > 0) |
218 ReportUmaStep(SW_REPORTER_RESET_RETRIES); | 224 ReportUmaStep(SW_REPORTER_RESET_RETRIES); |
219 // This is an explicit call, so let's forget about previous incomplete | 225 // This is an explicit call, so let's forget about previous incomplete |
220 // execution attempts and start from scratch. | 226 // execution attempts and start from scratch. |
221 prefs->SetInteger(prefs::kSwReporterExecuteTryCount, kMaxRetry); | 227 prefs->SetInteger(prefs::kSwReporterExecuteTryCount, kMaxRetry); |
222 ReportUmaStep(SW_REPORTER_EXPLICIT_REQUEST); | 228 ReportUmaStep(SW_REPORTER_EXPLICIT_REQUEST); |
| 229 // Record the requested trigger time. |
| 230 prefs->SetInt64(prefs::kSwReporterLastTimeTriggered, |
| 231 base::Time::Now().ToInternalValue()); |
223 const std::vector<std::string> registered_components(cus->GetComponentIDs()); | 232 const std::vector<std::string> registered_components(cus->GetComponentIDs()); |
224 if (std::find(registered_components.begin(), | 233 if (std::find(registered_components.begin(), |
225 registered_components.end(), | 234 registered_components.end(), |
226 SwReporterInstallerTraits::ID()) == | 235 SwReporterInstallerTraits::ID()) == |
227 registered_components.end()) { | 236 registered_components.end()) { |
228 RegisterComponent(cus, prefs); | 237 RegisterComponent(cus, prefs); |
229 } else if (!SwReporterInstallerTraits::VersionPath().empty()) { | 238 } else if (!SwReporterInstallerTraits::VersionPath().empty()) { |
230 // Here, we already have a fully registered and installed component | 239 // Here, we already have a fully registered and installed component |
231 // available for immediate use. This doesn't handle cases where the version | 240 // available for immediate use. This doesn't handle cases where the version |
232 // folder is there but the executable is not within in. This is a corruption | 241 // folder is there but the executable is not within in. This is a corruption |
233 // we don't want to handle here. | 242 // we don't want to handle here. |
234 ExecuteReporter(SwReporterInstallerTraits::VersionPath()); | 243 ExecuteReporter(SwReporterInstallerTraits::VersionPath()); |
235 } | 244 } |
236 // If the component is registered but the version path is not available, it | 245 // If the component is registered but the version path is not available, it |
237 // means the component was not fully installed yet, and it should run the | 246 // means the component was not fully installed yet, and it should run the |
238 // reporter when ComponentReady is called. | 247 // reporter when ComponentReady is called. |
239 } | 248 } |
240 | 249 |
241 void ExecutePendingSwReporter(ComponentUpdateService* cus, PrefService* prefs) { | 250 void ExecuteSwReporterIfRequired(ComponentUpdateService* cus, |
| 251 PrefService* prefs) { |
242 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | 252 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
243 | 253 |
| 254 // The Sw reporter shouldn't run if the user isn't reporting metrics. |
| 255 if (!ChromeMetricsServiceAccessor::IsMetricsReportingEnabled()) |
| 256 return; |
| 257 |
244 // Register the existing component for updates. | 258 // Register the existing component for updates. |
245 base::PostTaskAndReplyWithResult( | 259 base::PostTaskAndReplyWithResult( |
246 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE), | 260 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE), |
247 FROM_HERE, | 261 FROM_HERE, |
248 base::Bind(&base::PathExists, SwReporterInstallerTraits::install_dir()), | 262 base::Bind(&base::PathExists, SwReporterInstallerTraits::install_dir()), |
249 base::Bind(&MaybeRegisterComponent, cus, prefs)); | 263 base::Bind(&MaybeRegisterComponent, cus, prefs)); |
250 | 264 |
251 // Run the reporter if there is a pending execution request. | 265 // Run the reporter if there is a pending execution request. |
252 int execute_try_count = prefs->GetInteger(prefs::kSwReporterExecuteTryCount); | 266 int execute_try_count = prefs->GetInteger(prefs::kSwReporterExecuteTryCount); |
253 if (execute_try_count > 0) { | 267 if (execute_try_count > 0) { |
(...skipping 11 matching lines...) Expand all Loading... |
265 | 279 |
266 // The previous request has not completed. The reporter will run again | 280 // The previous request has not completed. The reporter will run again |
267 // when ComponentReady is called or the request is abandoned if it has | 281 // when ComponentReady is called or the request is abandoned if it has |
268 // been tried too many times. | 282 // been tried too many times. |
269 prefs->SetInteger(prefs::kSwReporterExecuteTryCount, --execute_try_count); | 283 prefs->SetInteger(prefs::kSwReporterExecuteTryCount, --execute_try_count); |
270 if (execute_try_count > 0) | 284 if (execute_try_count > 0) |
271 ReportUmaStep(SW_REPORTER_STARTUP_RETRY); | 285 ReportUmaStep(SW_REPORTER_STARTUP_RETRY); |
272 else | 286 else |
273 ReportUmaStep(SW_REPORTER_RETRIED_TOO_MANY_TIMES); | 287 ReportUmaStep(SW_REPORTER_RETRIED_TOO_MANY_TIMES); |
274 } | 288 } |
| 289 |
| 290 // Run the reporter if it hasn't been triggered in the last |
| 291 // kDaysBetweenSwReporterRuns days. |
| 292 const base::Time last_time_triggered = base::Time::FromInternalValue( |
| 293 prefs->GetInt64(prefs::kSwReporterLastTimeTriggered)); |
| 294 if ((base::Time::Now() - last_time_triggered).InDays() >= |
| 295 kDaysBetweenSwReporterRuns) { |
| 296 ExecuteSwReporter(cus, prefs); |
| 297 } |
275 } | 298 } |
276 | 299 |
277 void RegisterPrefsForSwReporter(PrefRegistrySimple* registry) { | 300 void RegisterPrefsForSwReporter(PrefRegistrySimple* registry) { |
278 registry->RegisterIntegerPref(prefs::kSwReporterExecuteTryCount, 0); | 301 registry->RegisterIntegerPref(prefs::kSwReporterExecuteTryCount, 0); |
| 302 registry->RegisterInt64Pref(prefs::kSwReporterLastTimeTriggered, 0); |
279 } | 303 } |
280 | 304 |
281 } // namespace component_updater | 305 } // namespace component_updater |
OLD | NEW |