blob: 68a5befbe42d2c5c2dd0c3858054a6e476a64e38 [file] [log] [blame]
sorin30474f02017-04-27 00:45:481// Copyright 2017 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 "components/update_client/component.h"
6
7#include <algorithm>
8#include <utility>
9
10#include "base/bind.h"
sorin30474f02017-04-27 00:45:4811#include "base/files/file_util.h"
Sorin Jianuf40ab4b32017-10-06 22:53:4112#include "base/files/scoped_temp_dir.h"
sorin30474f02017-04-27 00:45:4813#include "base/location.h"
14#include "base/logging.h"
sorin30474f02017-04-27 00:45:4815#include "base/single_thread_task_runner.h"
Sorin Jianu039032b2018-10-12 21:48:1316#include "base/strings/string_number_conversions.h"
Gabriel Charette44db1422018-08-06 11:19:3317#include "base/task/post_task.h"
Gabriel Charettedd8d5985e2020-02-26 18:38:3518#include "base/task/thread_pool.h"
sorin30474f02017-04-27 00:45:4819#include "base/threading/thread_task_runner_handle.h"
Sorin Jianu039032b2018-10-12 21:48:1320#include "base/values.h"
Sorin Jianu4ab7c292017-06-15 18:40:2121#include "components/update_client/action_runner.h"
sorincca1c122017-05-11 17:43:2222#include "components/update_client/component_unpacker.h"
sorin30474f02017-04-27 00:45:4823#include "components/update_client/configurator.h"
Sorin Jianu75e6bf22019-02-12 16:07:1224#include "components/update_client/network.h"
Joshua Pawlickid5409e12019-04-06 00:23:1125#include "components/update_client/patcher.h"
Sorin Jianub9258672020-02-19 16:52:1726#include "components/update_client/persisted_data.h"
Sorin Jianu55587d32018-11-14 21:43:2727#include "components/update_client/protocol_definition.h"
Sorin Jianu039032b2018-10-12 21:48:1328#include "components/update_client/protocol_serializer.h"
Sorin Jianuebd652462017-07-23 02:00:5829#include "components/update_client/task_traits.h"
Joshua Pawlickid5409e12019-04-06 00:23:1130#include "components/update_client/unzipper.h"
sorin30474f02017-04-27 00:45:4831#include "components/update_client/update_client.h"
32#include "components/update_client/update_client_errors.h"
33#include "components/update_client/update_engine.h"
34#include "components/update_client/utils.h"
35
36// The state machine representing how a CRX component changes during an update.
37//
Sorin Jianu7c22795b2018-04-26 22:16:5238// +------------------------- kNew
39// | |
40// | V
41// | kChecking
42// | |
43// V error V no no
Sorin Jianu4ab7c292017-06-15 18:40:2144// kUpdateError <------------- [update?] -> [action?] -> kUpToDate kUpdated
45// ^ | | ^ ^
46// | yes | | yes | |
47// | V | | |
48// | kCanUpdate +--------> kRun |
49// | | |
50// | no V |
51// | +-<- [differential update?] |
52// | | | |
53// | | yes | |
54// | | error V |
55// | +-<----- kDownloadingDiff kRun---->-+
56// | | | ^ |
57// | | | yes | |
58// | | error V | |
59// | +-<----- kUpdatingDiff ---------> [action?] ->-+
60// | | ^ no
61// | error V |
62// +-<-------- kDownloading |
63// | | |
64// | | |
65// | error V |
66// +-<-------- kUpdating --------------------------------+
sorin30474f02017-04-27 00:45:4867
68namespace update_client {
69
70namespace {
71
Sorin Jianua8ef73d2017-11-02 16:55:1772using InstallOnBlockingTaskRunnerCompleteCallback = base::OnceCallback<
Minh X. Nguyena4640cb2018-05-23 21:29:1073 void(ErrorCategory error_category, int error_code, int extra_code1)>;
sorin30474f02017-04-27 00:45:4874
Sorin Jianuf40ab4b32017-10-06 22:53:4175void InstallComplete(
Sorin Jianu49126332018-02-13 17:07:4276 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
Sorin Jianua8ef73d2017-11-02 16:55:1777 InstallOnBlockingTaskRunnerCompleteCallback callback,
sorin30474f02017-04-27 00:45:4878 const base::FilePath& unpack_path,
Sorin Jianuf40ab4b32017-10-06 22:53:4179 const CrxInstaller::Result& result) {
Gabriel Charettedd8d5985e2020-02-26 18:38:3580 base::ThreadPool::PostTask(
81 FROM_HERE, {base::TaskPriority::BEST_EFFORT, base::MayBlock()},
Sorin Jianuf40ab4b32017-10-06 22:53:4182 base::BindOnce(
Sorin Jianu49126332018-02-13 17:07:4283 [](scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
Sorin Jianua8ef73d2017-11-02 16:55:1784 InstallOnBlockingTaskRunnerCompleteCallback callback,
Sorin Jianuf40ab4b32017-10-06 22:53:4185 const base::FilePath& unpack_path,
86 const CrxInstaller::Result& result) {
Lei Zhang091d4232019-10-15 21:12:5187 base::DeleteFileRecursively(unpack_path);
Sorin Jianuf40ab4b32017-10-06 22:53:4188 const ErrorCategory error_category =
Sorin Jianuafcb70dd2018-05-16 20:14:1589 result.error ? ErrorCategory::kInstall : ErrorCategory::kNone;
Sorin Jianuf40ab4b32017-10-06 22:53:4190 main_task_runner->PostTask(
Minh X. Nguyena4640cb2018-05-23 21:29:1091 FROM_HERE, base::BindOnce(std::move(callback), error_category,
Sorin Jianua8ef73d2017-11-02 16:55:1792 static_cast<int>(result.error),
93 result.extended_error));
Sorin Jianuf40ab4b32017-10-06 22:53:4194 },
Sorin Jianua8ef73d2017-11-02 16:55:1795 main_task_runner, std::move(callback), unpack_path, result));
sorin30474f02017-04-27 00:45:4896}
97
98void InstallOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:4299 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:48100 const base::FilePath& unpack_path,
Minh X. Nguyenaafd7632017-10-19 21:12:35101 const std::string& public_key,
sorin30474f02017-04-27 00:45:48102 const std::string& fingerprint,
Sorin Jianu9d64af672020-02-05 19:14:34103 std::unique_ptr<CrxInstaller::InstallParams> install_params,
Sorin Jianu49126332018-02-13 17:07:42104 scoped_refptr<CrxInstaller> installer,
Sorin Jianua8ef73d2017-11-02 16:55:17105 InstallOnBlockingTaskRunnerCompleteCallback callback) {
Sorin Jianu23f70f752017-05-30 16:21:58106 DCHECK(base::DirectoryExists(unpack_path));
sorin30474f02017-04-27 00:45:48107
Sorin Jianuf40ab4b32017-10-06 22:53:41108 // Acquire the ownership of the |unpack_path|.
109 base::ScopedTempDir unpack_path_owner;
110 ignore_result(unpack_path_owner.Set(unpack_path));
111
112 if (static_cast<int>(fingerprint.size()) !=
113 base::WriteFile(
114 unpack_path.Append(FILE_PATH_LITERAL("manifest.fingerprint")),
115 fingerprint.c_str(), base::checked_cast<int>(fingerprint.size()))) {
116 const CrxInstaller::Result result(InstallError::FINGERPRINT_WRITE_FAILED);
117 main_task_runner->PostTask(
118 FROM_HERE,
Minh X. Nguyena4640cb2018-05-23 21:29:10119 base::BindOnce(std::move(callback), ErrorCategory::kInstall,
Sorin Jianuf40ab4b32017-10-06 22:53:41120 static_cast<int>(result.error), result.extended_error));
121 return;
122 }
123
Sorin Jianua8ef73d2017-11-02 16:55:17124 installer->Install(
Sorin Jianu9d64af672020-02-05 19:14:34125 unpack_path, public_key, std::move(install_params),
Sorin Jianua8ef73d2017-11-02 16:55:17126 base::BindOnce(&InstallComplete, main_task_runner, std::move(callback),
127 unpack_path_owner.Take()));
sorin30474f02017-04-27 00:45:48128}
129
130void UnpackCompleteOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:42131 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:48132 const base::FilePath& crx_path,
133 const std::string& fingerprint,
Sorin Jianu9d64af672020-02-05 19:14:34134 std::unique_ptr<CrxInstaller::InstallParams> install_params,
Sorin Jianu49126332018-02-13 17:07:42135 scoped_refptr<CrxInstaller> installer,
Sorin Jianua8ef73d2017-11-02 16:55:17136 InstallOnBlockingTaskRunnerCompleteCallback callback,
sorin30474f02017-04-27 00:45:48137 const ComponentUnpacker::Result& result) {
sorin30474f02017-04-27 00:45:48138 update_client::DeleteFileAndEmptyParentDirectory(crx_path);
139
140 if (result.error != UnpackerError::kNone) {
141 main_task_runner->PostTask(
142 FROM_HERE,
Minh X. Nguyena4640cb2018-05-23 21:29:10143 base::BindOnce(std::move(callback), ErrorCategory::kUnpack,
Sorin Jianuebd652462017-07-23 02:00:58144 static_cast<int>(result.error), result.extended_error));
sorin30474f02017-04-27 00:45:48145 return;
146 }
147
Gabriel Charette41ea14152020-03-02 15:43:11148 base::ThreadPool::PostTask(
149 FROM_HERE, kTaskTraits,
150 base::BindOnce(&InstallOnBlockingTaskRunner, main_task_runner,
151 result.unpack_path, result.public_key, fingerprint,
152 std::move(install_params), installer,
153 std::move(callback)));
sorin30474f02017-04-27 00:45:48154}
155
156void StartInstallOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:42157 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:48158 const std::vector<uint8_t>& pk_hash,
159 const base::FilePath& crx_path,
160 const std::string& fingerprint,
Sorin Jianu9d64af672020-02-05 19:14:34161 std::unique_ptr<CrxInstaller::InstallParams> install_params,
Sorin Jianu49126332018-02-13 17:07:42162 scoped_refptr<CrxInstaller> installer,
Joshua Pawlickid5409e12019-04-06 00:23:11163 std::unique_ptr<Unzipper> unzipper_,
164 scoped_refptr<Patcher> patcher_,
Joshua Pawlickif4b33f382018-08-17 17:36:51165 crx_file::VerifierFormat crx_format,
Sorin Jianua8ef73d2017-11-02 16:55:17166 InstallOnBlockingTaskRunnerCompleteCallback callback) {
Taiju Tsuiki36c517d2017-05-18 06:45:43167 auto unpacker = base::MakeRefCounted<ComponentUnpacker>(
Joshua Pawlickid5409e12019-04-06 00:23:11168 pk_hash, crx_path, installer, std::move(unzipper_), std::move(patcher_),
169 crx_format);
sorin30474f02017-04-27 00:45:48170
Sorin Jianu9d64af672020-02-05 19:14:34171 unpacker->Unpack(base::BindOnce(
172 &UnpackCompleteOnBlockingTaskRunner, main_task_runner, crx_path,
173 fingerprint, std::move(install_params), installer, std::move(callback)));
sorin30474f02017-04-27 00:45:48174}
175
Sorin Jianu039032b2018-10-12 21:48:13176// Returns a string literal corresponding to the value of the downloader |d|.
177const char* DownloaderToString(CrxDownloader::DownloadMetrics::Downloader d) {
178 switch (d) {
179 case CrxDownloader::DownloadMetrics::kUrlFetcher:
180 return "direct";
181 case CrxDownloader::DownloadMetrics::kBits:
182 return "bits";
183 default:
184 return "unknown";
185 }
186}
187
sorin30474f02017-04-27 00:45:48188} // namespace
189
190Component::Component(const UpdateContext& update_context, const std::string& id)
191 : id_(id),
Jinho Bangda4e4282018-01-03 13:21:23192 state_(std::make_unique<StateNew>(this)),
sorin30474f02017-04-27 00:45:48193 update_context_(update_context) {}
194
Sorin Jianudfb12a42020-03-10 04:12:03195Component::~Component() = default;
sorin30474f02017-04-27 00:45:48196
Sorin Jianub3296162017-12-13 20:26:32197scoped_refptr<Configurator> Component::config() const {
198 return update_context_.config;
199}
200
Sorin Jianud69d4372018-02-07 19:44:22201std::string Component::session_id() const {
202 return update_context_.session_id;
203}
204
Sorin Jianub41a592a2018-03-02 16:30:27205bool Component::is_foreground() const {
206 return update_context_.is_foreground;
207}
208
Sorin Jianuee5c0db2018-04-12 23:38:47209void Component::Handle(CallbackHandleComplete callback_handle_complete) {
sorin30474f02017-04-27 00:45:48210 DCHECK(thread_checker_.CalledOnValidThread());
211 DCHECK(state_);
212
Sorin Jianuee5c0db2018-04-12 23:38:47213 callback_handle_complete_ = std::move(callback_handle_complete);
sorin30474f02017-04-27 00:45:48214
Sorin Jianua8ef73d2017-11-02 16:55:17215 state_->Handle(
216 base::BindOnce(&Component::ChangeState, base::Unretained(this)));
sorin30474f02017-04-27 00:45:48217}
218
219void Component::ChangeState(std::unique_ptr<State> next_state) {
220 DCHECK(thread_checker_.CalledOnValidThread());
221
Sorin Jianu4ab7c292017-06-15 18:40:21222 previous_state_ = state();
sorin30474f02017-04-27 00:45:48223 if (next_state)
224 state_ = std::move(next_state);
Sorin Jianuee5c0db2018-04-12 23:38:47225 else
226 is_handled_ = true;
sorin30474f02017-04-27 00:45:48227
Sorin Jianua8ef73d2017-11-02 16:55:17228 base::ThreadTaskRunnerHandle::Get()->PostTask(
229 FROM_HERE, std::move(callback_handle_complete_));
sorin30474f02017-04-27 00:45:48230}
231
232CrxUpdateItem Component::GetCrxUpdateItem() const {
233 DCHECK(thread_checker_.CalledOnValidThread());
234
235 CrxUpdateItem crx_update_item;
236 crx_update_item.state = state_->state();
237 crx_update_item.id = id_;
Sorin Jianu7c22795b2018-04-26 22:16:52238 if (crx_component_)
239 crx_update_item.component = *crx_component_;
sorin30474f02017-04-27 00:45:48240 crx_update_item.last_check = last_check_;
241 crx_update_item.next_version = next_version_;
242 crx_update_item.next_fp = next_fp_;
Sorin Jianuafcb70dd2018-05-16 20:14:15243 crx_update_item.error_category = error_category_;
244 crx_update_item.error_code = error_code_;
245 crx_update_item.extra_code1 = extra_code1_;
Joshua Pawlickica4f8fa12020-02-25 23:46:06246 crx_update_item.custom_updatecheck_data = custom_attrs_;
sorin30474f02017-04-27 00:45:48247
248 return crx_update_item;
249}
250
sorin7cff6e52017-05-17 16:37:23251void Component::SetParseResult(const ProtocolParser::Result& result) {
sorin30474f02017-04-27 00:45:48252 DCHECK(thread_checker_.CalledOnValidThread());
253
254 DCHECK_EQ(0, update_check_error_);
255
256 status_ = result.status;
sorin519656c2017-04-28 22:39:34257 action_run_ = result.action_run;
Joshua Pawlickica4f8fa12020-02-25 23:46:06258 custom_attrs_ = result.custom_attributes;
sorin30474f02017-04-27 00:45:48259
260 if (result.manifest.packages.empty())
261 return;
262
263 next_version_ = base::Version(result.manifest.version);
264 const auto& package = result.manifest.packages.front();
265 next_fp_ = package.fingerprint;
266
267 // Resolve the urls by combining the base urls with the package names.
268 for (const auto& crx_url : result.crx_urls) {
269 const GURL url = crx_url.Resolve(package.name);
270 if (url.is_valid())
271 crx_urls_.push_back(url);
272 }
273 for (const auto& crx_diffurl : result.crx_diffurls) {
274 const GURL url = crx_diffurl.Resolve(package.namediff);
275 if (url.is_valid())
276 crx_diffurls_.push_back(url);
277 }
278
279 hash_sha256_ = package.hash_sha256;
280 hashdiff_sha256_ = package.hashdiff_sha256;
Sorin Jianu9d64af672020-02-05 19:14:34281
282 if (!result.manifest.run.empty()) {
283 install_params_ = base::make_optional(CrxInstaller::InstallParams(
284 result.manifest.run, result.manifest.arguments));
285 }
sorin30474f02017-04-27 00:45:48286}
287
288void Component::Uninstall(const base::Version& version, int reason) {
289 DCHECK(thread_checker_.CalledOnValidThread());
290
291 DCHECK_EQ(ComponentState::kNew, state());
292
Sorin Jianu73900242018-08-17 01:11:53293 crx_component_ = CrxComponent();
Sorin Jianu7c22795b2018-04-26 22:16:52294 crx_component_->version = version;
295
sorin30474f02017-04-27 00:45:48296 previous_version_ = version;
297 next_version_ = base::Version("0");
298 extra_code1_ = reason;
299
Jinho Bangda4e4282018-01-03 13:21:23300 state_ = std::make_unique<StateUninstalled>(this);
sorin30474f02017-04-27 00:45:48301}
302
Sorin Jianu888ec292018-06-01 15:35:42303void Component::SetUpdateCheckResult(
304 const base::Optional<ProtocolParser::Result>& result,
305 ErrorCategory error_category,
306 int error) {
sorin30474f02017-04-27 00:45:48307 DCHECK(thread_checker_.CalledOnValidThread());
sorin30474f02017-04-27 00:45:48308 DCHECK_EQ(ComponentState::kChecking, state());
309
Sorin Jianu888ec292018-06-01 15:35:42310 error_category_ = error_category;
311 error_code_ = error;
Joshua Pawlickica4f8fa12020-02-25 23:46:06312
Sorin Jianu888ec292018-06-01 15:35:42313 if (result)
314 SetParseResult(result.value());
315
Sorin Jianua8ef73d2017-11-02 16:55:17316 base::ThreadTaskRunnerHandle::Get()->PostTask(
317 FROM_HERE, std::move(update_check_complete_));
sorin30474f02017-04-27 00:45:48318}
319
Sorin Jianudfb12a42020-03-10 04:12:03320void Component::NotifyWait() {
321 DCHECK(thread_checker_.CalledOnValidThread());
322 NotifyObservers(Events::COMPONENT_WAIT);
323}
324
sorin30474f02017-04-27 00:45:48325bool Component::CanDoBackgroundDownload() const {
Sorin Jianub41a592a2018-03-02 16:30:27326 // Foreground component updates are always downloaded in foreground.
Sorin Jianu7c22795b2018-04-26 22:16:52327 return !is_foreground() &&
328 (crx_component() && crx_component()->allows_background_download) &&
sorin30474f02017-04-27 00:45:48329 update_context_.config->EnabledBackgroundDownloader();
330}
331
Sorin Jianu039032b2018-10-12 21:48:13332void Component::AppendEvent(base::Value event) {
333 events_.push_back(std::move(event));
sorin30474f02017-04-27 00:45:48334}
335
336void Component::NotifyObservers(UpdateClient::Observer::Events event) const {
337 DCHECK(thread_checker_.CalledOnValidThread());
Sorin Jianudfb12a42020-03-10 04:12:03338
339 // There is no corresponding component state for the COMPONENT_WAIT event.
340 if (update_context_.crx_state_change_callback &&
341 event != UpdateClient::Observer::Events::COMPONENT_WAIT) {
342 base::ThreadTaskRunnerHandle::Get()->PostTask(
343 FROM_HERE,
344 base::BindRepeating(update_context_.crx_state_change_callback,
345 GetCrxUpdateItem()));
346 }
sorin30474f02017-04-27 00:45:48347 update_context_.notify_observers_callback.Run(event, id_);
348}
349
350base::TimeDelta Component::GetUpdateDuration() const {
351 DCHECK(thread_checker_.CalledOnValidThread());
352
353 if (update_begin_.is_null())
354 return base::TimeDelta();
355
356 const base::TimeDelta update_cost(base::TimeTicks::Now() - update_begin_);
357 DCHECK_GE(update_cost, base::TimeDelta());
358 const base::TimeDelta max_update_delay =
359 base::TimeDelta::FromSeconds(update_context_.config->UpdateDelay());
360 return std::min(update_cost, max_update_delay);
361}
362
Sorin Jianu039032b2018-10-12 21:48:13363base::Value Component::MakeEventUpdateComplete() const {
364 base::Value event(base::Value::Type::DICTIONARY);
365 event.SetKey("eventtype", base::Value(3));
366 event.SetKey(
367 "eventresult",
368 base::Value(static_cast<int>(state() == ComponentState::kUpdated)));
369 if (error_category() != ErrorCategory::kNone)
370 event.SetKey("errorcat", base::Value(static_cast<int>(error_category())));
371 if (error_code())
372 event.SetKey("errorcode", base::Value(error_code()));
373 if (extra_code1())
374 event.SetKey("extracode1", base::Value(extra_code1()));
375 if (HasDiffUpdate(*this)) {
376 const int diffresult = static_cast<int>(!diff_update_failed());
377 event.SetKey("diffresult", base::Value(diffresult));
378 }
379 if (diff_error_category() != ErrorCategory::kNone) {
380 const int differrorcat = static_cast<int>(diff_error_category());
381 event.SetKey("differrorcat", base::Value(differrorcat));
382 }
383 if (diff_error_code())
384 event.SetKey("differrorcode", base::Value(diff_error_code()));
385 if (diff_extra_code1())
386 event.SetKey("diffextracode1", base::Value(diff_extra_code1()));
387 if (!previous_fp().empty())
388 event.SetKey("previousfp", base::Value(previous_fp()));
389 if (!next_fp().empty())
390 event.SetKey("nextfp", base::Value(next_fp()));
391 DCHECK(previous_version().IsValid());
392 event.SetKey("previousversion", base::Value(previous_version().GetString()));
393 if (next_version().IsValid())
394 event.SetKey("nextversion", base::Value(next_version().GetString()));
395 return event;
396}
397
398base::Value Component::MakeEventDownloadMetrics(
399 const CrxDownloader::DownloadMetrics& dm) const {
400 base::Value event(base::Value::Type::DICTIONARY);
401 event.SetKey("eventtype", base::Value(14));
402 event.SetKey("eventresult", base::Value(static_cast<int>(dm.error == 0)));
403 event.SetKey("downloader", base::Value(DownloaderToString(dm.downloader)));
404 if (dm.error)
405 event.SetKey("errorcode", base::Value(dm.error));
406 event.SetKey("url", base::Value(dm.url.spec()));
407
408 // -1 means that the byte counts are not known.
Sorin Jianu55587d32018-11-14 21:43:27409 if (dm.total_bytes != -1 && dm.total_bytes < kProtocolMaxInt)
410 event.SetKey("total", base::Value(static_cast<double>(dm.total_bytes)));
411 if (dm.downloaded_bytes != -1 && dm.total_bytes < kProtocolMaxInt) {
Sorin Jianu039032b2018-10-12 21:48:13412 event.SetKey("downloaded",
Sorin Jianu55587d32018-11-14 21:43:27413 base::Value(static_cast<double>(dm.downloaded_bytes)));
Sorin Jianu039032b2018-10-12 21:48:13414 }
Sorin Jianu55587d32018-11-14 21:43:27415 if (dm.download_time_ms && dm.total_bytes < kProtocolMaxInt) {
Sorin Jianu039032b2018-10-12 21:48:13416 event.SetKey("download_time_ms",
Sorin Jianu55587d32018-11-14 21:43:27417 base::Value(static_cast<double>(dm.download_time_ms)));
Sorin Jianu039032b2018-10-12 21:48:13418 }
419 DCHECK(previous_version().IsValid());
420 event.SetKey("previousversion", base::Value(previous_version().GetString()));
421 if (next_version().IsValid())
422 event.SetKey("nextversion", base::Value(next_version().GetString()));
423 return event;
424}
425
426base::Value Component::MakeEventUninstalled() const {
427 DCHECK(state() == ComponentState::kUninstalled);
428 base::Value event(base::Value::Type::DICTIONARY);
429 event.SetKey("eventtype", base::Value(4));
430 event.SetKey("eventresult", base::Value(1));
431 if (extra_code1())
432 event.SetKey("extracode1", base::Value(extra_code1()));
433 DCHECK(previous_version().IsValid());
434 event.SetKey("previousversion", base::Value(previous_version().GetString()));
435 DCHECK(next_version().IsValid());
436 event.SetKey("nextversion", base::Value(next_version().GetString()));
437 return event;
438}
439
440base::Value Component::MakeEventActionRun(bool succeeded,
441 int error_code,
442 int extra_code1) const {
443 base::Value event(base::Value::Type::DICTIONARY);
444 event.SetKey("eventtype", base::Value(42));
445 event.SetKey("eventresult", base::Value(static_cast<int>(succeeded)));
446 if (error_code)
447 event.SetKey("errorcode", base::Value(error_code));
448 if (extra_code1)
449 event.SetKey("extracode1", base::Value(extra_code1));
450 return event;
451}
452
453std::vector<base::Value> Component::GetEvents() const {
454 std::vector<base::Value> events;
455 for (const auto& event : events_)
456 events.push_back(event.Clone());
457 return events;
458}
459
Sorin Jianu9d64af672020-02-05 19:14:34460std::unique_ptr<CrxInstaller::InstallParams> Component::install_params() const {
461 return install_params_
462 ? std::make_unique<CrxInstaller::InstallParams>(*install_params_)
463 : nullptr;
464}
465
sorin30474f02017-04-27 00:45:48466Component::State::State(Component* component, ComponentState state)
467 : state_(state), component_(*component) {}
468
Sorin Jianu30881152020-03-16 14:31:19469Component::State::~State() = default;
sorin30474f02017-04-27 00:45:48470
Sorin Jianuee5c0db2018-04-12 23:38:47471void Component::State::Handle(CallbackNextState callback_next_state) {
sorin30474f02017-04-27 00:45:48472 DCHECK(thread_checker_.CalledOnValidThread());
473
Sorin Jianuee5c0db2018-04-12 23:38:47474 callback_next_state_ = std::move(callback_next_state);
sorin30474f02017-04-27 00:45:48475
sorin30474f02017-04-27 00:45:48476 DoHandle();
477}
478
479void Component::State::TransitionState(std::unique_ptr<State> next_state) {
Sorin Jianuee5c0db2018-04-12 23:38:47480 DCHECK(thread_checker_.CalledOnValidThread());
481 DCHECK(next_state);
sorin30474f02017-04-27 00:45:48482
483 base::ThreadTaskRunnerHandle::Get()->PostTask(
Sorin Jianuee5c0db2018-04-12 23:38:47484 FROM_HERE,
485 base::BindOnce(std::move(callback_next_state_), std::move(next_state)));
486}
487
488void Component::State::EndState() {
489 DCHECK(thread_checker_.CalledOnValidThread());
490
491 base::ThreadTaskRunnerHandle::Get()->PostTask(
492 FROM_HERE, base::BindOnce(std::move(callback_next_state_), nullptr));
sorin30474f02017-04-27 00:45:48493}
494
495Component::StateNew::StateNew(Component* component)
496 : State(component, ComponentState::kNew) {}
497
498Component::StateNew::~StateNew() {
499 DCHECK(thread_checker_.CalledOnValidThread());
500}
501
502void Component::StateNew::DoHandle() {
503 DCHECK(thread_checker_.CalledOnValidThread());
504
505 auto& component = State::component();
Sorin Jianucb4431a2018-04-30 20:59:24506 if (component.crx_component()) {
507 TransitionState(std::make_unique<StateChecking>(&component));
508 } else {
509 component.error_code_ = static_cast<int>(Error::CRX_NOT_FOUND);
Minh X. Nguyena4640cb2018-05-23 21:29:10510 component.error_category_ = ErrorCategory::kService;
Sorin Jianucb4431a2018-04-30 20:59:24511 TransitionState(std::make_unique<StateUpdateError>(&component));
512 }
sorin30474f02017-04-27 00:45:48513}
514
515Component::StateChecking::StateChecking(Component* component)
516 : State(component, ComponentState::kChecking) {}
517
518Component::StateChecking::~StateChecking() {
519 DCHECK(thread_checker_.CalledOnValidThread());
520}
521
522// Unlike how other states are handled, this function does not change the
523// state right away. The state transition happens when the UpdateChecker
524// calls Component::UpdateCheckComplete and |update_check_complete_| is invoked.
525// This is an artifact of how multiple components must be checked for updates
526// together but the state machine defines the transitions for one component
527// at a time.
528void Component::StateChecking::DoHandle() {
529 DCHECK(thread_checker_.CalledOnValidThread());
530
531 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52532 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48533
534 component.last_check_ = base::TimeTicks::Now();
Sorin Jianua8ef73d2017-11-02 16:55:17535 component.update_check_complete_ = base::BindOnce(
sorin30474f02017-04-27 00:45:48536 &Component::StateChecking::UpdateCheckComplete, base::Unretained(this));
537
538 component.NotifyObservers(Events::COMPONENT_CHECKING_FOR_UPDATES);
539}
540
541void Component::StateChecking::UpdateCheckComplete() {
542 DCHECK(thread_checker_.CalledOnValidThread());
543 auto& component = State::component();
Sorin Jianuafcb70dd2018-05-16 20:14:15544 if (!component.error_code_) {
sorin30474f02017-04-27 00:45:48545 if (component.status_ == "ok") {
Jinho Bangda4e4282018-01-03 13:21:23546 TransitionState(std::make_unique<StateCanUpdate>(&component));
sorin30474f02017-04-27 00:45:48547 return;
548 }
549
550 if (component.status_ == "noupdate") {
Sorin Jianu4ab7c292017-06-15 18:40:21551 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23552 TransitionState(std::make_unique<StateUpToDate>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21553 else
Jinho Bangda4e4282018-01-03 13:21:23554 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48555 return;
556 }
557 }
558
Jinho Bangda4e4282018-01-03 13:21:23559 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48560}
561
562Component::StateUpdateError::StateUpdateError(Component* component)
563 : State(component, ComponentState::kUpdateError) {}
564
565Component::StateUpdateError::~StateUpdateError() {
566 DCHECK(thread_checker_.CalledOnValidThread());
567}
568
569void Component::StateUpdateError::DoHandle() {
570 DCHECK(thread_checker_.CalledOnValidThread());
571
572 auto& component = State::component();
sorin117334f2017-05-19 02:36:25573
Sorin Jianu888ec292018-06-01 15:35:42574 DCHECK_NE(ErrorCategory::kNone, component.error_category_);
575 DCHECK_NE(0, component.error_code_);
576
sorin117334f2017-05-19 02:36:25577 // Create an event only when the server response included an update.
578 if (component.IsUpdateAvailable())
Sorin Jianu039032b2018-10-12 21:48:13579 component.AppendEvent(component.MakeEventUpdateComplete());
sorin117334f2017-05-19 02:36:25580
Sorin Jianuee5c0db2018-04-12 23:38:47581 EndState();
Sorin Jianucbb10e12018-01-23 18:01:44582 component.NotifyObservers(Events::COMPONENT_UPDATE_ERROR);
sorin30474f02017-04-27 00:45:48583}
584
585Component::StateCanUpdate::StateCanUpdate(Component* component)
586 : State(component, ComponentState::kCanUpdate) {}
587
588Component::StateCanUpdate::~StateCanUpdate() {
589 DCHECK(thread_checker_.CalledOnValidThread());
590}
591
592void Component::StateCanUpdate::DoHandle() {
593 DCHECK(thread_checker_.CalledOnValidThread());
594
595 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52596 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48597
598 component.is_update_available_ = true;
599 component.NotifyObservers(Events::COMPONENT_UPDATE_FOUND);
600
Sorin Jianu7c22795b2018-04-26 22:16:52601 if (component.crx_component()
602 ->supports_group_policy_enable_component_updates &&
sorin30474f02017-04-27 00:45:48603 !component.update_context_.enabled_component_updates) {
Minh X. Nguyena4640cb2018-05-23 21:29:10604 component.error_category_ = ErrorCategory::kService;
sorin30474f02017-04-27 00:45:48605 component.error_code_ = static_cast<int>(ServiceError::UPDATE_DISABLED);
606 component.extra_code1_ = 0;
Jinho Bangda4e4282018-01-03 13:21:23607 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48608 return;
609 }
610
611 // Start computing the cost of the this update from here on.
612 component.update_begin_ = base::TimeTicks::Now();
613
614 if (CanTryDiffUpdate())
Jinho Bangda4e4282018-01-03 13:21:23615 TransitionState(std::make_unique<StateDownloadingDiff>(&component));
sorin30474f02017-04-27 00:45:48616 else
Jinho Bangda4e4282018-01-03 13:21:23617 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48618}
619
620// Returns true if a differential update is available, it has not failed yet,
621// and the configuration allows this update.
622bool Component::StateCanUpdate::CanTryDiffUpdate() const {
623 const auto& component = Component::State::component();
624 return HasDiffUpdate(component) && !component.diff_error_code_ &&
625 component.update_context_.config->EnabledDeltas();
626}
627
628Component::StateUpToDate::StateUpToDate(Component* component)
629 : State(component, ComponentState::kUpToDate) {}
630
631Component::StateUpToDate::~StateUpToDate() {
632 DCHECK(thread_checker_.CalledOnValidThread());
633}
634
635void Component::StateUpToDate::DoHandle() {
636 DCHECK(thread_checker_.CalledOnValidThread());
637
638 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52639 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48640
sorin30474f02017-04-27 00:45:48641 component.NotifyObservers(Events::COMPONENT_NOT_UPDATED);
Sorin Jianuee5c0db2018-04-12 23:38:47642 EndState();
sorin30474f02017-04-27 00:45:48643}
644
645Component::StateDownloadingDiff::StateDownloadingDiff(Component* component)
646 : State(component, ComponentState::kDownloadingDiff) {}
647
648Component::StateDownloadingDiff::~StateDownloadingDiff() {
649 DCHECK(thread_checker_.CalledOnValidThread());
650}
651
652void Component::StateDownloadingDiff::DoHandle() {
653 DCHECK(thread_checker_.CalledOnValidThread());
654
655 const auto& component = Component::State::component();
656 const auto& update_context = component.update_context_;
657
Sorin Jianu7c22795b2018-04-26 22:16:52658 DCHECK(component.crx_component());
659
sorin30474f02017-04-27 00:45:48660 crx_downloader_ = update_context.crx_downloader_factory(
661 component.CanDoBackgroundDownload(),
Sorin Jianu75e6bf22019-02-12 16:07:12662 update_context.config->GetNetworkFetcherFactory());
sorin30474f02017-04-27 00:45:48663
664 const auto& id = component.id_;
665 crx_downloader_->set_progress_callback(
Ken Rockot942a6382019-12-18 21:31:41666 base::BindRepeating(&Component::StateDownloadingDiff::DownloadProgress,
667 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48668 crx_downloader_->StartDownload(
669 component.crx_diffurls_, component.hashdiff_sha256_,
Sorin Jianua8ef73d2017-11-02 16:55:17670 base::BindOnce(&Component::StateDownloadingDiff::DownloadComplete,
671 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48672
673 component.NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
674}
675
Antonio Gomes31237fb2018-08-27 19:11:03676// Called when progress is being made downloading a CRX. Can be called multiple
677// times due to how the CRX downloader switches between different downloaders
678// and fallback urls.
679void Component::StateDownloadingDiff::DownloadProgress(const std::string& id) {
sorin30474f02017-04-27 00:45:48680 DCHECK(thread_checker_.CalledOnValidThread());
681
682 component().NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
683}
684
685void Component::StateDownloadingDiff::DownloadComplete(
686 const std::string& id,
687 const CrxDownloader::Result& download_result) {
688 DCHECK(thread_checker_.CalledOnValidThread());
689
690 auto& component = Component::State::component();
Sorin Jianu039032b2018-10-12 21:48:13691 for (const auto& download_metrics : crx_downloader_->download_metrics())
692 component.AppendEvent(component.MakeEventDownloadMetrics(download_metrics));
sorin30474f02017-04-27 00:45:48693
694 crx_downloader_.reset();
695
696 if (download_result.error) {
Sorin Jianu52de5fa2017-10-09 23:19:33697 DCHECK(download_result.response.empty());
Minh X. Nguyena4640cb2018-05-23 21:29:10698 component.diff_error_category_ = ErrorCategory::kDownload;
sorin30474f02017-04-27 00:45:48699 component.diff_error_code_ = download_result.error;
700
Jinho Bangda4e4282018-01-03 13:21:23701 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48702 return;
703 }
704
705 component.crx_path_ = download_result.response;
706
Jinho Bangda4e4282018-01-03 13:21:23707 TransitionState(std::make_unique<StateUpdatingDiff>(&component));
sorin30474f02017-04-27 00:45:48708}
709
710Component::StateDownloading::StateDownloading(Component* component)
711 : State(component, ComponentState::kDownloading) {}
712
713Component::StateDownloading::~StateDownloading() {
714 DCHECK(thread_checker_.CalledOnValidThread());
715}
716
717void Component::StateDownloading::DoHandle() {
718 DCHECK(thread_checker_.CalledOnValidThread());
719
720 const auto& component = Component::State::component();
721 const auto& update_context = component.update_context_;
722
Sorin Jianu7c22795b2018-04-26 22:16:52723 DCHECK(component.crx_component());
724
sorin30474f02017-04-27 00:45:48725 crx_downloader_ = update_context.crx_downloader_factory(
726 component.CanDoBackgroundDownload(),
Sorin Jianu75e6bf22019-02-12 16:07:12727 update_context.config->GetNetworkFetcherFactory());
sorin30474f02017-04-27 00:45:48728
729 const auto& id = component.id_;
730 crx_downloader_->set_progress_callback(
Ken Rockot942a6382019-12-18 21:31:41731 base::BindRepeating(&Component::StateDownloading::DownloadProgress,
732 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48733 crx_downloader_->StartDownload(
734 component.crx_urls_, component.hash_sha256_,
Sorin Jianua8ef73d2017-11-02 16:55:17735 base::BindOnce(&Component::StateDownloading::DownloadComplete,
736 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48737
738 component.NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
739}
740
Antonio Gomes31237fb2018-08-27 19:11:03741// Called when progress is being made downloading a CRX. Can be called multiple
742// times due to how the CRX downloader switches between different downloaders
743// and fallback urls.
744void Component::StateDownloading::DownloadProgress(const std::string& id) {
sorin30474f02017-04-27 00:45:48745 DCHECK(thread_checker_.CalledOnValidThread());
746
747 component().NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
748}
749
750void Component::StateDownloading::DownloadComplete(
751 const std::string& id,
752 const CrxDownloader::Result& download_result) {
753 DCHECK(thread_checker_.CalledOnValidThread());
754
755 auto& component = Component::State::component();
756
Sorin Jianu039032b2018-10-12 21:48:13757 for (const auto& download_metrics : crx_downloader_->download_metrics())
758 component.AppendEvent(component.MakeEventDownloadMetrics(download_metrics));
sorin30474f02017-04-27 00:45:48759
760 crx_downloader_.reset();
761
762 if (download_result.error) {
Sorin Jianu52de5fa2017-10-09 23:19:33763 DCHECK(download_result.response.empty());
Minh X. Nguyena4640cb2018-05-23 21:29:10764 component.error_category_ = ErrorCategory::kDownload;
sorin30474f02017-04-27 00:45:48765 component.error_code_ = download_result.error;
766
Jinho Bangda4e4282018-01-03 13:21:23767 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48768 return;
769 }
770
771 component.crx_path_ = download_result.response;
772
Jinho Bangda4e4282018-01-03 13:21:23773 TransitionState(std::make_unique<StateUpdating>(&component));
sorin30474f02017-04-27 00:45:48774}
775
776Component::StateUpdatingDiff::StateUpdatingDiff(Component* component)
777 : State(component, ComponentState::kUpdatingDiff) {}
778
779Component::StateUpdatingDiff::~StateUpdatingDiff() {
780 DCHECK(thread_checker_.CalledOnValidThread());
781}
782
783void Component::StateUpdatingDiff::DoHandle() {
784 DCHECK(thread_checker_.CalledOnValidThread());
785
786 const auto& component = Component::State::component();
787 const auto& update_context = component.update_context_;
788
Sorin Jianu7c22795b2018-04-26 22:16:52789 DCHECK(component.crx_component());
790
sorin30474f02017-04-27 00:45:48791 component.NotifyObservers(Events::COMPONENT_UPDATE_READY);
792
Gabriel Charette41ea14152020-03-02 15:43:11793 base::ThreadPool::CreateSequencedTaskRunner(kTaskTraits)
Sorin Jianua8ef73d2017-11-02 16:55:17794 ->PostTask(
795 FROM_HERE,
796 base::BindOnce(
797 &update_client::StartInstallOnBlockingTaskRunner,
798 base::ThreadTaskRunnerHandle::Get(),
Sorin Jianu7c22795b2018-04-26 22:16:52799 component.crx_component()->pk_hash, component.crx_path_,
Sorin Jianu9d64af672020-02-05 19:14:34800 component.next_fp_, component.install_params(),
801 component.crx_component()->installer,
Joshua Pawlickid5409e12019-04-06 00:23:11802 update_context.config->GetUnzipperFactory()->Create(),
803 update_context.config->GetPatcherFactory()->Create(),
Joshua Pawlickif4b33f382018-08-17 17:36:51804 component.crx_component()->crx_format_requirement,
Sorin Jianua8ef73d2017-11-02 16:55:17805 base::BindOnce(&Component::StateUpdatingDiff::InstallComplete,
806 base::Unretained(this))));
sorin30474f02017-04-27 00:45:48807}
808
Minh X. Nguyena4640cb2018-05-23 21:29:10809void Component::StateUpdatingDiff::InstallComplete(ErrorCategory error_category,
sorin30474f02017-04-27 00:45:48810 int error_code,
811 int extra_code1) {
812 DCHECK(thread_checker_.CalledOnValidThread());
813
814 auto& component = Component::State::component();
815
816 component.diff_error_category_ = error_category;
817 component.diff_error_code_ = error_code;
818 component.diff_extra_code1_ = extra_code1;
819
820 if (component.diff_error_code_ != 0) {
Jinho Bangda4e4282018-01-03 13:21:23821 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48822 return;
823 }
824
Minh X. Nguyena4640cb2018-05-23 21:29:10825 DCHECK_EQ(ErrorCategory::kNone, component.diff_error_category_);
sorin30474f02017-04-27 00:45:48826 DCHECK_EQ(0, component.diff_error_code_);
827 DCHECK_EQ(0, component.diff_extra_code1_);
828
Minh X. Nguyena4640cb2018-05-23 21:29:10829 DCHECK_EQ(ErrorCategory::kNone, component.error_category_);
sorin30474f02017-04-27 00:45:48830 DCHECK_EQ(0, component.error_code_);
831 DCHECK_EQ(0, component.extra_code1_);
832
Sorin Jianu4ab7c292017-06-15 18:40:21833 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23834 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21835 else
Jinho Bangda4e4282018-01-03 13:21:23836 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48837}
838
839Component::StateUpdating::StateUpdating(Component* component)
sorinff403ef2017-05-16 21:48:42840 : State(component, ComponentState::kUpdating) {}
sorin30474f02017-04-27 00:45:48841
842Component::StateUpdating::~StateUpdating() {
843 DCHECK(thread_checker_.CalledOnValidThread());
844}
845
846void Component::StateUpdating::DoHandle() {
847 DCHECK(thread_checker_.CalledOnValidThread());
848
849 const auto& component = Component::State::component();
850 const auto& update_context = component.update_context_;
851
Sorin Jianu7c22795b2018-04-26 22:16:52852 DCHECK(component.crx_component());
853
sorin30474f02017-04-27 00:45:48854 component.NotifyObservers(Events::COMPONENT_UPDATE_READY);
855
Gabriel Charette41ea14152020-03-02 15:43:11856 base::ThreadPool::CreateSequencedTaskRunner(kTaskTraits)
Sorin Jianua8ef73d2017-11-02 16:55:17857 ->PostTask(FROM_HERE,
858 base::BindOnce(
859 &update_client::StartInstallOnBlockingTaskRunner,
860 base::ThreadTaskRunnerHandle::Get(),
Sorin Jianu7c22795b2018-04-26 22:16:52861 component.crx_component()->pk_hash, component.crx_path_,
Sorin Jianu9d64af672020-02-05 19:14:34862 component.next_fp_, component.install_params(),
863 component.crx_component()->installer,
Joshua Pawlickid5409e12019-04-06 00:23:11864 update_context.config->GetUnzipperFactory()->Create(),
865 update_context.config->GetPatcherFactory()->Create(),
Joshua Pawlickif4b33f382018-08-17 17:36:51866 component.crx_component()->crx_format_requirement,
Sorin Jianua8ef73d2017-11-02 16:55:17867 base::BindOnce(&Component::StateUpdating::InstallComplete,
Sorin Jianuebd652462017-07-23 02:00:58868 base::Unretained(this))));
sorin30474f02017-04-27 00:45:48869}
870
Minh X. Nguyena4640cb2018-05-23 21:29:10871void Component::StateUpdating::InstallComplete(ErrorCategory error_category,
sorin30474f02017-04-27 00:45:48872 int error_code,
873 int extra_code1) {
874 DCHECK(thread_checker_.CalledOnValidThread());
875
876 auto& component = Component::State::component();
877
878 component.error_category_ = error_category;
879 component.error_code_ = error_code;
880 component.extra_code1_ = extra_code1;
881
882 if (component.error_code_ != 0) {
Jinho Bangda4e4282018-01-03 13:21:23883 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48884 return;
885 }
886
Minh X. Nguyena4640cb2018-05-23 21:29:10887 DCHECK_EQ(ErrorCategory::kNone, component.error_category_);
sorin30474f02017-04-27 00:45:48888 DCHECK_EQ(0, component.error_code_);
889 DCHECK_EQ(0, component.extra_code1_);
890
Sorin Jianu4ab7c292017-06-15 18:40:21891 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23892 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21893 else
Jinho Bangda4e4282018-01-03 13:21:23894 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48895}
896
897Component::StateUpdated::StateUpdated(Component* component)
898 : State(component, ComponentState::kUpdated) {
899 DCHECK(thread_checker_.CalledOnValidThread());
900}
901
902Component::StateUpdated::~StateUpdated() {
903 DCHECK(thread_checker_.CalledOnValidThread());
904}
905
906void Component::StateUpdated::DoHandle() {
907 DCHECK(thread_checker_.CalledOnValidThread());
908
909 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52910 DCHECK(component.crx_component());
911
912 component.crx_component_->version = component.next_version_;
913 component.crx_component_->fingerprint = component.next_fp_;
sorin30474f02017-04-27 00:45:48914
Sorin Jianub9258672020-02-19 16:52:17915 component.update_context_.persisted_data->SetProductVersion(
916 component.id(), component.crx_component_->version);
917 component.update_context_.persisted_data->SetFingerprint(
918 component.id(), component.crx_component_->fingerprint);
919
Sorin Jianu039032b2018-10-12 21:48:13920 component.AppendEvent(component.MakeEventUpdateComplete());
sorin117334f2017-05-19 02:36:25921
sorin30474f02017-04-27 00:45:48922 component.NotifyObservers(Events::COMPONENT_UPDATED);
Sorin Jianuee5c0db2018-04-12 23:38:47923 EndState();
sorin30474f02017-04-27 00:45:48924}
925
926Component::StateUninstalled::StateUninstalled(Component* component)
927 : State(component, ComponentState::kUninstalled) {
928 DCHECK(thread_checker_.CalledOnValidThread());
929}
930
931Component::StateUninstalled::~StateUninstalled() {
932 DCHECK(thread_checker_.CalledOnValidThread());
933}
934
935void Component::StateUninstalled::DoHandle() {
936 DCHECK(thread_checker_.CalledOnValidThread());
sorin117334f2017-05-19 02:36:25937
938 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52939 DCHECK(component.crx_component());
940
Sorin Jianu039032b2018-10-12 21:48:13941 component.AppendEvent(component.MakeEventUninstalled());
sorin117334f2017-05-19 02:36:25942
Sorin Jianuee5c0db2018-04-12 23:38:47943 EndState();
sorin30474f02017-04-27 00:45:48944}
945
Sorin Jianu4ab7c292017-06-15 18:40:21946Component::StateRun::StateRun(Component* component)
947 : State(component, ComponentState::kRun) {}
948
949Component::StateRun::~StateRun() {
950 DCHECK(thread_checker_.CalledOnValidThread());
951}
952
953void Component::StateRun::DoHandle() {
954 DCHECK(thread_checker_.CalledOnValidThread());
955
956 const auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52957 DCHECK(component.crx_component());
958
Jinho Bangda4e4282018-01-03 13:21:23959 action_runner_ = std::make_unique<ActionRunner>(component);
Sorin Jianu4ab7c292017-06-15 18:40:21960 action_runner_->Run(
Sorin Jianua8ef73d2017-11-02 16:55:17961 base::BindOnce(&StateRun::ActionRunComplete, base::Unretained(this)));
Sorin Jianu4ab7c292017-06-15 18:40:21962}
963
964void Component::StateRun::ActionRunComplete(bool succeeded,
965 int error_code,
966 int extra_code1) {
967 DCHECK(thread_checker_.CalledOnValidThread());
968
969 auto& component = State::component();
970
971 component.AppendEvent(
Sorin Jianu039032b2018-10-12 21:48:13972 component.MakeEventActionRun(succeeded, error_code, extra_code1));
Sorin Jianu4ab7c292017-06-15 18:40:21973 switch (component.previous_state_) {
974 case ComponentState::kChecking:
Jinho Bangda4e4282018-01-03 13:21:23975 TransitionState(std::make_unique<StateUpToDate>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21976 return;
977 case ComponentState::kUpdating:
978 case ComponentState::kUpdatingDiff:
Jinho Bangda4e4282018-01-03 13:21:23979 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21980 return;
981 default:
982 break;
983 }
984 NOTREACHED();
985}
986
sorin30474f02017-04-27 00:45:48987} // namespace update_client