blob: 36e11d573802e8ff82ad5535a48d075e84271ddb [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"
sorin30474f02017-04-27 00:45:4818#include "base/threading/thread_task_runner_handle.h"
Sorin Jianu039032b2018-10-12 21:48:1319#include "base/values.h"
Sorin Jianu4ab7c292017-06-15 18:40:2120#include "components/update_client/action_runner.h"
sorincca1c122017-05-11 17:43:2221#include "components/update_client/component_unpacker.h"
sorin30474f02017-04-27 00:45:4822#include "components/update_client/configurator.h"
Sorin Jianu75e6bf22019-02-12 16:07:1223#include "components/update_client/network.h"
Joshua Pawlickid5409e12019-04-06 00:23:1124#include "components/update_client/patcher.h"
Sorin Jianu55587d32018-11-14 21:43:2725#include "components/update_client/protocol_definition.h"
Sorin Jianu039032b2018-10-12 21:48:1326#include "components/update_client/protocol_serializer.h"
Sorin Jianuebd652462017-07-23 02:00:5827#include "components/update_client/task_traits.h"
Joshua Pawlickid5409e12019-04-06 00:23:1128#include "components/update_client/unzipper.h"
sorin30474f02017-04-27 00:45:4829#include "components/update_client/update_client.h"
30#include "components/update_client/update_client_errors.h"
31#include "components/update_client/update_engine.h"
32#include "components/update_client/utils.h"
33
34// The state machine representing how a CRX component changes during an update.
35//
Sorin Jianu7c22795b2018-04-26 22:16:5236// +------------------------- kNew
37// | |
38// | V
39// | kChecking
40// | |
41// V error V no no
Sorin Jianu4ab7c292017-06-15 18:40:2142// kUpdateError <------------- [update?] -> [action?] -> kUpToDate kUpdated
43// ^ | | ^ ^
44// | yes | | yes | |
45// | V | | |
46// | kCanUpdate +--------> kRun |
47// | | |
48// | no V |
49// | +-<- [differential update?] |
50// | | | |
51// | | yes | |
52// | | error V |
53// | +-<----- kDownloadingDiff kRun---->-+
54// | | | ^ |
55// | | | yes | |
56// | | error V | |
57// | +-<----- kUpdatingDiff ---------> [action?] ->-+
58// | | ^ no
59// | error V |
60// +-<-------- kDownloading |
61// | | |
62// | | |
63// | error V |
64// +-<-------- kUpdating --------------------------------+
sorin30474f02017-04-27 00:45:4865
66namespace update_client {
67
68namespace {
69
Sorin Jianua8ef73d2017-11-02 16:55:1770using InstallOnBlockingTaskRunnerCompleteCallback = base::OnceCallback<
Minh X. Nguyena4640cb2018-05-23 21:29:1071 void(ErrorCategory error_category, int error_code, int extra_code1)>;
sorin30474f02017-04-27 00:45:4872
Sorin Jianuf40ab4b32017-10-06 22:53:4173void InstallComplete(
Sorin Jianu49126332018-02-13 17:07:4274 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
Sorin Jianua8ef73d2017-11-02 16:55:1775 InstallOnBlockingTaskRunnerCompleteCallback callback,
sorin30474f02017-04-27 00:45:4876 const base::FilePath& unpack_path,
Sorin Jianuf40ab4b32017-10-06 22:53:4177 const CrxInstaller::Result& result) {
Sami Kyostila14ca7642019-08-01 17:52:2578 base::PostTask(
79 FROM_HERE,
80 {base::ThreadPool(), base::TaskPriority::BEST_EFFORT, base::MayBlock()},
Sorin Jianuf40ab4b32017-10-06 22:53:4181 base::BindOnce(
Sorin Jianu49126332018-02-13 17:07:4282 [](scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
Sorin Jianua8ef73d2017-11-02 16:55:1783 InstallOnBlockingTaskRunnerCompleteCallback callback,
Sorin Jianuf40ab4b32017-10-06 22:53:4184 const base::FilePath& unpack_path,
85 const CrxInstaller::Result& result) {
Lei Zhang091d4232019-10-15 21:12:5186 base::DeleteFileRecursively(unpack_path);
Sorin Jianuf40ab4b32017-10-06 22:53:4187 const ErrorCategory error_category =
Sorin Jianuafcb70dd2018-05-16 20:14:1588 result.error ? ErrorCategory::kInstall : ErrorCategory::kNone;
Sorin Jianuf40ab4b32017-10-06 22:53:4189 main_task_runner->PostTask(
Minh X. Nguyena4640cb2018-05-23 21:29:1090 FROM_HERE, base::BindOnce(std::move(callback), error_category,
Sorin Jianua8ef73d2017-11-02 16:55:1791 static_cast<int>(result.error),
92 result.extended_error));
Sorin Jianuf40ab4b32017-10-06 22:53:4193 },
Sorin Jianua8ef73d2017-11-02 16:55:1794 main_task_runner, std::move(callback), unpack_path, result));
sorin30474f02017-04-27 00:45:4895}
96
97void InstallOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:4298 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:4899 const base::FilePath& unpack_path,
Minh X. Nguyenaafd7632017-10-19 21:12:35100 const std::string& public_key,
sorin30474f02017-04-27 00:45:48101 const std::string& fingerprint,
Sorin Jianu49126332018-02-13 17:07:42102 scoped_refptr<CrxInstaller> installer,
Sorin Jianua8ef73d2017-11-02 16:55:17103 InstallOnBlockingTaskRunnerCompleteCallback callback) {
Sorin Jianu23f70f752017-05-30 16:21:58104 DCHECK(base::DirectoryExists(unpack_path));
sorin30474f02017-04-27 00:45:48105
Sorin Jianuf40ab4b32017-10-06 22:53:41106 // Acquire the ownership of the |unpack_path|.
107 base::ScopedTempDir unpack_path_owner;
108 ignore_result(unpack_path_owner.Set(unpack_path));
109
110 if (static_cast<int>(fingerprint.size()) !=
111 base::WriteFile(
112 unpack_path.Append(FILE_PATH_LITERAL("manifest.fingerprint")),
113 fingerprint.c_str(), base::checked_cast<int>(fingerprint.size()))) {
114 const CrxInstaller::Result result(InstallError::FINGERPRINT_WRITE_FAILED);
115 main_task_runner->PostTask(
116 FROM_HERE,
Minh X. Nguyena4640cb2018-05-23 21:29:10117 base::BindOnce(std::move(callback), ErrorCategory::kInstall,
Sorin Jianuf40ab4b32017-10-06 22:53:41118 static_cast<int>(result.error), result.extended_error));
119 return;
120 }
121
Sorin Jianua8ef73d2017-11-02 16:55:17122 installer->Install(
123 unpack_path, public_key,
124 base::BindOnce(&InstallComplete, main_task_runner, std::move(callback),
125 unpack_path_owner.Take()));
sorin30474f02017-04-27 00:45:48126}
127
128void UnpackCompleteOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:42129 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:48130 const base::FilePath& crx_path,
131 const std::string& fingerprint,
Sorin Jianu49126332018-02-13 17:07:42132 scoped_refptr<CrxInstaller> installer,
Sorin Jianua8ef73d2017-11-02 16:55:17133 InstallOnBlockingTaskRunnerCompleteCallback callback,
sorin30474f02017-04-27 00:45:48134 const ComponentUnpacker::Result& result) {
sorin30474f02017-04-27 00:45:48135 update_client::DeleteFileAndEmptyParentDirectory(crx_path);
136
137 if (result.error != UnpackerError::kNone) {
138 main_task_runner->PostTask(
139 FROM_HERE,
Minh X. Nguyena4640cb2018-05-23 21:29:10140 base::BindOnce(std::move(callback), ErrorCategory::kUnpack,
Sorin Jianuebd652462017-07-23 02:00:58141 static_cast<int>(result.error), result.extended_error));
sorin30474f02017-04-27 00:45:48142 return;
143 }
144
Sami Kyostila14ca7642019-08-01 17:52:25145 base::PostTask(FROM_HERE, kTaskTraits,
146 base::BindOnce(&InstallOnBlockingTaskRunner, main_task_runner,
147 result.unpack_path, result.public_key,
148 fingerprint, installer, std::move(callback)));
sorin30474f02017-04-27 00:45:48149}
150
151void StartInstallOnBlockingTaskRunner(
Sorin Jianu49126332018-02-13 17:07:42152 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
sorin30474f02017-04-27 00:45:48153 const std::vector<uint8_t>& pk_hash,
154 const base::FilePath& crx_path,
155 const std::string& fingerprint,
Sorin Jianu49126332018-02-13 17:07:42156 scoped_refptr<CrxInstaller> installer,
Joshua Pawlickid5409e12019-04-06 00:23:11157 std::unique_ptr<Unzipper> unzipper_,
158 scoped_refptr<Patcher> patcher_,
Joshua Pawlickif4b33f382018-08-17 17:36:51159 crx_file::VerifierFormat crx_format,
Sorin Jianua8ef73d2017-11-02 16:55:17160 InstallOnBlockingTaskRunnerCompleteCallback callback) {
Taiju Tsuiki36c517d2017-05-18 06:45:43161 auto unpacker = base::MakeRefCounted<ComponentUnpacker>(
Joshua Pawlickid5409e12019-04-06 00:23:11162 pk_hash, crx_path, installer, std::move(unzipper_), std::move(patcher_),
163 crx_format);
sorin30474f02017-04-27 00:45:48164
Sorin Jianua8ef73d2017-11-02 16:55:17165 unpacker->Unpack(base::BindOnce(&UnpackCompleteOnBlockingTaskRunner,
166 main_task_runner, crx_path, fingerprint,
167 installer, std::move(callback)));
sorin30474f02017-04-27 00:45:48168}
169
Sorin Jianu039032b2018-10-12 21:48:13170// Returns a string literal corresponding to the value of the downloader |d|.
171const char* DownloaderToString(CrxDownloader::DownloadMetrics::Downloader d) {
172 switch (d) {
173 case CrxDownloader::DownloadMetrics::kUrlFetcher:
174 return "direct";
175 case CrxDownloader::DownloadMetrics::kBits:
176 return "bits";
177 default:
178 return "unknown";
179 }
180}
181
sorin30474f02017-04-27 00:45:48182} // namespace
183
184Component::Component(const UpdateContext& update_context, const std::string& id)
185 : id_(id),
Jinho Bangda4e4282018-01-03 13:21:23186 state_(std::make_unique<StateNew>(this)),
sorin30474f02017-04-27 00:45:48187 update_context_(update_context) {}
188
189Component::~Component() {}
190
Sorin Jianub3296162017-12-13 20:26:32191scoped_refptr<Configurator> Component::config() const {
192 return update_context_.config;
193}
194
Sorin Jianud69d4372018-02-07 19:44:22195std::string Component::session_id() const {
196 return update_context_.session_id;
197}
198
Sorin Jianub41a592a2018-03-02 16:30:27199bool Component::is_foreground() const {
200 return update_context_.is_foreground;
201}
202
Sorin Jianuee5c0db2018-04-12 23:38:47203void Component::Handle(CallbackHandleComplete callback_handle_complete) {
sorin30474f02017-04-27 00:45:48204 DCHECK(thread_checker_.CalledOnValidThread());
205 DCHECK(state_);
206
Sorin Jianuee5c0db2018-04-12 23:38:47207 callback_handle_complete_ = std::move(callback_handle_complete);
sorin30474f02017-04-27 00:45:48208
Sorin Jianua8ef73d2017-11-02 16:55:17209 state_->Handle(
210 base::BindOnce(&Component::ChangeState, base::Unretained(this)));
sorin30474f02017-04-27 00:45:48211}
212
213void Component::ChangeState(std::unique_ptr<State> next_state) {
214 DCHECK(thread_checker_.CalledOnValidThread());
215
Sorin Jianu4ab7c292017-06-15 18:40:21216 previous_state_ = state();
sorin30474f02017-04-27 00:45:48217 if (next_state)
218 state_ = std::move(next_state);
Sorin Jianuee5c0db2018-04-12 23:38:47219 else
220 is_handled_ = true;
sorin30474f02017-04-27 00:45:48221
Sorin Jianua8ef73d2017-11-02 16:55:17222 base::ThreadTaskRunnerHandle::Get()->PostTask(
223 FROM_HERE, std::move(callback_handle_complete_));
sorin30474f02017-04-27 00:45:48224}
225
226CrxUpdateItem Component::GetCrxUpdateItem() const {
227 DCHECK(thread_checker_.CalledOnValidThread());
228
229 CrxUpdateItem crx_update_item;
230 crx_update_item.state = state_->state();
231 crx_update_item.id = id_;
Sorin Jianu7c22795b2018-04-26 22:16:52232 if (crx_component_)
233 crx_update_item.component = *crx_component_;
sorin30474f02017-04-27 00:45:48234 crx_update_item.last_check = last_check_;
235 crx_update_item.next_version = next_version_;
236 crx_update_item.next_fp = next_fp_;
Sorin Jianuafcb70dd2018-05-16 20:14:15237 crx_update_item.error_category = error_category_;
238 crx_update_item.error_code = error_code_;
239 crx_update_item.extra_code1 = extra_code1_;
sorin30474f02017-04-27 00:45:48240
241 return crx_update_item;
242}
243
sorin7cff6e52017-05-17 16:37:23244void Component::SetParseResult(const ProtocolParser::Result& result) {
sorin30474f02017-04-27 00:45:48245 DCHECK(thread_checker_.CalledOnValidThread());
246
247 DCHECK_EQ(0, update_check_error_);
248
249 status_ = result.status;
sorin519656c2017-04-28 22:39:34250 action_run_ = result.action_run;
sorin30474f02017-04-27 00:45:48251
252 if (result.manifest.packages.empty())
253 return;
254
255 next_version_ = base::Version(result.manifest.version);
256 const auto& package = result.manifest.packages.front();
257 next_fp_ = package.fingerprint;
258
259 // Resolve the urls by combining the base urls with the package names.
260 for (const auto& crx_url : result.crx_urls) {
261 const GURL url = crx_url.Resolve(package.name);
262 if (url.is_valid())
263 crx_urls_.push_back(url);
264 }
265 for (const auto& crx_diffurl : result.crx_diffurls) {
266 const GURL url = crx_diffurl.Resolve(package.namediff);
267 if (url.is_valid())
268 crx_diffurls_.push_back(url);
269 }
270
271 hash_sha256_ = package.hash_sha256;
272 hashdiff_sha256_ = package.hashdiff_sha256;
273}
274
275void Component::Uninstall(const base::Version& version, int reason) {
276 DCHECK(thread_checker_.CalledOnValidThread());
277
278 DCHECK_EQ(ComponentState::kNew, state());
279
Sorin Jianu73900242018-08-17 01:11:53280 crx_component_ = CrxComponent();
Sorin Jianu7c22795b2018-04-26 22:16:52281 crx_component_->version = version;
282
sorin30474f02017-04-27 00:45:48283 previous_version_ = version;
284 next_version_ = base::Version("0");
285 extra_code1_ = reason;
286
Jinho Bangda4e4282018-01-03 13:21:23287 state_ = std::make_unique<StateUninstalled>(this);
sorin30474f02017-04-27 00:45:48288}
289
Sorin Jianu888ec292018-06-01 15:35:42290void Component::SetUpdateCheckResult(
291 const base::Optional<ProtocolParser::Result>& result,
292 ErrorCategory error_category,
293 int error) {
sorin30474f02017-04-27 00:45:48294 DCHECK(thread_checker_.CalledOnValidThread());
sorin30474f02017-04-27 00:45:48295 DCHECK_EQ(ComponentState::kChecking, state());
296
Sorin Jianu888ec292018-06-01 15:35:42297 error_category_ = error_category;
298 error_code_ = error;
299 if (result)
300 SetParseResult(result.value());
301
Sorin Jianua8ef73d2017-11-02 16:55:17302 base::ThreadTaskRunnerHandle::Get()->PostTask(
303 FROM_HERE, std::move(update_check_complete_));
sorin30474f02017-04-27 00:45:48304}
305
306bool Component::CanDoBackgroundDownload() const {
Sorin Jianub41a592a2018-03-02 16:30:27307 // Foreground component updates are always downloaded in foreground.
Sorin Jianu7c22795b2018-04-26 22:16:52308 return !is_foreground() &&
309 (crx_component() && crx_component()->allows_background_download) &&
sorin30474f02017-04-27 00:45:48310 update_context_.config->EnabledBackgroundDownloader();
311}
312
Sorin Jianu039032b2018-10-12 21:48:13313void Component::AppendEvent(base::Value event) {
314 events_.push_back(std::move(event));
sorin30474f02017-04-27 00:45:48315}
316
317void Component::NotifyObservers(UpdateClient::Observer::Events event) const {
318 DCHECK(thread_checker_.CalledOnValidThread());
319 update_context_.notify_observers_callback.Run(event, id_);
320}
321
322base::TimeDelta Component::GetUpdateDuration() const {
323 DCHECK(thread_checker_.CalledOnValidThread());
324
325 if (update_begin_.is_null())
326 return base::TimeDelta();
327
328 const base::TimeDelta update_cost(base::TimeTicks::Now() - update_begin_);
329 DCHECK_GE(update_cost, base::TimeDelta());
330 const base::TimeDelta max_update_delay =
331 base::TimeDelta::FromSeconds(update_context_.config->UpdateDelay());
332 return std::min(update_cost, max_update_delay);
333}
334
Sorin Jianu039032b2018-10-12 21:48:13335base::Value Component::MakeEventUpdateComplete() const {
336 base::Value event(base::Value::Type::DICTIONARY);
337 event.SetKey("eventtype", base::Value(3));
338 event.SetKey(
339 "eventresult",
340 base::Value(static_cast<int>(state() == ComponentState::kUpdated)));
341 if (error_category() != ErrorCategory::kNone)
342 event.SetKey("errorcat", base::Value(static_cast<int>(error_category())));
343 if (error_code())
344 event.SetKey("errorcode", base::Value(error_code()));
345 if (extra_code1())
346 event.SetKey("extracode1", base::Value(extra_code1()));
347 if (HasDiffUpdate(*this)) {
348 const int diffresult = static_cast<int>(!diff_update_failed());
349 event.SetKey("diffresult", base::Value(diffresult));
350 }
351 if (diff_error_category() != ErrorCategory::kNone) {
352 const int differrorcat = static_cast<int>(diff_error_category());
353 event.SetKey("differrorcat", base::Value(differrorcat));
354 }
355 if (diff_error_code())
356 event.SetKey("differrorcode", base::Value(diff_error_code()));
357 if (diff_extra_code1())
358 event.SetKey("diffextracode1", base::Value(diff_extra_code1()));
359 if (!previous_fp().empty())
360 event.SetKey("previousfp", base::Value(previous_fp()));
361 if (!next_fp().empty())
362 event.SetKey("nextfp", base::Value(next_fp()));
363 DCHECK(previous_version().IsValid());
364 event.SetKey("previousversion", base::Value(previous_version().GetString()));
365 if (next_version().IsValid())
366 event.SetKey("nextversion", base::Value(next_version().GetString()));
367 return event;
368}
369
370base::Value Component::MakeEventDownloadMetrics(
371 const CrxDownloader::DownloadMetrics& dm) const {
372 base::Value event(base::Value::Type::DICTIONARY);
373 event.SetKey("eventtype", base::Value(14));
374 event.SetKey("eventresult", base::Value(static_cast<int>(dm.error == 0)));
375 event.SetKey("downloader", base::Value(DownloaderToString(dm.downloader)));
376 if (dm.error)
377 event.SetKey("errorcode", base::Value(dm.error));
378 event.SetKey("url", base::Value(dm.url.spec()));
379
380 // -1 means that the byte counts are not known.
Sorin Jianu55587d32018-11-14 21:43:27381 if (dm.total_bytes != -1 && dm.total_bytes < kProtocolMaxInt)
382 event.SetKey("total", base::Value(static_cast<double>(dm.total_bytes)));
383 if (dm.downloaded_bytes != -1 && dm.total_bytes < kProtocolMaxInt) {
Sorin Jianu039032b2018-10-12 21:48:13384 event.SetKey("downloaded",
Sorin Jianu55587d32018-11-14 21:43:27385 base::Value(static_cast<double>(dm.downloaded_bytes)));
Sorin Jianu039032b2018-10-12 21:48:13386 }
Sorin Jianu55587d32018-11-14 21:43:27387 if (dm.download_time_ms && dm.total_bytes < kProtocolMaxInt) {
Sorin Jianu039032b2018-10-12 21:48:13388 event.SetKey("download_time_ms",
Sorin Jianu55587d32018-11-14 21:43:27389 base::Value(static_cast<double>(dm.download_time_ms)));
Sorin Jianu039032b2018-10-12 21:48:13390 }
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::MakeEventUninstalled() const {
399 DCHECK(state() == ComponentState::kUninstalled);
400 base::Value event(base::Value::Type::DICTIONARY);
401 event.SetKey("eventtype", base::Value(4));
402 event.SetKey("eventresult", base::Value(1));
403 if (extra_code1())
404 event.SetKey("extracode1", base::Value(extra_code1()));
405 DCHECK(previous_version().IsValid());
406 event.SetKey("previousversion", base::Value(previous_version().GetString()));
407 DCHECK(next_version().IsValid());
408 event.SetKey("nextversion", base::Value(next_version().GetString()));
409 return event;
410}
411
412base::Value Component::MakeEventActionRun(bool succeeded,
413 int error_code,
414 int extra_code1) const {
415 base::Value event(base::Value::Type::DICTIONARY);
416 event.SetKey("eventtype", base::Value(42));
417 event.SetKey("eventresult", base::Value(static_cast<int>(succeeded)));
418 if (error_code)
419 event.SetKey("errorcode", base::Value(error_code));
420 if (extra_code1)
421 event.SetKey("extracode1", base::Value(extra_code1));
422 return event;
423}
424
425std::vector<base::Value> Component::GetEvents() const {
426 std::vector<base::Value> events;
427 for (const auto& event : events_)
428 events.push_back(event.Clone());
429 return events;
430}
431
sorin30474f02017-04-27 00:45:48432Component::State::State(Component* component, ComponentState state)
433 : state_(state), component_(*component) {}
434
435Component::State::~State() {}
436
Sorin Jianuee5c0db2018-04-12 23:38:47437void Component::State::Handle(CallbackNextState callback_next_state) {
sorin30474f02017-04-27 00:45:48438 DCHECK(thread_checker_.CalledOnValidThread());
439
Sorin Jianuee5c0db2018-04-12 23:38:47440 callback_next_state_ = std::move(callback_next_state);
sorin30474f02017-04-27 00:45:48441
sorin30474f02017-04-27 00:45:48442 DoHandle();
443}
444
445void Component::State::TransitionState(std::unique_ptr<State> next_state) {
Sorin Jianuee5c0db2018-04-12 23:38:47446 DCHECK(thread_checker_.CalledOnValidThread());
447 DCHECK(next_state);
sorin30474f02017-04-27 00:45:48448
449 base::ThreadTaskRunnerHandle::Get()->PostTask(
Sorin Jianuee5c0db2018-04-12 23:38:47450 FROM_HERE,
451 base::BindOnce(std::move(callback_next_state_), std::move(next_state)));
452}
453
454void Component::State::EndState() {
455 DCHECK(thread_checker_.CalledOnValidThread());
456
457 base::ThreadTaskRunnerHandle::Get()->PostTask(
458 FROM_HERE, base::BindOnce(std::move(callback_next_state_), nullptr));
sorin30474f02017-04-27 00:45:48459}
460
461Component::StateNew::StateNew(Component* component)
462 : State(component, ComponentState::kNew) {}
463
464Component::StateNew::~StateNew() {
465 DCHECK(thread_checker_.CalledOnValidThread());
466}
467
468void Component::StateNew::DoHandle() {
469 DCHECK(thread_checker_.CalledOnValidThread());
470
471 auto& component = State::component();
Sorin Jianucb4431a2018-04-30 20:59:24472 if (component.crx_component()) {
473 TransitionState(std::make_unique<StateChecking>(&component));
474 } else {
475 component.error_code_ = static_cast<int>(Error::CRX_NOT_FOUND);
Minh X. Nguyena4640cb2018-05-23 21:29:10476 component.error_category_ = ErrorCategory::kService;
Sorin Jianucb4431a2018-04-30 20:59:24477 TransitionState(std::make_unique<StateUpdateError>(&component));
478 }
sorin30474f02017-04-27 00:45:48479}
480
481Component::StateChecking::StateChecking(Component* component)
482 : State(component, ComponentState::kChecking) {}
483
484Component::StateChecking::~StateChecking() {
485 DCHECK(thread_checker_.CalledOnValidThread());
486}
487
488// Unlike how other states are handled, this function does not change the
489// state right away. The state transition happens when the UpdateChecker
490// calls Component::UpdateCheckComplete and |update_check_complete_| is invoked.
491// This is an artifact of how multiple components must be checked for updates
492// together but the state machine defines the transitions for one component
493// at a time.
494void Component::StateChecking::DoHandle() {
495 DCHECK(thread_checker_.CalledOnValidThread());
496
497 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52498 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48499
500 component.last_check_ = base::TimeTicks::Now();
Sorin Jianua8ef73d2017-11-02 16:55:17501 component.update_check_complete_ = base::BindOnce(
sorin30474f02017-04-27 00:45:48502 &Component::StateChecking::UpdateCheckComplete, base::Unretained(this));
503
504 component.NotifyObservers(Events::COMPONENT_CHECKING_FOR_UPDATES);
505}
506
507void Component::StateChecking::UpdateCheckComplete() {
508 DCHECK(thread_checker_.CalledOnValidThread());
509 auto& component = State::component();
Sorin Jianuafcb70dd2018-05-16 20:14:15510 if (!component.error_code_) {
sorin30474f02017-04-27 00:45:48511 if (component.status_ == "ok") {
Jinho Bangda4e4282018-01-03 13:21:23512 TransitionState(std::make_unique<StateCanUpdate>(&component));
sorin30474f02017-04-27 00:45:48513 return;
514 }
515
516 if (component.status_ == "noupdate") {
Sorin Jianu4ab7c292017-06-15 18:40:21517 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23518 TransitionState(std::make_unique<StateUpToDate>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21519 else
Jinho Bangda4e4282018-01-03 13:21:23520 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48521 return;
522 }
523 }
524
Jinho Bangda4e4282018-01-03 13:21:23525 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48526}
527
528Component::StateUpdateError::StateUpdateError(Component* component)
529 : State(component, ComponentState::kUpdateError) {}
530
531Component::StateUpdateError::~StateUpdateError() {
532 DCHECK(thread_checker_.CalledOnValidThread());
533}
534
535void Component::StateUpdateError::DoHandle() {
536 DCHECK(thread_checker_.CalledOnValidThread());
537
538 auto& component = State::component();
sorin117334f2017-05-19 02:36:25539
Sorin Jianu888ec292018-06-01 15:35:42540 DCHECK_NE(ErrorCategory::kNone, component.error_category_);
541 DCHECK_NE(0, component.error_code_);
542
sorin117334f2017-05-19 02:36:25543 // Create an event only when the server response included an update.
544 if (component.IsUpdateAvailable())
Sorin Jianu039032b2018-10-12 21:48:13545 component.AppendEvent(component.MakeEventUpdateComplete());
sorin117334f2017-05-19 02:36:25546
Sorin Jianuee5c0db2018-04-12 23:38:47547 EndState();
Sorin Jianucbb10e12018-01-23 18:01:44548 component.NotifyObservers(Events::COMPONENT_UPDATE_ERROR);
sorin30474f02017-04-27 00:45:48549}
550
551Component::StateCanUpdate::StateCanUpdate(Component* component)
552 : State(component, ComponentState::kCanUpdate) {}
553
554Component::StateCanUpdate::~StateCanUpdate() {
555 DCHECK(thread_checker_.CalledOnValidThread());
556}
557
558void Component::StateCanUpdate::DoHandle() {
559 DCHECK(thread_checker_.CalledOnValidThread());
560
561 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52562 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48563
564 component.is_update_available_ = true;
565 component.NotifyObservers(Events::COMPONENT_UPDATE_FOUND);
566
Sorin Jianu7c22795b2018-04-26 22:16:52567 if (component.crx_component()
568 ->supports_group_policy_enable_component_updates &&
sorin30474f02017-04-27 00:45:48569 !component.update_context_.enabled_component_updates) {
Minh X. Nguyena4640cb2018-05-23 21:29:10570 component.error_category_ = ErrorCategory::kService;
sorin30474f02017-04-27 00:45:48571 component.error_code_ = static_cast<int>(ServiceError::UPDATE_DISABLED);
572 component.extra_code1_ = 0;
Jinho Bangda4e4282018-01-03 13:21:23573 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48574 return;
575 }
576
577 // Start computing the cost of the this update from here on.
578 component.update_begin_ = base::TimeTicks::Now();
579
580 if (CanTryDiffUpdate())
Jinho Bangda4e4282018-01-03 13:21:23581 TransitionState(std::make_unique<StateDownloadingDiff>(&component));
sorin30474f02017-04-27 00:45:48582 else
Jinho Bangda4e4282018-01-03 13:21:23583 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48584}
585
586// Returns true if a differential update is available, it has not failed yet,
587// and the configuration allows this update.
588bool Component::StateCanUpdate::CanTryDiffUpdate() const {
589 const auto& component = Component::State::component();
590 return HasDiffUpdate(component) && !component.diff_error_code_ &&
591 component.update_context_.config->EnabledDeltas();
592}
593
594Component::StateUpToDate::StateUpToDate(Component* component)
595 : State(component, ComponentState::kUpToDate) {}
596
597Component::StateUpToDate::~StateUpToDate() {
598 DCHECK(thread_checker_.CalledOnValidThread());
599}
600
601void Component::StateUpToDate::DoHandle() {
602 DCHECK(thread_checker_.CalledOnValidThread());
603
604 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52605 DCHECK(component.crx_component());
sorin30474f02017-04-27 00:45:48606
sorin30474f02017-04-27 00:45:48607 component.NotifyObservers(Events::COMPONENT_NOT_UPDATED);
Sorin Jianuee5c0db2018-04-12 23:38:47608 EndState();
sorin30474f02017-04-27 00:45:48609}
610
611Component::StateDownloadingDiff::StateDownloadingDiff(Component* component)
612 : State(component, ComponentState::kDownloadingDiff) {}
613
614Component::StateDownloadingDiff::~StateDownloadingDiff() {
615 DCHECK(thread_checker_.CalledOnValidThread());
616}
617
618void Component::StateDownloadingDiff::DoHandle() {
619 DCHECK(thread_checker_.CalledOnValidThread());
620
621 const auto& component = Component::State::component();
622 const auto& update_context = component.update_context_;
623
Sorin Jianu7c22795b2018-04-26 22:16:52624 DCHECK(component.crx_component());
625
sorin30474f02017-04-27 00:45:48626 crx_downloader_ = update_context.crx_downloader_factory(
627 component.CanDoBackgroundDownload(),
Sorin Jianu75e6bf22019-02-12 16:07:12628 update_context.config->GetNetworkFetcherFactory());
sorin30474f02017-04-27 00:45:48629
630 const auto& id = component.id_;
631 crx_downloader_->set_progress_callback(
632 base::Bind(&Component::StateDownloadingDiff::DownloadProgress,
633 base::Unretained(this), id));
634 crx_downloader_->StartDownload(
635 component.crx_diffurls_, component.hashdiff_sha256_,
Sorin Jianua8ef73d2017-11-02 16:55:17636 base::BindOnce(&Component::StateDownloadingDiff::DownloadComplete,
637 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48638
639 component.NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
640}
641
Antonio Gomes31237fb2018-08-27 19:11:03642// Called when progress is being made downloading a CRX. Can be called multiple
643// times due to how the CRX downloader switches between different downloaders
644// and fallback urls.
645void Component::StateDownloadingDiff::DownloadProgress(const std::string& id) {
sorin30474f02017-04-27 00:45:48646 DCHECK(thread_checker_.CalledOnValidThread());
647
648 component().NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
649}
650
651void Component::StateDownloadingDiff::DownloadComplete(
652 const std::string& id,
653 const CrxDownloader::Result& download_result) {
654 DCHECK(thread_checker_.CalledOnValidThread());
655
656 auto& component = Component::State::component();
Sorin Jianu039032b2018-10-12 21:48:13657 for (const auto& download_metrics : crx_downloader_->download_metrics())
658 component.AppendEvent(component.MakeEventDownloadMetrics(download_metrics));
sorin30474f02017-04-27 00:45:48659
660 crx_downloader_.reset();
661
662 if (download_result.error) {
Sorin Jianu52de5fa2017-10-09 23:19:33663 DCHECK(download_result.response.empty());
Minh X. Nguyena4640cb2018-05-23 21:29:10664 component.diff_error_category_ = ErrorCategory::kDownload;
sorin30474f02017-04-27 00:45:48665 component.diff_error_code_ = download_result.error;
666
Jinho Bangda4e4282018-01-03 13:21:23667 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48668 return;
669 }
670
671 component.crx_path_ = download_result.response;
672
Jinho Bangda4e4282018-01-03 13:21:23673 TransitionState(std::make_unique<StateUpdatingDiff>(&component));
sorin30474f02017-04-27 00:45:48674}
675
676Component::StateDownloading::StateDownloading(Component* component)
677 : State(component, ComponentState::kDownloading) {}
678
679Component::StateDownloading::~StateDownloading() {
680 DCHECK(thread_checker_.CalledOnValidThread());
681}
682
683void Component::StateDownloading::DoHandle() {
684 DCHECK(thread_checker_.CalledOnValidThread());
685
686 const auto& component = Component::State::component();
687 const auto& update_context = component.update_context_;
688
Sorin Jianu7c22795b2018-04-26 22:16:52689 DCHECK(component.crx_component());
690
sorin30474f02017-04-27 00:45:48691 crx_downloader_ = update_context.crx_downloader_factory(
692 component.CanDoBackgroundDownload(),
Sorin Jianu75e6bf22019-02-12 16:07:12693 update_context.config->GetNetworkFetcherFactory());
sorin30474f02017-04-27 00:45:48694
695 const auto& id = component.id_;
696 crx_downloader_->set_progress_callback(
697 base::Bind(&Component::StateDownloading::DownloadProgress,
698 base::Unretained(this), id));
699 crx_downloader_->StartDownload(
700 component.crx_urls_, component.hash_sha256_,
Sorin Jianua8ef73d2017-11-02 16:55:17701 base::BindOnce(&Component::StateDownloading::DownloadComplete,
702 base::Unretained(this), id));
sorin30474f02017-04-27 00:45:48703
704 component.NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
705}
706
Antonio Gomes31237fb2018-08-27 19:11:03707// Called when progress is being made downloading a CRX. Can be called multiple
708// times due to how the CRX downloader switches between different downloaders
709// and fallback urls.
710void Component::StateDownloading::DownloadProgress(const std::string& id) {
sorin30474f02017-04-27 00:45:48711 DCHECK(thread_checker_.CalledOnValidThread());
712
713 component().NotifyObservers(Events::COMPONENT_UPDATE_DOWNLOADING);
714}
715
716void Component::StateDownloading::DownloadComplete(
717 const std::string& id,
718 const CrxDownloader::Result& download_result) {
719 DCHECK(thread_checker_.CalledOnValidThread());
720
721 auto& component = Component::State::component();
722
Sorin Jianu039032b2018-10-12 21:48:13723 for (const auto& download_metrics : crx_downloader_->download_metrics())
724 component.AppendEvent(component.MakeEventDownloadMetrics(download_metrics));
sorin30474f02017-04-27 00:45:48725
726 crx_downloader_.reset();
727
728 if (download_result.error) {
Sorin Jianu52de5fa2017-10-09 23:19:33729 DCHECK(download_result.response.empty());
Minh X. Nguyena4640cb2018-05-23 21:29:10730 component.error_category_ = ErrorCategory::kDownload;
sorin30474f02017-04-27 00:45:48731 component.error_code_ = download_result.error;
732
Jinho Bangda4e4282018-01-03 13:21:23733 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48734 return;
735 }
736
737 component.crx_path_ = download_result.response;
738
Jinho Bangda4e4282018-01-03 13:21:23739 TransitionState(std::make_unique<StateUpdating>(&component));
sorin30474f02017-04-27 00:45:48740}
741
742Component::StateUpdatingDiff::StateUpdatingDiff(Component* component)
743 : State(component, ComponentState::kUpdatingDiff) {}
744
745Component::StateUpdatingDiff::~StateUpdatingDiff() {
746 DCHECK(thread_checker_.CalledOnValidThread());
747}
748
749void Component::StateUpdatingDiff::DoHandle() {
750 DCHECK(thread_checker_.CalledOnValidThread());
751
752 const auto& component = Component::State::component();
753 const auto& update_context = component.update_context_;
754
Sorin Jianu7c22795b2018-04-26 22:16:52755 DCHECK(component.crx_component());
756
sorin30474f02017-04-27 00:45:48757 component.NotifyObservers(Events::COMPONENT_UPDATE_READY);
758
Sami Kyostila14ca7642019-08-01 17:52:25759 base::CreateSequencedTaskRunner(kTaskTraits)
Sorin Jianua8ef73d2017-11-02 16:55:17760 ->PostTask(
761 FROM_HERE,
762 base::BindOnce(
763 &update_client::StartInstallOnBlockingTaskRunner,
764 base::ThreadTaskRunnerHandle::Get(),
Sorin Jianu7c22795b2018-04-26 22:16:52765 component.crx_component()->pk_hash, component.crx_path_,
766 component.next_fp_, component.crx_component()->installer,
Joshua Pawlickid5409e12019-04-06 00:23:11767 update_context.config->GetUnzipperFactory()->Create(),
768 update_context.config->GetPatcherFactory()->Create(),
Joshua Pawlickif4b33f382018-08-17 17:36:51769 component.crx_component()->crx_format_requirement,
Sorin Jianua8ef73d2017-11-02 16:55:17770 base::BindOnce(&Component::StateUpdatingDiff::InstallComplete,
771 base::Unretained(this))));
sorin30474f02017-04-27 00:45:48772}
773
Minh X. Nguyena4640cb2018-05-23 21:29:10774void Component::StateUpdatingDiff::InstallComplete(ErrorCategory error_category,
sorin30474f02017-04-27 00:45:48775 int error_code,
776 int extra_code1) {
777 DCHECK(thread_checker_.CalledOnValidThread());
778
779 auto& component = Component::State::component();
780
781 component.diff_error_category_ = error_category;
782 component.diff_error_code_ = error_code;
783 component.diff_extra_code1_ = extra_code1;
784
785 if (component.diff_error_code_ != 0) {
Jinho Bangda4e4282018-01-03 13:21:23786 TransitionState(std::make_unique<StateDownloading>(&component));
sorin30474f02017-04-27 00:45:48787 return;
788 }
789
Minh X. Nguyena4640cb2018-05-23 21:29:10790 DCHECK_EQ(ErrorCategory::kNone, component.diff_error_category_);
sorin30474f02017-04-27 00:45:48791 DCHECK_EQ(0, component.diff_error_code_);
792 DCHECK_EQ(0, component.diff_extra_code1_);
793
Minh X. Nguyena4640cb2018-05-23 21:29:10794 DCHECK_EQ(ErrorCategory::kNone, component.error_category_);
sorin30474f02017-04-27 00:45:48795 DCHECK_EQ(0, component.error_code_);
796 DCHECK_EQ(0, component.extra_code1_);
797
Sorin Jianu4ab7c292017-06-15 18:40:21798 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23799 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21800 else
Jinho Bangda4e4282018-01-03 13:21:23801 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48802}
803
804Component::StateUpdating::StateUpdating(Component* component)
sorinff403ef2017-05-16 21:48:42805 : State(component, ComponentState::kUpdating) {}
sorin30474f02017-04-27 00:45:48806
807Component::StateUpdating::~StateUpdating() {
808 DCHECK(thread_checker_.CalledOnValidThread());
809}
810
811void Component::StateUpdating::DoHandle() {
812 DCHECK(thread_checker_.CalledOnValidThread());
813
814 const auto& component = Component::State::component();
815 const auto& update_context = component.update_context_;
816
Sorin Jianu7c22795b2018-04-26 22:16:52817 DCHECK(component.crx_component());
818
sorin30474f02017-04-27 00:45:48819 component.NotifyObservers(Events::COMPONENT_UPDATE_READY);
820
Sami Kyostila14ca7642019-08-01 17:52:25821 base::CreateSequencedTaskRunner(kTaskTraits)
Sorin Jianua8ef73d2017-11-02 16:55:17822 ->PostTask(FROM_HERE,
823 base::BindOnce(
824 &update_client::StartInstallOnBlockingTaskRunner,
825 base::ThreadTaskRunnerHandle::Get(),
Sorin Jianu7c22795b2018-04-26 22:16:52826 component.crx_component()->pk_hash, component.crx_path_,
827 component.next_fp_, component.crx_component()->installer,
Joshua Pawlickid5409e12019-04-06 00:23:11828 update_context.config->GetUnzipperFactory()->Create(),
829 update_context.config->GetPatcherFactory()->Create(),
Joshua Pawlickif4b33f382018-08-17 17:36:51830 component.crx_component()->crx_format_requirement,
Sorin Jianua8ef73d2017-11-02 16:55:17831 base::BindOnce(&Component::StateUpdating::InstallComplete,
Sorin Jianuebd652462017-07-23 02:00:58832 base::Unretained(this))));
sorin30474f02017-04-27 00:45:48833}
834
Minh X. Nguyena4640cb2018-05-23 21:29:10835void Component::StateUpdating::InstallComplete(ErrorCategory error_category,
sorin30474f02017-04-27 00:45:48836 int error_code,
837 int extra_code1) {
838 DCHECK(thread_checker_.CalledOnValidThread());
839
840 auto& component = Component::State::component();
841
842 component.error_category_ = error_category;
843 component.error_code_ = error_code;
844 component.extra_code1_ = extra_code1;
845
846 if (component.error_code_ != 0) {
Jinho Bangda4e4282018-01-03 13:21:23847 TransitionState(std::make_unique<StateUpdateError>(&component));
sorin30474f02017-04-27 00:45:48848 return;
849 }
850
Minh X. Nguyena4640cb2018-05-23 21:29:10851 DCHECK_EQ(ErrorCategory::kNone, component.error_category_);
sorin30474f02017-04-27 00:45:48852 DCHECK_EQ(0, component.error_code_);
853 DCHECK_EQ(0, component.extra_code1_);
854
Sorin Jianu4ab7c292017-06-15 18:40:21855 if (component.action_run_.empty())
Jinho Bangda4e4282018-01-03 13:21:23856 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21857 else
Jinho Bangda4e4282018-01-03 13:21:23858 TransitionState(std::make_unique<StateRun>(&component));
sorin30474f02017-04-27 00:45:48859}
860
861Component::StateUpdated::StateUpdated(Component* component)
862 : State(component, ComponentState::kUpdated) {
863 DCHECK(thread_checker_.CalledOnValidThread());
864}
865
866Component::StateUpdated::~StateUpdated() {
867 DCHECK(thread_checker_.CalledOnValidThread());
868}
869
870void Component::StateUpdated::DoHandle() {
871 DCHECK(thread_checker_.CalledOnValidThread());
872
873 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52874 DCHECK(component.crx_component());
875
876 component.crx_component_->version = component.next_version_;
877 component.crx_component_->fingerprint = component.next_fp_;
sorin30474f02017-04-27 00:45:48878
Sorin Jianu039032b2018-10-12 21:48:13879 component.AppendEvent(component.MakeEventUpdateComplete());
sorin117334f2017-05-19 02:36:25880
sorin30474f02017-04-27 00:45:48881 component.NotifyObservers(Events::COMPONENT_UPDATED);
Sorin Jianuee5c0db2018-04-12 23:38:47882 EndState();
sorin30474f02017-04-27 00:45:48883}
884
885Component::StateUninstalled::StateUninstalled(Component* component)
886 : State(component, ComponentState::kUninstalled) {
887 DCHECK(thread_checker_.CalledOnValidThread());
888}
889
890Component::StateUninstalled::~StateUninstalled() {
891 DCHECK(thread_checker_.CalledOnValidThread());
892}
893
894void Component::StateUninstalled::DoHandle() {
895 DCHECK(thread_checker_.CalledOnValidThread());
sorin117334f2017-05-19 02:36:25896
897 auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52898 DCHECK(component.crx_component());
899
Sorin Jianu039032b2018-10-12 21:48:13900 component.AppendEvent(component.MakeEventUninstalled());
sorin117334f2017-05-19 02:36:25901
Sorin Jianuee5c0db2018-04-12 23:38:47902 EndState();
sorin30474f02017-04-27 00:45:48903}
904
Sorin Jianu4ab7c292017-06-15 18:40:21905Component::StateRun::StateRun(Component* component)
906 : State(component, ComponentState::kRun) {}
907
908Component::StateRun::~StateRun() {
909 DCHECK(thread_checker_.CalledOnValidThread());
910}
911
912void Component::StateRun::DoHandle() {
913 DCHECK(thread_checker_.CalledOnValidThread());
914
915 const auto& component = State::component();
Sorin Jianu7c22795b2018-04-26 22:16:52916 DCHECK(component.crx_component());
917
Jinho Bangda4e4282018-01-03 13:21:23918 action_runner_ = std::make_unique<ActionRunner>(component);
Sorin Jianu4ab7c292017-06-15 18:40:21919 action_runner_->Run(
Sorin Jianua8ef73d2017-11-02 16:55:17920 base::BindOnce(&StateRun::ActionRunComplete, base::Unretained(this)));
Sorin Jianu4ab7c292017-06-15 18:40:21921}
922
923void Component::StateRun::ActionRunComplete(bool succeeded,
924 int error_code,
925 int extra_code1) {
926 DCHECK(thread_checker_.CalledOnValidThread());
927
928 auto& component = State::component();
929
930 component.AppendEvent(
Sorin Jianu039032b2018-10-12 21:48:13931 component.MakeEventActionRun(succeeded, error_code, extra_code1));
Sorin Jianu4ab7c292017-06-15 18:40:21932 switch (component.previous_state_) {
933 case ComponentState::kChecking:
Jinho Bangda4e4282018-01-03 13:21:23934 TransitionState(std::make_unique<StateUpToDate>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21935 return;
936 case ComponentState::kUpdating:
937 case ComponentState::kUpdatingDiff:
Jinho Bangda4e4282018-01-03 13:21:23938 TransitionState(std::make_unique<StateUpdated>(&component));
Sorin Jianu4ab7c292017-06-15 18:40:21939 return;
940 default:
941 break;
942 }
943 NOTREACHED();
944}
945
sorin30474f02017-04-27 00:45:48946} // namespace update_client