blob: 9eb162be654b6b9e69a93d0e93b001a9513a7897 [file] [log] [blame]
[email protected]de0fdca22014-08-19 05:26:091// Copyright 2014 The Chromium Authors. All rights reserved.
[email protected]e3e696d32013-06-21 20:41:362// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
sorin52ac0882015-01-24 01:15:005#include "components/update_client/component_patcher_operation.h"
[email protected]e3e696d32013-06-21 20:41:366
sorin5cb1f5492014-09-23 04:07:447#include <stdint.h>
Sorin Jianua8ef73d2017-11-02 16:55:178#include <utility>
[email protected]e3e696d32013-06-21 20:41:369#include <vector>
10
[email protected]94a481b2014-03-28 19:41:5511#include "base/bind.h"
thestig819adcc82014-09-10 22:24:5312#include "base/files/file_util.h"
[email protected]e3e696d32013-06-21 20:41:3613#include "base/files/memory_mapped_file.h"
[email protected]e260af72014-08-05 07:52:3914#include "base/location.h"
[email protected]e3e696d32013-06-21 20:41:3615#include "base/strings/string_number_conversions.h"
Sorin Jianuebd652462017-07-23 02:00:5816#include "base/task_scheduler/post_task.h"
17#include "base/threading/sequenced_task_runner_handle.h"
sorincca1c122017-05-11 17:43:2218#include "components/update_client/out_of_process_patcher.h"
sorin52ac0882015-01-24 01:15:0019#include "components/update_client/update_client.h"
sorin7b8650522016-11-02 18:23:4120#include "components/update_client/update_client_errors.h"
sorin74e70672016-02-03 03:13:1021#include "components/update_client/utils.h"
[email protected]94a481b2014-03-28 19:41:5522#include "courgette/courgette.h"
altimin979ea2e12016-05-18 16:16:2423#include "courgette/third_party/bsdiff/bsdiff.h"
[email protected]e3e696d32013-06-21 20:41:3624
sorin52ac0882015-01-24 01:15:0025namespace update_client {
[email protected]055981f2014-01-17 20:22:3226
[email protected]e3e696d32013-06-21 20:41:3627namespace {
28
[email protected]e3e696d32013-06-21 20:41:3629const char kOutput[] = "output";
[email protected]e3e696d32013-06-21 20:41:3630const char kSha256[] = "sha256";
31
[email protected]94a481b2014-03-28 19:41:5532// The integer offset disambiguates between overlapping error ranges.
33const int kCourgetteErrorOffset = 300;
34const int kBsdiffErrorOffset = 600;
35
[email protected]e3e696d32013-06-21 20:41:3636} // namespace
37
[email protected]e260af72014-08-05 07:52:3938const char kOp[] = "op";
39const char kBsdiff[] = "bsdiff";
40const char kCourgette[] = "courgette";
41const char kInput[] = "input";
42const char kPatch[] = "patch";
[email protected]94a481b2014-03-28 19:41:5543
[email protected]e260af72014-08-05 07:52:3944DeltaUpdateOp* CreateDeltaUpdateOp(
45 const std::string& operation,
bauerb810e60f42015-02-05 01:09:1046 const scoped_refptr<OutOfProcessPatcher>& out_of_process_patcher) {
[email protected]94a481b2014-03-28 19:41:5547 if (operation == "copy") {
[email protected]e3e696d32013-06-21 20:41:3648 return new DeltaUpdateOpCopy();
[email protected]94a481b2014-03-28 19:41:5549 } else if (operation == "create") {
[email protected]e3e696d32013-06-21 20:41:3650 return new DeltaUpdateOpCreate();
[email protected]e260af72014-08-05 07:52:3951 } else if (operation == "bsdiff" || operation == "courgette") {
52 return new DeltaUpdateOpPatch(operation, out_of_process_patcher);
[email protected]94a481b2014-03-28 19:41:5553 }
Ivan Kotenkov75b1c3a2017-10-24 14:47:2454 return nullptr;
[email protected]e3e696d32013-06-21 20:41:3655}
56
[email protected]e260af72014-08-05 07:52:3957DeltaUpdateOp::DeltaUpdateOp() {
[email protected]d0c8b8b42014-05-06 05:11:4558}
[email protected]e3e696d32013-06-21 20:41:3659
[email protected]d0c8b8b42014-05-06 05:11:4560DeltaUpdateOp::~DeltaUpdateOp() {
61}
[email protected]e3e696d32013-06-21 20:41:3662
Sorin Jianuebd652462017-07-23 02:00:5863void DeltaUpdateOp::Run(const base::DictionaryValue* command_args,
64 const base::FilePath& input_dir,
65 const base::FilePath& unpack_dir,
66 const scoped_refptr<CrxInstaller>& installer,
Sorin Jianua8ef73d2017-11-02 16:55:1767 ComponentPatcher::Callback callback) {
68 callback_ = std::move(callback);
[email protected]e3e696d32013-06-21 20:41:3669 std::string output_rel_path;
70 if (!command_args->GetString(kOutput, &output_rel_path) ||
[email protected]94a481b2014-03-28 19:41:5571 !command_args->GetString(kSha256, &output_sha256_)) {
sorin7b8650522016-11-02 18:23:4172 DoneRunning(UnpackerError::kDeltaBadCommands, 0);
[email protected]94a481b2014-03-28 19:41:5573 return;
74 }
[email protected]e3e696d32013-06-21 20:41:3675
[email protected]d0c8b8b42014-05-06 05:11:4576 output_abs_path_ =
77 unpack_dir.Append(base::FilePath::FromUTF8Unsafe(output_rel_path));
sorin7b8650522016-11-02 18:23:4178 UnpackerError parse_result =
[email protected]d0c8b8b42014-05-06 05:11:4579 DoParseArguments(command_args, input_dir, installer);
sorin7b8650522016-11-02 18:23:4180 if (parse_result != UnpackerError::kNone) {
[email protected]94a481b2014-03-28 19:41:5581 DoneRunning(parse_result, 0);
82 return;
83 }
[email protected]e3e696d32013-06-21 20:41:3684
85 const base::FilePath parent = output_abs_path_.DirName();
[email protected]dcd16612013-07-15 20:18:0986 if (!base::DirectoryExists(parent)) {
[email protected]94a481b2014-03-28 19:41:5587 if (!base::CreateDirectory(parent)) {
sorin7b8650522016-11-02 18:23:4188 DoneRunning(UnpackerError::kIoError, 0);
[email protected]94a481b2014-03-28 19:41:5589 return;
90 }
[email protected]e3e696d32013-06-21 20:41:3691 }
92
Sorin Jianua8ef73d2017-11-02 16:55:1793 DoRun(base::BindOnce(&DeltaUpdateOp::DoneRunning,
94 scoped_refptr<DeltaUpdateOp>(this)));
[email protected]94a481b2014-03-28 19:41:5595}
[email protected]e3e696d32013-06-21 20:41:3696
sorin7b8650522016-11-02 18:23:4197void DeltaUpdateOp::DoneRunning(UnpackerError error, int extended_error) {
98 if (error == UnpackerError::kNone)
[email protected]94a481b2014-03-28 19:41:5599 error = CheckHash();
Sorin Jianuebd652462017-07-23 02:00:58100 base::SequencedTaskRunnerHandle::Get()->PostTask(
Sorin Jianua8ef73d2017-11-02 16:55:17101 FROM_HERE, base::BindOnce(std::move(callback_), error, extended_error));
[email protected]e3e696d32013-06-21 20:41:36102}
103
104// Uses the hash as a checksum to confirm that the file now residing in the
105// output directory probably has the contents it should.
sorin7b8650522016-11-02 18:23:41106UnpackerError DeltaUpdateOp::CheckHash() {
sorin74e70672016-02-03 03:13:10107 return VerifyFileHash256(output_abs_path_, output_sha256_)
sorin7b8650522016-11-02 18:23:41108 ? UnpackerError::kNone
109 : UnpackerError::kDeltaVerificationFailure;
[email protected]e3e696d32013-06-21 20:41:36110}
111
[email protected]d0c8b8b42014-05-06 05:11:45112DeltaUpdateOpCopy::DeltaUpdateOpCopy() {
113}
[email protected]e3e696d32013-06-21 20:41:36114
[email protected]d0c8b8b42014-05-06 05:11:45115DeltaUpdateOpCopy::~DeltaUpdateOpCopy() {
116}
[email protected]94a481b2014-03-28 19:41:55117
sorin7b8650522016-11-02 18:23:41118UnpackerError DeltaUpdateOpCopy::DoParseArguments(
[email protected]94a481b2014-03-28 19:41:55119 const base::DictionaryValue* command_args,
[email protected]e3e696d32013-06-21 20:41:36120 const base::FilePath& input_dir,
sorin9797aba2015-04-17 17:15:03121 const scoped_refptr<CrxInstaller>& installer) {
[email protected]e3e696d32013-06-21 20:41:36122 std::string input_rel_path;
123 if (!command_args->GetString(kInput, &input_rel_path))
sorin7b8650522016-11-02 18:23:41124 return UnpackerError::kDeltaBadCommands;
[email protected]e3e696d32013-06-21 20:41:36125
126 if (!installer->GetInstalledFile(input_rel_path, &input_abs_path_))
sorin7b8650522016-11-02 18:23:41127 return UnpackerError::kDeltaMissingExistingFile;
[email protected]e3e696d32013-06-21 20:41:36128
sorin7b8650522016-11-02 18:23:41129 return UnpackerError::kNone;
[email protected]e3e696d32013-06-21 20:41:36130}
131
Sorin Jianua8ef73d2017-11-02 16:55:17132void DeltaUpdateOpCopy::DoRun(ComponentPatcher::Callback callback) {
[email protected]f0ff2ad2013-07-09 17:42:26133 if (!base::CopyFile(input_abs_path_, output_abs_path_))
Sorin Jianua8ef73d2017-11-02 16:55:17134 std::move(callback).Run(UnpackerError::kDeltaOperationFailure, 0);
[email protected]94a481b2014-03-28 19:41:55135 else
Sorin Jianua8ef73d2017-11-02 16:55:17136 std::move(callback).Run(UnpackerError::kNone, 0);
[email protected]e3e696d32013-06-21 20:41:36137}
138
[email protected]d0c8b8b42014-05-06 05:11:45139DeltaUpdateOpCreate::DeltaUpdateOpCreate() {
140}
[email protected]e3e696d32013-06-21 20:41:36141
[email protected]d0c8b8b42014-05-06 05:11:45142DeltaUpdateOpCreate::~DeltaUpdateOpCreate() {
143}
[email protected]94a481b2014-03-28 19:41:55144
sorin7b8650522016-11-02 18:23:41145UnpackerError DeltaUpdateOpCreate::DoParseArguments(
[email protected]94a481b2014-03-28 19:41:55146 const base::DictionaryValue* command_args,
[email protected]e3e696d32013-06-21 20:41:36147 const base::FilePath& input_dir,
sorin9797aba2015-04-17 17:15:03148 const scoped_refptr<CrxInstaller>& installer) {
[email protected]e3e696d32013-06-21 20:41:36149 std::string patch_rel_path;
150 if (!command_args->GetString(kPatch, &patch_rel_path))
sorin7b8650522016-11-02 18:23:41151 return UnpackerError::kDeltaBadCommands;
[email protected]e3e696d32013-06-21 20:41:36152
[email protected]d0c8b8b42014-05-06 05:11:45153 patch_abs_path_ =
154 input_dir.Append(base::FilePath::FromUTF8Unsafe(patch_rel_path));
[email protected]e3e696d32013-06-21 20:41:36155
sorin7b8650522016-11-02 18:23:41156 return UnpackerError::kNone;
[email protected]e3e696d32013-06-21 20:41:36157}
158
Sorin Jianua8ef73d2017-11-02 16:55:17159void DeltaUpdateOpCreate::DoRun(ComponentPatcher::Callback callback) {
[email protected]5553d5b2013-07-01 23:07:36160 if (!base::Move(patch_abs_path_, output_abs_path_))
Sorin Jianua8ef73d2017-11-02 16:55:17161 std::move(callback).Run(UnpackerError::kDeltaOperationFailure, 0);
[email protected]94a481b2014-03-28 19:41:55162 else
Sorin Jianua8ef73d2017-11-02 16:55:17163 std::move(callback).Run(UnpackerError::kNone, 0);
[email protected]e3e696d32013-06-21 20:41:36164}
165
[email protected]94a481b2014-03-28 19:41:55166DeltaUpdateOpPatch::DeltaUpdateOpPatch(
[email protected]e260af72014-08-05 07:52:39167 const std::string& operation,
sorincca1c122017-05-11 17:43:22168 const scoped_refptr<OutOfProcessPatcher>& out_of_process_patcher)
[email protected]e260af72014-08-05 07:52:39169 : operation_(operation), out_of_process_patcher_(out_of_process_patcher) {
170 DCHECK(operation == kBsdiff || operation == kCourgette);
[email protected]94a481b2014-03-28 19:41:55171}
172
173DeltaUpdateOpPatch::~DeltaUpdateOpPatch() {
174}
175
sorin7b8650522016-11-02 18:23:41176UnpackerError DeltaUpdateOpPatch::DoParseArguments(
[email protected]94a481b2014-03-28 19:41:55177 const base::DictionaryValue* command_args,
[email protected]e3e696d32013-06-21 20:41:36178 const base::FilePath& input_dir,
sorin9797aba2015-04-17 17:15:03179 const scoped_refptr<CrxInstaller>& installer) {
[email protected]e3e696d32013-06-21 20:41:36180 std::string patch_rel_path;
181 std::string input_rel_path;
182 if (!command_args->GetString(kPatch, &patch_rel_path) ||
183 !command_args->GetString(kInput, &input_rel_path))
sorin7b8650522016-11-02 18:23:41184 return UnpackerError::kDeltaBadCommands;
[email protected]e3e696d32013-06-21 20:41:36185
186 if (!installer->GetInstalledFile(input_rel_path, &input_abs_path_))
sorin7b8650522016-11-02 18:23:41187 return UnpackerError::kDeltaMissingExistingFile;
[email protected]e3e696d32013-06-21 20:41:36188
[email protected]d0c8b8b42014-05-06 05:11:45189 patch_abs_path_ =
190 input_dir.Append(base::FilePath::FromUTF8Unsafe(patch_rel_path));
[email protected]e3e696d32013-06-21 20:41:36191
sorin7b8650522016-11-02 18:23:41192 return UnpackerError::kNone;
[email protected]e3e696d32013-06-21 20:41:36193}
194
Sorin Jianua8ef73d2017-11-02 16:55:17195void DeltaUpdateOpPatch::DoRun(ComponentPatcher::Callback callback) {
[email protected]e260af72014-08-05 07:52:39196 if (out_of_process_patcher_.get()) {
197 out_of_process_patcher_->Patch(
Sorin Jianuebd652462017-07-23 02:00:58198 operation_, input_abs_path_, patch_abs_path_, output_abs_path_,
Sorin Jianua8ef73d2017-11-02 16:55:17199 base::BindOnce(&DeltaUpdateOpPatch::DonePatching, this,
200 std::move(callback)));
[email protected]94a481b2014-03-28 19:41:55201 return;
202 }
[email protected]e260af72014-08-05 07:52:39203
204 if (operation_ == kBsdiff) {
Sorin Jianua8ef73d2017-11-02 16:55:17205 DonePatching(std::move(callback),
huangs7054b5a22016-07-26 21:46:13206 bsdiff::ApplyBinaryPatch(input_abs_path_, patch_abs_path_,
207 output_abs_path_));
[email protected]e260af72014-08-05 07:52:39208 } else if (operation_ == kCourgette) {
Sorin Jianua8ef73d2017-11-02 16:55:17209 DonePatching(std::move(callback), courgette::ApplyEnsemblePatch(
210 input_abs_path_.value().c_str(),
211 patch_abs_path_.value().c_str(),
212 output_abs_path_.value().c_str()));
[email protected]e260af72014-08-05 07:52:39213 } else {
214 NOTREACHED();
215 }
[email protected]e3e696d32013-06-21 20:41:36216}
217
Sorin Jianua8ef73d2017-11-02 16:55:17218void DeltaUpdateOpPatch::DonePatching(ComponentPatcher::Callback callback,
219 int result) {
[email protected]e260af72014-08-05 07:52:39220 if (operation_ == kBsdiff) {
huangs7054b5a22016-07-26 21:46:13221 if (result == bsdiff::OK) {
Sorin Jianua8ef73d2017-11-02 16:55:17222 std::move(callback).Run(UnpackerError::kNone, 0);
[email protected]e260af72014-08-05 07:52:39223 } else {
Sorin Jianua8ef73d2017-11-02 16:55:17224 std::move(callback).Run(UnpackerError::kDeltaOperationFailure,
225 result + kBsdiffErrorOffset);
[email protected]e260af72014-08-05 07:52:39226 }
227 } else if (operation_ == kCourgette) {
228 if (result == courgette::C_OK) {
Sorin Jianua8ef73d2017-11-02 16:55:17229 std::move(callback).Run(UnpackerError::kNone, 0);
[email protected]e260af72014-08-05 07:52:39230 } else {
Sorin Jianua8ef73d2017-11-02 16:55:17231 std::move(callback).Run(UnpackerError::kDeltaOperationFailure,
232 result + kCourgetteErrorOffset);
[email protected]e260af72014-08-05 07:52:39233 }
234 } else {
235 NOTREACHED();
[email protected]94a481b2014-03-28 19:41:55236 }
[email protected]e3e696d32013-06-21 20:41:36237}
238
sorin52ac0882015-01-24 01:15:00239} // namespace update_client