blob: 4698256cbc142fdbc0c8988613aaa7fd386a93bd [file] [log] [blame]
Mike Frysinger8155d082012-04-06 15:23:18 -04001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Andrew de los Reyes09e56d62010-04-23 13:45:53 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Alex Deymo8427b4a2014-11-05 14:00:32 -08005#include "update_engine/delta_performer.h"
6
Andrew de los Reyes09e56d62010-04-23 13:45:53 -07007#include <inttypes.h>
Alex Deymo161c4a12014-05-16 15:56:21 -07008#include <sys/mount.h>
Andrew de los Reyes09e56d62010-04-23 13:45:53 -07009
10#include <algorithm>
11#include <string>
12#include <vector>
13
Allie Wood78750a42015-02-11 15:42:11 -080014#include <base/files/file_path.h>
Ben Chan06c76a42014-09-05 08:21:06 -070015#include <base/files/file_util.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070016#include <base/strings/stringprintf.h>
Alex Deymof1cbe172015-03-05 15:58:37 -080017#include <base/strings/string_util.h>
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070018#include <google/protobuf/repeated_field.h>
19#include <gtest/gtest.h>
20
Jay Srinivasand29695d2013-04-08 15:08:05 -070021#include "update_engine/constants.h"
Andrew de los Reyes353777c2010-10-08 10:34:30 -070022#include "update_engine/extent_ranges.h"
David Zeuthene7f89172013-10-31 10:21:04 -070023#include "update_engine/fake_hardware.h"
Gilad Arnold5bb4c902014-04-10 12:32:13 -070024#include "update_engine/fake_system_state.h"
Alex Deymo8427b4a2014-11-05 14:00:32 -080025#include "update_engine/mock_prefs.h"
Alex Deymo161c4a12014-05-16 15:56:21 -070026#include "update_engine/payload_constants.h"
27#include "update_engine/payload_generator/delta_diff_generator.h"
Alex Deymo923d8fa2014-07-15 17:58:51 -070028#include "update_engine/payload_generator/payload_signer.h"
29#include "update_engine/payload_verifier.h"
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070030#include "update_engine/test_utils.h"
31#include "update_engine/update_metadata.pb.h"
32#include "update_engine/utils.h"
33
34namespace chromeos_update_engine {
35
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070036using std::string;
37using std::vector;
Darin Petkov73058b42010-10-06 16:32:19 -070038using testing::Return;
Alex Deymo161c4a12014-05-16 15:56:21 -070039using testing::_;
Alex Deymo10875d92014-11-10 21:52:57 -080040using test_utils::kRandomString;
41using test_utils::ScopedLoopMounter;
42using test_utils::System;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070043
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -070044extern const char* kUnittestPrivateKeyPath;
Darin Petkovd7061ab2010-10-06 14:37:09 -070045extern const char* kUnittestPublicKeyPath;
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -070046extern const char* kUnittestPrivateKey2Path;
47extern const char* kUnittestPublicKey2Path;
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -070048
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -070049static const char* kBogusMetadataSignature1 =
50 "awSFIUdUZz2VWFiR+ku0Pj00V7bPQPQFYQSXjEXr3vaw3TE4xHV5CraY3/YrZpBv"
51 "J5z4dSBskoeuaO1TNC/S6E05t+yt36tE4Fh79tMnJ/z9fogBDXWgXLEUyG78IEQr"
52 "YH6/eBsQGT2RJtBgXIXbZ9W+5G9KmGDoPOoiaeNsDuqHiBc/58OFsrxskH8E6vMS"
53 "BmMGGk82mvgzic7ApcoURbCGey1b3Mwne/hPZ/bb9CIyky8Og9IfFMdL2uAweOIR"
54 "fjoTeLYZpt+WN65Vu7jJ0cQN8e1y+2yka5112wpRf/LLtPgiAjEZnsoYpLUd7CoV"
55 "pLRtClp97kN2+tXGNBQqkA==";
Jay Srinivasan738fdf32012-12-07 17:40:54 -080056
Alex Vakulenkod2779df2014-06-16 13:19:00 -070057static const int kDefaultKernelSize = 4096; // Something small for a test
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080058static const uint8_t kNewData[] = {'T', 'h', 'i', 's', ' ', 'i', 's', ' ',
59 'n', 'e', 'w', ' ', 'd', 'a', 't', 'a', '.'};
Jay Srinivasan738fdf32012-12-07 17:40:54 -080060
Andrew de los Reyes27f7d372010-10-07 11:26:07 -070061namespace {
Jay Srinivasan738fdf32012-12-07 17:40:54 -080062struct DeltaState {
63 string a_img;
64 string b_img;
65 int image_size;
Andrew de los Reyes27f7d372010-10-07 11:26:07 -070066
Jay Srinivasan738fdf32012-12-07 17:40:54 -080067 string delta_path;
68 uint64_t metadata_size;
Andrew de los Reyes27f7d372010-10-07 11:26:07 -070069
Jay Srinivasan738fdf32012-12-07 17:40:54 -080070 string old_kernel;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080071 chromeos::Blob old_kernel_data;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070072
Jay Srinivasan738fdf32012-12-07 17:40:54 -080073 string new_kernel;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080074 chromeos::Blob new_kernel_data;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070075
Jay Srinivasan738fdf32012-12-07 17:40:54 -080076 // The in-memory copy of delta file.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080077 chromeos::Blob delta;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -080078
79 // The mock system state object with which we initialize the
80 // delta performer.
Gilad Arnold5bb4c902014-04-10 12:32:13 -070081 FakeSystemState fake_system_state;
Jay Srinivasan738fdf32012-12-07 17:40:54 -080082};
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070083
Jay Srinivasan738fdf32012-12-07 17:40:54 -080084enum SignatureTest {
85 kSignatureNone, // No payload signing.
86 kSignatureGenerator, // Sign the payload at generation time.
87 kSignatureGenerated, // Sign the payload after it's generated.
Don Garrett2ae37872013-10-25 13:33:20 -070088 kSignatureGeneratedPlaceholder, // Insert placeholder signatures, then real.
Alex Vakulenkod2779df2014-06-16 13:19:00 -070089 kSignatureGeneratedPlaceholderMismatch, // Insert a wrong sized placeholder.
Jay Srinivasan738fdf32012-12-07 17:40:54 -080090 kSignatureGeneratedShell, // Sign the generated payload through shell cmds.
91 kSignatureGeneratedShellBadKey, // Sign with a bad key through shell cmds.
92 kSignatureGeneratedShellRotateCl1, // Rotate key, test client v1
93 kSignatureGeneratedShellRotateCl2, // Rotate key, test client v2
94};
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070095
Jay Srinivasan738fdf32012-12-07 17:40:54 -080096// Different options that determine what we should fill into the
97// install_plan.metadata_signature to simulate the contents received in the
98// Omaha response.
99enum MetadataSignatureTest {
100 kEmptyMetadataSignature,
101 kInvalidMetadataSignature,
102 kValidMetadataSignature,
103};
104
105enum OperationHashTest {
106 kInvalidOperationData,
107 kValidOperationData,
108};
109
Alex Deymof1cbe172015-03-05 15:58:37 -0800110// Chuck size used for full payloads during test.
111size_t kDefaultFullChunkSize = 1024 * 1024;
112
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700113} // namespace
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800114
115static void CompareFilesByBlock(const string& a_file, const string& b_file) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800116 chromeos::Blob a_data, b_data;
Andrew de los Reyes3270f742010-07-15 22:28:14 -0700117 EXPECT_TRUE(utils::ReadFile(a_file, &a_data)) << "file failed: " << a_file;
118 EXPECT_TRUE(utils::ReadFile(b_file, &b_data)) << "file failed: " << b_file;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700119
120 EXPECT_EQ(a_data.size(), b_data.size());
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700121 EXPECT_EQ(0, a_data.size() % kBlockSize);
122 for (size_t i = 0; i < a_data.size(); i += kBlockSize) {
123 EXPECT_EQ(0, i % kBlockSize);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800124 chromeos::Blob a_sub(&a_data[i], &a_data[i + kBlockSize]);
125 chromeos::Blob b_sub(&b_data[i], &b_data[i + kBlockSize]);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700126 EXPECT_TRUE(a_sub == b_sub) << "Block " << (i/kBlockSize) << " differs";
127 }
128}
129
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800130static bool WriteSparseFile(const string& path, off_t size) {
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700131 int fd = open(path.c_str(), O_CREAT | O_TRUNC | O_WRONLY, 0644);
132 TEST_AND_RETURN_FALSE_ERRNO(fd >= 0);
133 ScopedFdCloser fd_closer(&fd);
134 off_t rc = lseek(fd, size + 1, SEEK_SET);
135 TEST_AND_RETURN_FALSE_ERRNO(rc != static_cast<off_t>(-1));
136 int return_code = ftruncate(fd, size);
137 TEST_AND_RETURN_FALSE_ERRNO(return_code == 0);
138 return true;
139}
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700140
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800141static size_t GetSignatureSize(const string& private_key_path) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800142 const chromeos::Blob data(1, 'x');
143 chromeos::Blob hash;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800144 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(data, &hash));
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800145 chromeos::Blob signature;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800146 EXPECT_TRUE(PayloadSigner::SignHash(hash,
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800147 private_key_path,
Darin Petkovcbfb0172011-01-14 15:24:45 -0800148 &signature));
149 return signature.size();
150}
151
Don Garrett2ae37872013-10-25 13:33:20 -0700152static bool InsertSignaturePlaceholder(int signature_size,
153 const string& payload_path,
154 uint64_t* out_metadata_size) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800155 vector<chromeos::Blob> signatures;
156 signatures.push_back(chromeos::Blob(signature_size, 0));
Don Garrett2ae37872013-10-25 13:33:20 -0700157
158 return PayloadSigner::AddSignatureToPayload(
159 payload_path,
160 signatures,
161 payload_path,
162 out_metadata_size);
163}
164
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800165static void SignGeneratedPayload(const string& payload_path,
166 uint64_t* out_metadata_size) {
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800167 int signature_size = GetSignatureSize(kUnittestPrivateKeyPath);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800168 chromeos::Blob hash;
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700169 ASSERT_TRUE(PayloadSigner::HashPayloadForSigning(
170 payload_path,
171 vector<int>(1, signature_size),
172 &hash));
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800173 chromeos::Blob signature;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800174 ASSERT_TRUE(PayloadSigner::SignHash(hash,
175 kUnittestPrivateKeyPath,
176 &signature));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700177 ASSERT_TRUE(PayloadSigner::AddSignatureToPayload(
178 payload_path,
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800179 vector<chromeos::Blob>(1, signature),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800180 payload_path,
181 out_metadata_size));
Alex Deymo923d8fa2014-07-15 17:58:51 -0700182 EXPECT_TRUE(PayloadVerifier::VerifySignedPayload(
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700183 payload_path,
184 kUnittestPublicKeyPath,
185 kSignatureMessageOriginalVersion));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800186}
187
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800188static void SignGeneratedShellPayload(SignatureTest signature_test,
189 const string& payload_path) {
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800190 string private_key_path = kUnittestPrivateKeyPath;
191 if (signature_test == kSignatureGeneratedShellBadKey) {
Gilad Arnolda6742b32014-01-11 00:18:34 -0800192 ASSERT_TRUE(utils::MakeTempFile("key.XXXXXX",
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800193 &private_key_path,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700194 nullptr));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800195 } else {
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700196 ASSERT_TRUE(signature_test == kSignatureGeneratedShell ||
197 signature_test == kSignatureGeneratedShellRotateCl1 ||
198 signature_test == kSignatureGeneratedShellRotateCl2);
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800199 }
200 ScopedPathUnlinker key_unlinker(private_key_path);
201 key_unlinker.set_should_remove(signature_test ==
202 kSignatureGeneratedShellBadKey);
203 // Generates a new private key that will not match the public key.
204 if (signature_test == kSignatureGeneratedShellBadKey) {
205 LOG(INFO) << "Generating a mismatched private key.";
Alex Deymo719bfff2014-07-11 12:12:32 -0700206 ASSERT_EQ(0, System(base::StringPrintf(
207 "%s genrsa -out %s 2048",
208 utils::GetPathOnBoard("openssl").c_str(), private_key_path.c_str())));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800209 }
210 int signature_size = GetSignatureSize(private_key_path);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800211 string hash_file;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700212 ASSERT_TRUE(utils::MakeTempFile("hash.XXXXXX", &hash_file, nullptr));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800213 ScopedPathUnlinker hash_unlinker(hash_file);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700214 string signature_size_string;
215 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
216 signature_test == kSignatureGeneratedShellRotateCl2)
Alex Vakulenko75039d72014-03-25 12:36:28 -0700217 signature_size_string = base::StringPrintf("%d:%d",
218 signature_size, signature_size);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700219 else
Alex Vakulenko75039d72014-03-25 12:36:28 -0700220 signature_size_string = base::StringPrintf("%d", signature_size);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800221 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700222 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700223 "./delta_generator -in_file=%s -signature_size=%s "
224 "-out_hash_file=%s",
Darin Petkovcbfb0172011-01-14 15:24:45 -0800225 payload_path.c_str(),
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700226 signature_size_string.c_str(),
Darin Petkovcbfb0172011-01-14 15:24:45 -0800227 hash_file.c_str())));
228
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700229 // Pad the hash
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800230 chromeos::Blob hash;
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700231 ASSERT_TRUE(utils::ReadFile(hash_file, &hash));
Alex Deymo923d8fa2014-07-15 17:58:51 -0700232 ASSERT_TRUE(PayloadVerifier::PadRSA2048SHA256Hash(&hash));
Alex Deymo10875d92014-11-10 21:52:57 -0800233 ASSERT_TRUE(test_utils::WriteFileVector(hash_file, hash));
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700234
Darin Petkovcbfb0172011-01-14 15:24:45 -0800235 string sig_file;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700236 ASSERT_TRUE(utils::MakeTempFile("signature.XXXXXX", &sig_file, nullptr));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800237 ScopedPathUnlinker sig_unlinker(sig_file);
238 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700239 System(base::StringPrintf(
Alex Deymo719bfff2014-07-11 12:12:32 -0700240 "%s rsautl -raw -sign -inkey %s -in %s -out %s",
241 utils::GetPathOnBoard("openssl").c_str(),
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800242 private_key_path.c_str(),
Darin Petkovcbfb0172011-01-14 15:24:45 -0800243 hash_file.c_str(),
244 sig_file.c_str())));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700245 string sig_file2;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700246 ASSERT_TRUE(utils::MakeTempFile("signature.XXXXXX", &sig_file2, nullptr));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700247 ScopedPathUnlinker sig2_unlinker(sig_file2);
248 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
249 signature_test == kSignatureGeneratedShellRotateCl2) {
250 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700251 System(base::StringPrintf(
Alex Deymo719bfff2014-07-11 12:12:32 -0700252 "%s rsautl -raw -sign -inkey %s -in %s -out %s",
253 utils::GetPathOnBoard("openssl").c_str(),
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700254 kUnittestPrivateKey2Path,
255 hash_file.c_str(),
256 sig_file2.c_str())));
257 // Append second sig file to first path
258 sig_file += ":" + sig_file2;
259 }
260
Darin Petkovcbfb0172011-01-14 15:24:45 -0800261 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700262 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700263 "./delta_generator -in_file=%s -signature_file=%s "
264 "-out_file=%s",
Darin Petkovcbfb0172011-01-14 15:24:45 -0800265 payload_path.c_str(),
266 sig_file.c_str(),
267 payload_path.c_str())));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800268 int verify_result =
Alex Vakulenko75039d72014-03-25 12:36:28 -0700269 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700270 "./delta_generator -in_file=%s -public_key=%s -public_key_version=%d",
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700271 payload_path.c_str(),
272 signature_test == kSignatureGeneratedShellRotateCl2 ?
273 kUnittestPublicKey2Path : kUnittestPublicKeyPath,
274 signature_test == kSignatureGeneratedShellRotateCl2 ? 2 : 1));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800275 if (signature_test == kSignatureGeneratedShellBadKey) {
276 ASSERT_NE(0, verify_result);
277 } else {
278 ASSERT_EQ(0, verify_result);
279 }
Darin Petkovcbfb0172011-01-14 15:24:45 -0800280}
281
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800282static void GenerateDeltaFile(bool full_kernel,
283 bool full_rootfs,
284 bool noop,
Darin Petkov8e447e02013-04-16 16:23:50 +0200285 off_t chunk_size,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800286 SignatureTest signature_test,
287 DeltaState *state) {
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700288 EXPECT_TRUE(utils::MakeTempFile("a_img.XXXXXX", &state->a_img, nullptr));
289 EXPECT_TRUE(utils::MakeTempFile("b_img.XXXXXX", &state->b_img, nullptr));
Alex Deymo10875d92014-11-10 21:52:57 -0800290 test_utils::CreateExtImageAtPath(state->a_img, nullptr);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700291
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800292 state->image_size = static_cast<int>(utils::FileSize(state->a_img));
Darin Petkov7ea32332010-10-13 10:46:11 -0700293
294 // Extend the "partitions" holding the file system a bit.
295 EXPECT_EQ(0, System(base::StringPrintf(
Alex Deymo1f93d032015-03-10 18:58:32 -0700296 "dd if=/dev/zero of=%s seek=%d bs=1 count=1 status=none",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800297 state->a_img.c_str(),
298 state->image_size + 1024 * 1024 - 1)));
299 EXPECT_EQ(state->image_size + 1024 * 1024, utils::FileSize(state->a_img));
Darin Petkov7ea32332010-10-13 10:46:11 -0700300
Don Garrett0dd39852013-04-03 16:55:42 -0700301 // Create ImageInfo A & B
302 ImageInfo old_image_info;
303 ImageInfo new_image_info;
304
305 if (!full_rootfs) {
306 old_image_info.set_channel("src-channel");
307 old_image_info.set_board("src-board");
308 old_image_info.set_version("src-version");
309 old_image_info.set_key("src-key");
310 old_image_info.set_build_channel("src-build-channel");
311 old_image_info.set_build_version("src-build-version");
312 }
313
314 new_image_info.set_channel("test-channel");
315 new_image_info.set_board("test-board");
316 new_image_info.set_version("test-version");
317 new_image_info.set_key("test-key");
318 new_image_info.set_build_channel("test-build-channel");
319 new_image_info.set_build_version("test-build-version");
320
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700321 // Make some changes to the A image.
322 {
323 string a_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800324 ScopedLoopMounter b_mounter(state->a_img, &a_mnt, 0);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700325
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800326 chromeos::Blob hardtocompress;
Darin Petkov8e447e02013-04-16 16:23:50 +0200327 while (hardtocompress.size() < 3 * kBlockSize) {
328 hardtocompress.insert(hardtocompress.end(),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800329 std::begin(kRandomString), std::end(kRandomString));
Darin Petkov8e447e02013-04-16 16:23:50 +0200330 }
Alex Vakulenko75039d72014-03-25 12:36:28 -0700331 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/hardtocompress",
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800332 a_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700333 hardtocompress.data(),
Darin Petkov8e447e02013-04-16 16:23:50 +0200334 hardtocompress.size()));
335
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800336 chromeos::Blob zeros(16 * 1024, 0);
Darin Petkov8a075a72013-04-25 14:46:09 +0200337 EXPECT_EQ(zeros.size(),
Ben Chan736fcb52014-05-21 18:28:22 -0700338 base::WriteFile(base::FilePath(base::StringPrintf(
339 "%s/move-to-sparse", a_mnt.c_str())),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800340 reinterpret_cast<const char*>(zeros.data()),
341 zeros.size()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200342
343 EXPECT_TRUE(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700344 WriteSparseFile(base::StringPrintf("%s/move-from-sparse",
345 a_mnt.c_str()), 16 * 1024));
Darin Petkov8a075a72013-04-25 14:46:09 +0200346
Alex Vakulenko75039d72014-03-25 12:36:28 -0700347 EXPECT_EQ(0,
Alex Deymo10875d92014-11-10 21:52:57 -0800348 System(base::StringPrintf("dd if=/dev/zero of=%s/move-semi-sparse"
Alex Deymo1f93d032015-03-10 18:58:32 -0700349 " bs=1 seek=4096 count=1 status=none",
Alex Deymo10875d92014-11-10 21:52:57 -0800350 a_mnt.c_str()).c_str()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200351
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700352 // Write 1 MiB of 0xff to try to catch the case where writing a bsdiff
353 // patch fails to zero out the final block.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800354 chromeos::Blob ones(1024 * 1024, 0xff);
Alex Vakulenko75039d72014-03-25 12:36:28 -0700355 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/ones",
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800356 a_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700357 ones.data(),
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700358 ones.size()));
359 }
360
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700361 if (noop) {
Alex Vakulenko75039d72014-03-25 12:36:28 -0700362 EXPECT_TRUE(base::CopyFile(base::FilePath(state->a_img),
363 base::FilePath(state->b_img)));
Don Garrett0dd39852013-04-03 16:55:42 -0700364 old_image_info = new_image_info;
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700365 } else {
Alex Deymo10875d92014-11-10 21:52:57 -0800366 test_utils::CreateExtImageAtPath(state->b_img, nullptr);
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700367 EXPECT_EQ(0, System(base::StringPrintf(
Alex Deymo1f93d032015-03-10 18:58:32 -0700368 "dd if=/dev/zero of=%s seek=%d bs=1 count=1 status=none",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800369 state->b_img.c_str(),
370 state->image_size + 1024 * 1024 - 1)));
371 EXPECT_EQ(state->image_size + 1024 * 1024, utils::FileSize(state->b_img));
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700372
373 // Make some changes to the B image.
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700374 string b_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800375 ScopedLoopMounter b_mounter(state->b_img, &b_mnt, 0);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700376
Alex Deymo10875d92014-11-10 21:52:57 -0800377 EXPECT_EQ(0, System(base::StringPrintf("cp %s/hello %s/hello2",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700378 b_mnt.c_str(),
379 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800380 EXPECT_EQ(0, System(base::StringPrintf("rm %s/hello",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700381 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800382 EXPECT_EQ(0, System(base::StringPrintf("mv %s/hello2 %s/hello",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700383 b_mnt.c_str(),
384 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800385 EXPECT_EQ(0, System(base::StringPrintf("echo foo > %s/foo",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700386 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800387 EXPECT_EQ(0, System(base::StringPrintf("touch %s/emptyfile",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700388 b_mnt.c_str()).c_str()));
389 EXPECT_TRUE(WriteSparseFile(base::StringPrintf("%s/fullsparse",
390 b_mnt.c_str()),
391 1024 * 1024));
Darin Petkov8a075a72013-04-25 14:46:09 +0200392
393 EXPECT_TRUE(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700394 WriteSparseFile(base::StringPrintf("%s/move-to-sparse", b_mnt.c_str()),
Darin Petkov8a075a72013-04-25 14:46:09 +0200395 16 * 1024));
396
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800397 chromeos::Blob zeros(16 * 1024, 0);
Darin Petkov8a075a72013-04-25 14:46:09 +0200398 EXPECT_EQ(zeros.size(),
Ben Chan736fcb52014-05-21 18:28:22 -0700399 base::WriteFile(base::FilePath(base::StringPrintf(
400 "%s/move-from-sparse", b_mnt.c_str())),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800401 reinterpret_cast<const char*>(zeros.data()),
402 zeros.size()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200403
Alex Deymo10875d92014-11-10 21:52:57 -0800404 EXPECT_EQ(0, System(base::StringPrintf("dd if=/dev/zero "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700405 "of=%s/move-semi-sparse "
Alex Deymo1f93d032015-03-10 18:58:32 -0700406 "bs=1 seek=4096 count=1 status=none",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700407 b_mnt.c_str()).c_str()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200408
Alex Deymo10875d92014-11-10 21:52:57 -0800409 EXPECT_EQ(0, System(base::StringPrintf("dd if=/dev/zero "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700410 "of=%s/partsparse bs=1 "
Alex Deymo1f93d032015-03-10 18:58:32 -0700411 "seek=4096 count=1 status=none",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700412 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800413 EXPECT_EQ(0, System(base::StringPrintf("cp %s/srchardlink0 %s/tmp && "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700414 "mv %s/tmp %s/srchardlink1",
415 b_mnt.c_str(),
416 b_mnt.c_str(),
417 b_mnt.c_str(),
418 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800419 EXPECT_EQ(0, System(
420 base::StringPrintf("rm %s/boguslink && echo foobar > %s/boguslink",
421 b_mnt.c_str(), b_mnt.c_str()).c_str()));
Darin Petkov8e447e02013-04-16 16:23:50 +0200422
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800423 chromeos::Blob hardtocompress;
Darin Petkov8e447e02013-04-16 16:23:50 +0200424 while (hardtocompress.size() < 3 * kBlockSize) {
425 hardtocompress.insert(hardtocompress.end(),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800426 std::begin(kRandomString), std::end(kRandomString));
Darin Petkov8e447e02013-04-16 16:23:50 +0200427 }
Alex Vakulenko75039d72014-03-25 12:36:28 -0700428 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/hardtocompress",
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700429 b_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700430 hardtocompress.data(),
Darin Petkov8e447e02013-04-16 16:23:50 +0200431 hardtocompress.size()));
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700432 }
433
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700434 string old_kernel;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800435 EXPECT_TRUE(utils::MakeTempFile("old_kernel.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800436 &state->old_kernel,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700437 nullptr));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700438
439 string new_kernel;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800440 EXPECT_TRUE(utils::MakeTempFile("new_kernel.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800441 &state->new_kernel,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700442 nullptr));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700443
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800444 state->old_kernel_data.resize(kDefaultKernelSize);
445 state->new_kernel_data.resize(state->old_kernel_data.size());
Alex Deymo10875d92014-11-10 21:52:57 -0800446 test_utils::FillWithData(&state->old_kernel_data);
447 test_utils::FillWithData(&state->new_kernel_data);
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700448
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700449 // change the new kernel data
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800450 std::copy(std::begin(kNewData), std::end(kNewData),
451 state->new_kernel_data.begin());
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700452
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700453 if (noop) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800454 state->old_kernel_data = state->new_kernel_data;
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700455 }
456
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700457 // Write kernels to disk
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800458 EXPECT_TRUE(utils::WriteFile(state->old_kernel.c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700459 state->old_kernel_data.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800460 state->old_kernel_data.size()));
461 EXPECT_TRUE(utils::WriteFile(state->new_kernel.c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700462 state->new_kernel_data.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800463 state->new_kernel_data.size()));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700464
Gilad Arnolda6742b32014-01-11 00:18:34 -0800465 EXPECT_TRUE(utils::MakeTempFile("delta.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800466 &state->delta_path,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700467 nullptr));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800468 LOG(INFO) << "delta path: " << state->delta_path;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700469 {
470 string a_mnt, b_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800471 ScopedLoopMounter a_mounter(state->a_img, &a_mnt, MS_RDONLY);
472 ScopedLoopMounter b_mounter(state->b_img, &b_mnt, MS_RDONLY);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800473 const string private_key =
474 signature_test == kSignatureGenerator ? kUnittestPrivateKeyPath : "";
Alex Deymof1cbe172015-03-05 15:58:37 -0800475
476 PayloadGenerationConfig payload_config;
477 payload_config.is_delta = !full_rootfs;
478 payload_config.chunk_size = chunk_size;
479 if (!full_rootfs) {
480 payload_config.source.rootfs_part = state->a_img;
481 payload_config.source.rootfs_mountpt = a_mnt;
482 if (!full_kernel)
483 payload_config.source.kernel_part = state->old_kernel;
484 payload_config.source.image_info = old_image_info;
485 EXPECT_TRUE(payload_config.source.LoadImageSize());
486
487 payload_config.minor_version =
488 DeltaPerformer::kSupportedMinorPayloadVersion;
489 } else {
490 payload_config.minor_version = DeltaPerformer::kFullPayloadMinorVersion;
491 if (payload_config.chunk_size == -1)
492 payload_config.chunk_size = kDefaultFullChunkSize;
493 }
494 payload_config.target.rootfs_part = state->b_img;
495 payload_config.target.rootfs_mountpt = b_mnt;
496 payload_config.target.kernel_part = state->new_kernel;
497 payload_config.target.image_info = new_image_info;
498 EXPECT_TRUE(payload_config.target.LoadImageSize());
499
500 EXPECT_TRUE(payload_config.Validate());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700501 EXPECT_TRUE(
Darin Petkov68c10d12010-10-14 09:24:37 -0700502 DeltaDiffGenerator::GenerateDeltaUpdateFile(
Alex Deymof1cbe172015-03-05 15:58:37 -0800503 payload_config,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800504 state->delta_path,
505 private_key,
Chris Sosad5ae1562013-04-23 13:20:18 -0700506 kRootFSPartitionSize,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800507 &state->metadata_size));
Darin Petkov9574f7e2011-01-13 10:48:12 -0800508 }
509
Don Garrett2ae37872013-10-25 13:33:20 -0700510 if (signature_test == kSignatureGeneratedPlaceholder ||
511 signature_test == kSignatureGeneratedPlaceholderMismatch) {
Don Garrett2ae37872013-10-25 13:33:20 -0700512 int signature_size = GetSignatureSize(kUnittestPrivateKeyPath);
513 LOG(INFO) << "Inserting placeholder signature.";
514 ASSERT_TRUE(InsertSignaturePlaceholder(signature_size, state->delta_path,
515 &state->metadata_size));
516
517 if (signature_test == kSignatureGeneratedPlaceholderMismatch) {
518 signature_size -= 1;
519 LOG(INFO) << "Inserting mismatched placeholder signature.";
520 ASSERT_FALSE(InsertSignaturePlaceholder(signature_size, state->delta_path,
521 &state->metadata_size));
522 return;
523 }
524 }
525
526 if (signature_test == kSignatureGenerated ||
527 signature_test == kSignatureGeneratedPlaceholder ||
528 signature_test == kSignatureGeneratedPlaceholderMismatch) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800529 // Generate the signed payload and update the metadata size in state to
530 // reflect the new size after adding the signature operation to the
531 // manifest.
Don Garrett2ae37872013-10-25 13:33:20 -0700532 LOG(INFO) << "Signing payload.";
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800533 SignGeneratedPayload(state->delta_path, &state->metadata_size);
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800534 } else if (signature_test == kSignatureGeneratedShell ||
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700535 signature_test == kSignatureGeneratedShellBadKey ||
536 signature_test == kSignatureGeneratedShellRotateCl1 ||
537 signature_test == kSignatureGeneratedShellRotateCl2) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800538 SignGeneratedShellPayload(signature_test, state->delta_path);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700539 }
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800540}
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700541
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800542static void ApplyDeltaFile(bool full_kernel, bool full_rootfs, bool noop,
543 SignatureTest signature_test, DeltaState* state,
544 bool hash_checks_mandatory,
545 OperationHashTest op_hash_test,
546 DeltaPerformer** performer) {
Darin Petkov36a58222010-10-07 22:00:09 -0700547 // Check the metadata.
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700548 {
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700549 DeltaArchiveManifest manifest;
Alex Deymo923d8fa2014-07-15 17:58:51 -0700550 EXPECT_TRUE(PayloadVerifier::LoadPayload(state->delta_path,
551 &state->delta,
552 &manifest,
553 &state->metadata_size));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800554 LOG(INFO) << "Metadata size: " << state->metadata_size;
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700555
Don Garrett0dd39852013-04-03 16:55:42 -0700556
557
Darin Petkovcbfb0172011-01-14 15:24:45 -0800558 if (signature_test == kSignatureNone) {
559 EXPECT_FALSE(manifest.has_signatures_offset());
560 EXPECT_FALSE(manifest.has_signatures_size());
561 } else {
562 EXPECT_TRUE(manifest.has_signatures_offset());
563 EXPECT_TRUE(manifest.has_signatures_size());
564 Signatures sigs_message;
565 EXPECT_TRUE(sigs_message.ParseFromArray(
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800566 &state->delta[state->metadata_size + manifest.signatures_offset()],
Darin Petkovcbfb0172011-01-14 15:24:45 -0800567 manifest.signatures_size()));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700568 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
569 signature_test == kSignatureGeneratedShellRotateCl2)
570 EXPECT_EQ(2, sigs_message.signatures_size());
571 else
572 EXPECT_EQ(1, sigs_message.signatures_size());
Darin Petkovcbfb0172011-01-14 15:24:45 -0800573 const Signatures_Signature& signature = sigs_message.signatures(0);
574 EXPECT_EQ(1, signature.version());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700575
Darin Petkovcbfb0172011-01-14 15:24:45 -0800576 uint64_t expected_sig_data_length = 0;
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700577 vector<string> key_paths{kUnittestPrivateKeyPath};
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700578 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
579 signature_test == kSignatureGeneratedShellRotateCl2) {
580 key_paths.push_back(kUnittestPrivateKey2Path);
581 }
Darin Petkovcbfb0172011-01-14 15:24:45 -0800582 EXPECT_TRUE(PayloadSigner::SignatureBlobLength(
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700583 key_paths,
584 &expected_sig_data_length));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800585 EXPECT_EQ(expected_sig_data_length, manifest.signatures_size());
586 EXPECT_FALSE(signature.data().empty());
587 }
Darin Petkov36a58222010-10-07 22:00:09 -0700588
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700589 if (noop) {
Gilad Arnold063181c2015-02-05 12:24:49 -0800590 EXPECT_EQ(0, manifest.install_operations_size());
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700591 EXPECT_EQ(1, manifest.kernel_install_operations_size());
592 }
593
Darin Petkovd43d6902010-10-14 11:17:50 -0700594 if (full_kernel) {
595 EXPECT_FALSE(manifest.has_old_kernel_info());
596 } else {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800597 EXPECT_EQ(state->old_kernel_data.size(),
598 manifest.old_kernel_info().size());
Darin Petkovd43d6902010-10-14 11:17:50 -0700599 EXPECT_FALSE(manifest.old_kernel_info().hash().empty());
600 }
Darin Petkov698d0412010-10-13 10:59:44 -0700601
Don Garrett0dd39852013-04-03 16:55:42 -0700602 EXPECT_EQ(manifest.new_image_info().channel(), "test-channel");
603 EXPECT_EQ(manifest.new_image_info().board(), "test-board");
604 EXPECT_EQ(manifest.new_image_info().version(), "test-version");
605 EXPECT_EQ(manifest.new_image_info().key(), "test-key");
606 EXPECT_EQ(manifest.new_image_info().build_channel(), "test-build-channel");
607 EXPECT_EQ(manifest.new_image_info().build_version(), "test-build-version");
608
609 if (!full_rootfs) {
Don Garrett0dd39852013-04-03 16:55:42 -0700610 if (noop) {
611 EXPECT_EQ(manifest.old_image_info().channel(), "test-channel");
612 EXPECT_EQ(manifest.old_image_info().board(), "test-board");
613 EXPECT_EQ(manifest.old_image_info().version(), "test-version");
614 EXPECT_EQ(manifest.old_image_info().key(), "test-key");
615 EXPECT_EQ(manifest.old_image_info().build_channel(),
616 "test-build-channel");
617 EXPECT_EQ(manifest.old_image_info().build_version(),
618 "test-build-version");
619 } else {
620 EXPECT_EQ(manifest.old_image_info().channel(), "src-channel");
621 EXPECT_EQ(manifest.old_image_info().board(), "src-board");
622 EXPECT_EQ(manifest.old_image_info().version(), "src-version");
623 EXPECT_EQ(manifest.old_image_info().key(), "src-key");
624 EXPECT_EQ(manifest.old_image_info().build_channel(),
625 "src-build-channel");
626 EXPECT_EQ(manifest.old_image_info().build_version(),
627 "src-build-version");
628 }
629 }
630
631
Darin Petkov7a22d792010-11-08 14:10:00 -0800632 if (full_rootfs) {
633 EXPECT_FALSE(manifest.has_old_rootfs_info());
Don Garrett0dd39852013-04-03 16:55:42 -0700634 EXPECT_FALSE(manifest.has_old_image_info());
635 EXPECT_TRUE(manifest.has_new_image_info());
Darin Petkov7a22d792010-11-08 14:10:00 -0800636 } else {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800637 EXPECT_EQ(state->image_size, manifest.old_rootfs_info().size());
Darin Petkov7a22d792010-11-08 14:10:00 -0800638 EXPECT_FALSE(manifest.old_rootfs_info().hash().empty());
639 }
640
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800641 EXPECT_EQ(state->new_kernel_data.size(), manifest.new_kernel_info().size());
642 EXPECT_EQ(state->image_size, manifest.new_rootfs_info().size());
Darin Petkov36a58222010-10-07 22:00:09 -0700643
Darin Petkov36a58222010-10-07 22:00:09 -0700644 EXPECT_FALSE(manifest.new_kernel_info().hash().empty());
Darin Petkov36a58222010-10-07 22:00:09 -0700645 EXPECT_FALSE(manifest.new_rootfs_info().hash().empty());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700646 }
647
Alex Deymo8427b4a2014-11-05 14:00:32 -0800648 MockPrefs prefs;
Darin Petkov73058b42010-10-06 16:32:19 -0700649 EXPECT_CALL(prefs, SetInt64(kPrefsManifestMetadataSize,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800650 state->metadata_size)).WillOnce(Return(true));
Darin Petkov73058b42010-10-06 16:32:19 -0700651 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextOperation, _))
652 .WillRepeatedly(Return(true));
Darin Petkov9b230572010-10-08 10:20:09 -0700653 EXPECT_CALL(prefs, GetInt64(kPrefsUpdateStateNextOperation, _))
654 .WillOnce(Return(false));
Darin Petkov73058b42010-10-06 16:32:19 -0700655 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextDataOffset, _))
656 .WillRepeatedly(Return(true));
David Zeuthen41996ad2013-09-24 15:43:24 -0700657 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextDataLength, _))
658 .WillRepeatedly(Return(true));
Darin Petkov437adc42010-10-07 13:12:24 -0700659 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSHA256Context, _))
Darin Petkov73058b42010-10-06 16:32:19 -0700660 .WillRepeatedly(Return(true));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800661 if (op_hash_test == kValidOperationData && signature_test != kSignatureNone) {
Darin Petkovcbfb0172011-01-14 15:24:45 -0800662 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSignedSHA256Context, _))
663 .WillOnce(Return(true));
Darin Petkov4f0a07b2011-05-25 16:47:20 -0700664 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSignatureBlob, _))
665 .WillOnce(Return(true));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800666 }
Darin Petkov73058b42010-10-06 16:32:19 -0700667
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700668 // Update the A image in place.
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700669 InstallPlan install_plan;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800670 install_plan.hash_checks_mandatory = hash_checks_mandatory;
671 install_plan.metadata_size = state->metadata_size;
Don Garrettb8dd1d92013-11-22 17:40:02 -0800672 install_plan.is_full_update = full_kernel && full_rootfs;
673
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800674 LOG(INFO) << "Setting payload metadata size in Omaha = "
675 << state->metadata_size;
Jay Srinivasanf4318702012-09-24 11:56:24 -0700676 ASSERT_TRUE(PayloadSigner::GetMetadataSignature(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700677 state->delta.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800678 state->metadata_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700679 kUnittestPrivateKeyPath,
Jay Srinivasanf4318702012-09-24 11:56:24 -0700680 &install_plan.metadata_signature));
681 EXPECT_FALSE(install_plan.metadata_signature.empty());
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700682
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800683 *performer = new DeltaPerformer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700684 &state->fake_system_state,
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800685 &install_plan);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700686 EXPECT_TRUE(utils::FileExists(kUnittestPublicKeyPath));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800687 (*performer)->set_public_key_path(kUnittestPublicKeyPath);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700688
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800689 EXPECT_EQ(state->image_size,
690 OmahaHashCalculator::RawHashOfFile(state->a_img,
691 state->image_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700692 &install_plan.rootfs_hash));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800693 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(state->old_kernel_data,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700694 &install_plan.kernel_hash));
695
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800696 EXPECT_EQ(0, (*performer)->Open(state->a_img.c_str(), 0, 0));
697 EXPECT_TRUE((*performer)->OpenKernel(state->old_kernel.c_str()));
698
David Zeuthena99981f2013-04-29 13:42:47 -0700699 ErrorCode expected_error, actual_error;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800700 bool continue_writing;
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700701 switch (op_hash_test) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800702 case kInvalidOperationData: {
703 // Muck with some random offset post the metadata size so that
704 // some operation hash will result in a mismatch.
705 int some_offset = state->metadata_size + 300;
706 LOG(INFO) << "Tampered value at offset: " << some_offset;
707 state->delta[some_offset]++;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700708 expected_error = ErrorCode::kDownloadOperationHashMismatch;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800709 continue_writing = false;
710 break;
711 }
712
713 case kValidOperationData:
714 default:
715 // no change.
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700716 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800717 continue_writing = true;
718 break;
719 }
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700720
721 // Write at some number of bytes per operation. Arbitrarily chose 5.
722 const size_t kBytesPerWrite = 5;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800723 for (size_t i = 0; i < state->delta.size(); i += kBytesPerWrite) {
Alex Deymof329b932014-10-30 01:37:48 -0700724 size_t count = std::min(state->delta.size() - i, kBytesPerWrite);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800725 bool write_succeeded = ((*performer)->Write(&state->delta[i],
726 count,
727 &actual_error));
728 // Normally write_succeeded should be true every time and
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700729 // actual_error should be ErrorCode::kSuccess. If so, continue the loop.
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800730 // But if we seeded an operation hash error above, then write_succeeded
731 // will be false. The failure may happen at any operation n. So, all
732 // Writes until n-1 should succeed and the nth operation will fail with
733 // actual_error. In this case, we should bail out of the loop because
734 // we cannot proceed applying the delta.
735 if (!write_succeeded) {
736 LOG(INFO) << "Write failed. Checking if it failed with expected error";
737 EXPECT_EQ(expected_error, actual_error);
738 if (!continue_writing) {
739 LOG(INFO) << "Cannot continue writing. Bailing out.";
740 break;
741 }
742 }
743
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700744 EXPECT_EQ(ErrorCode::kSuccess, actual_error);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700745 }
746
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800747 // If we had continued all the way through, Close should succeed.
748 // Otherwise, it should fail. Check appropriately.
749 bool close_result = (*performer)->Close();
750 if (continue_writing)
751 EXPECT_EQ(0, close_result);
752 else
753 EXPECT_LE(0, close_result);
754}
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700755
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800756void VerifyPayloadResult(DeltaPerformer* performer,
757 DeltaState* state,
David Zeuthena99981f2013-04-29 13:42:47 -0700758 ErrorCode expected_result) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800759 if (!performer) {
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700760 EXPECT_TRUE(!"Skipping payload verification since performer is null.");
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800761 return;
762 }
763
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700764 int expected_times = (expected_result == ErrorCode::kSuccess) ? 1 : 0;
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700765 EXPECT_CALL(*(state->fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800766 DownloadComplete()).Times(expected_times);
767
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800768 LOG(INFO) << "Verifying payload for expected result "
769 << expected_result;
770 EXPECT_EQ(expected_result, performer->VerifyPayload(
771 OmahaHashCalculator::OmahaHashOfData(state->delta),
772 state->delta.size()));
773 LOG(INFO) << "Verified payload.";
774
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700775 if (expected_result != ErrorCode::kSuccess) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800776 // no need to verify new partition if VerifyPayload failed.
777 return;
778 }
779
780 CompareFilesByBlock(state->old_kernel, state->new_kernel);
781 CompareFilesByBlock(state->a_img, state->b_img);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700782
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800783 chromeos::Blob updated_kernel_partition;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800784 EXPECT_TRUE(utils::ReadFile(state->old_kernel, &updated_kernel_partition));
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800785 ASSERT_GE(updated_kernel_partition.size(), arraysize(kNewData));
786 EXPECT_TRUE(std::equal(std::begin(kNewData), std::end(kNewData),
787 updated_kernel_partition.begin()));
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700788
789 uint64_t new_kernel_size;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800790 chromeos::Blob new_kernel_hash;
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700791 uint64_t new_rootfs_size;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800792 chromeos::Blob new_rootfs_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800793 EXPECT_TRUE(performer->GetNewPartitionInfo(&new_kernel_size,
Don Garrett0dd39852013-04-03 16:55:42 -0700794 &new_kernel_hash,
795 &new_rootfs_size,
796 &new_rootfs_hash));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800797 EXPECT_EQ(kDefaultKernelSize, new_kernel_size);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800798 chromeos::Blob expected_new_kernel_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800799 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(state->new_kernel_data,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700800 &expected_new_kernel_hash));
801 EXPECT_TRUE(expected_new_kernel_hash == new_kernel_hash);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800802 EXPECT_EQ(state->image_size, new_rootfs_size);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800803 chromeos::Blob expected_new_rootfs_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800804 EXPECT_EQ(state->image_size,
805 OmahaHashCalculator::RawHashOfFile(state->b_img,
806 state->image_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700807 &expected_new_rootfs_hash));
808 EXPECT_TRUE(expected_new_rootfs_hash == new_rootfs_hash);
809}
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800810
811void VerifyPayload(DeltaPerformer* performer,
812 DeltaState* state,
813 SignatureTest signature_test) {
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700814 ErrorCode expected_result = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800815 switch (signature_test) {
816 case kSignatureNone:
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700817 expected_result = ErrorCode::kSignedDeltaPayloadExpectedError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800818 break;
819 case kSignatureGeneratedShellBadKey:
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700820 expected_result = ErrorCode::kDownloadPayloadPubKeyVerificationError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800821 break;
822 default: break; // appease gcc
823 }
824
825 VerifyPayloadResult(performer, state, expected_result);
826}
827
828void DoSmallImageTest(bool full_kernel, bool full_rootfs, bool noop,
Darin Petkov8e447e02013-04-16 16:23:50 +0200829 off_t chunk_size,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800830 SignatureTest signature_test,
831 bool hash_checks_mandatory) {
832 DeltaState state;
Alex Deymo52146ce2014-05-29 11:09:45 -0700833 DeltaPerformer *performer = nullptr;
Darin Petkov8e447e02013-04-16 16:23:50 +0200834 GenerateDeltaFile(full_kernel, full_rootfs, noop, chunk_size,
835 signature_test, &state);
Don Garrett0dd39852013-04-03 16:55:42 -0700836
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800837 ScopedPathUnlinker a_img_unlinker(state.a_img);
838 ScopedPathUnlinker b_img_unlinker(state.b_img);
839 ScopedPathUnlinker delta_unlinker(state.delta_path);
840 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
841 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
842 ApplyDeltaFile(full_kernel, full_rootfs, noop, signature_test,
843 &state, hash_checks_mandatory, kValidOperationData,
844 &performer);
845 VerifyPayload(performer, &state, signature_test);
Alex Deymo52146ce2014-05-29 11:09:45 -0700846 delete performer;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800847}
848
849// Calls delta performer's Write method by pretending to pass in bytes from a
850// delta file whose metadata size is actual_metadata_size and tests if all
851// checks are correctly performed if the install plan contains
852// expected_metadata_size and that the result of the parsing are as per
853// hash_checks_mandatory flag.
854void DoMetadataSizeTest(uint64_t expected_metadata_size,
855 uint64_t actual_metadata_size,
856 bool hash_checks_mandatory) {
Alex Deymo8427b4a2014-11-05 14:00:32 -0800857 MockPrefs prefs;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800858 InstallPlan install_plan;
859 install_plan.hash_checks_mandatory = hash_checks_mandatory;
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700860 FakeSystemState fake_system_state;
861 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800862 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
863 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
864
865 // Set a valid magic string and version number 1.
866 EXPECT_TRUE(performer.Write("CrAU", 4));
867 uint64_t version = htobe64(1);
868 EXPECT_TRUE(performer.Write(&version, 8));
869
870 install_plan.metadata_size = expected_metadata_size;
David Zeuthena99981f2013-04-29 13:42:47 -0700871 ErrorCode error_code;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800872 // When filling in size in manifest, exclude the size of the 20-byte header.
873 uint64_t size_in_manifest = htobe64(actual_metadata_size - 20);
874 bool result = performer.Write(&size_in_manifest, 8, &error_code);
875 if (expected_metadata_size == actual_metadata_size ||
876 !hash_checks_mandatory) {
877 EXPECT_TRUE(result);
878 } else {
879 EXPECT_FALSE(result);
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700880 EXPECT_EQ(ErrorCode::kDownloadInvalidMetadataSize, error_code);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800881 }
882
883 EXPECT_LT(performer.Close(), 0);
884}
885
886// Generates a valid delta file but tests the delta performer by suppling
887// different metadata signatures as per omaha_metadata_signature flag and
888// sees if the result of the parsing are as per hash_checks_mandatory flag.
889void DoMetadataSignatureTest(MetadataSignatureTest metadata_signature_test,
890 SignatureTest signature_test,
891 bool hash_checks_mandatory) {
892 DeltaState state;
893
894 // Using kSignatureNone since it doesn't affect the results of our test.
895 // If we've to use other signature options, then we'd have to get the
896 // metadata size again after adding the signing operation to the manifest.
Darin Petkov8e447e02013-04-16 16:23:50 +0200897 GenerateDeltaFile(true, true, false, -1, signature_test, &state);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800898
899 ScopedPathUnlinker a_img_unlinker(state.a_img);
900 ScopedPathUnlinker b_img_unlinker(state.b_img);
901 ScopedPathUnlinker delta_unlinker(state.delta_path);
902 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
903 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
904
905 // Loads the payload and parses the manifest.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800906 chromeos::Blob payload;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800907 EXPECT_TRUE(utils::ReadFile(state.delta_path, &payload));
908 LOG(INFO) << "Payload size: " << payload.size();
909
910 InstallPlan install_plan;
911 install_plan.hash_checks_mandatory = hash_checks_mandatory;
912 install_plan.metadata_size = state.metadata_size;
913
914 DeltaPerformer::MetadataParseResult expected_result, actual_result;
David Zeuthena99981f2013-04-29 13:42:47 -0700915 ErrorCode expected_error, actual_error;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800916
917 // Fill up the metadata signature in install plan according to the test.
918 switch (metadata_signature_test) {
919 case kEmptyMetadataSignature:
920 install_plan.metadata_signature.clear();
921 expected_result = DeltaPerformer::kMetadataParseError;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700922 expected_error = ErrorCode::kDownloadMetadataSignatureMissingError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800923 break;
924
925 case kInvalidMetadataSignature:
926 install_plan.metadata_signature = kBogusMetadataSignature1;
927 expected_result = DeltaPerformer::kMetadataParseError;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700928 expected_error = ErrorCode::kDownloadMetadataSignatureMismatch;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800929 break;
930
931 case kValidMetadataSignature:
932 default:
933 // Set the install plan's metadata size to be the same as the one
934 // in the manifest so that we pass the metadata size checks. Only
935 // then we can get to manifest signature checks.
936 ASSERT_TRUE(PayloadSigner::GetMetadataSignature(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700937 payload.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800938 state.metadata_size,
939 kUnittestPrivateKeyPath,
940 &install_plan.metadata_signature));
941 EXPECT_FALSE(install_plan.metadata_signature.empty());
942 expected_result = DeltaPerformer::kMetadataParseSuccess;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700943 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800944 break;
945 }
946
947 // Ignore the expected result/error if hash checks are not mandatory.
948 if (!hash_checks_mandatory) {
949 expected_result = DeltaPerformer::kMetadataParseSuccess;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700950 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800951 }
952
953 // Create the delta performer object.
Alex Deymo8427b4a2014-11-05 14:00:32 -0800954 MockPrefs prefs;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800955 DeltaPerformer delta_performer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700956 &state.fake_system_state,
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800957 &install_plan);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800958
959 // Use the public key corresponding to the private key used above to
960 // sign the metadata.
961 EXPECT_TRUE(utils::FileExists(kUnittestPublicKeyPath));
962 delta_performer.set_public_key_path(kUnittestPublicKeyPath);
963
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800964 // Init actual_error with an invalid value so that we make sure
965 // ParsePayloadMetadata properly populates it in all cases.
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700966 actual_error = ErrorCode::kUmaReportedMax;
Gilad Arnolddaa27402014-01-23 11:56:17 -0800967 actual_result = delta_performer.ParsePayloadMetadata(payload, &actual_error);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800968
969 EXPECT_EQ(expected_result, actual_result);
970 EXPECT_EQ(expected_error, actual_error);
971
972 // Check that the parsed metadata size is what's expected. This test
973 // implicitly confirms that the metadata signature is valid, if required.
Gilad Arnoldfe133932014-01-14 12:25:50 -0800974 EXPECT_EQ(state.metadata_size, delta_performer.GetMetadataSize());
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800975}
976
977void DoOperationHashMismatchTest(OperationHashTest op_hash_test,
978 bool hash_checks_mandatory) {
979 DeltaState state;
Darin Petkov8e447e02013-04-16 16:23:50 +0200980 GenerateDeltaFile(true, true, false, -1, kSignatureGenerated, &state);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800981 ScopedPathUnlinker a_img_unlinker(state.a_img);
982 ScopedPathUnlinker b_img_unlinker(state.b_img);
983 ScopedPathUnlinker delta_unlinker(state.delta_path);
984 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
985 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
Alex Deymo52146ce2014-05-29 11:09:45 -0700986 DeltaPerformer *performer = nullptr;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800987 ApplyDeltaFile(true, true, false, kSignatureGenerated,
988 &state, hash_checks_mandatory, op_hash_test, &performer);
Alex Deymo52146ce2014-05-29 11:09:45 -0700989 delete performer;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800990}
991
Don Garrettb8dd1d92013-11-22 17:40:02 -0800992
993class DeltaPerformerTest : public ::testing::Test {
Don Garrettb8dd1d92013-11-22 17:40:02 -0800994 public:
995 // Test helper placed where it can easily be friended from DeltaPerformer.
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700996 static void RunManifestValidation(const DeltaArchiveManifest& manifest,
Don Garrettb8dd1d92013-11-22 17:40:02 -0800997 bool full_payload,
998 ErrorCode expected) {
Alex Deymo8427b4a2014-11-05 14:00:32 -0800999 MockPrefs prefs;
Don Garrettb8dd1d92013-11-22 17:40:02 -08001000 InstallPlan install_plan;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001001 FakeSystemState fake_system_state;
1002 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001003
1004 // The install plan is for Full or Delta.
1005 install_plan.is_full_update = full_payload;
1006
1007 // The Manifest we are validating.
1008 performer.manifest_.CopyFrom(manifest);
1009
1010 EXPECT_EQ(expected, performer.ValidateManifest());
1011 }
1012};
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001013
1014TEST(DeltaPerformerTest, ExtentsToByteStringTest) {
1015 uint64_t test[] = {1, 1, 4, 2, kSparseHole, 1, 0, 1};
1016 COMPILE_ASSERT(arraysize(test) % 2 == 0, array_size_uneven);
1017 const uint64_t block_size = 4096;
1018 const uint64_t file_length = 5 * block_size - 13;
1019
1020 google::protobuf::RepeatedPtrField<Extent> extents;
1021 for (size_t i = 0; i < arraysize(test); i += 2) {
1022 Extent* extent = extents.Add();
1023 extent->set_start_block(test[i]);
1024 extent->set_num_blocks(test[i + 1]);
1025 }
1026
1027 string expected_output = "4096:4096,16384:8192,-1:4096,0:4083";
1028 string actual_output;
1029 EXPECT_TRUE(DeltaPerformer::ExtentsToBsdiffPositionsString(extents,
1030 block_size,
1031 file_length,
1032 &actual_output));
1033 EXPECT_EQ(expected_output, actual_output);
1034}
Darin Petkov68c10d12010-10-14 09:24:37 -07001035
Don Garrettb8dd1d92013-11-22 17:40:02 -08001036TEST(DeltaPerformerTest, ValidateManifestFullGoodTest) {
1037 // The Manifest we are validating.
1038 DeltaArchiveManifest manifest;
1039 manifest.mutable_new_kernel_info();
1040 manifest.mutable_new_rootfs_info();
1041 manifest.set_minor_version(DeltaPerformer::kFullPayloadMinorVersion);
1042
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001043 DeltaPerformerTest::RunManifestValidation(manifest, true,
1044 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001045}
1046
1047TEST(DeltaPerformerTest, ValidateManifestDeltaGoodTest) {
1048 // The Manifest we are validating.
1049 DeltaArchiveManifest manifest;
1050 manifest.mutable_old_kernel_info();
1051 manifest.mutable_old_rootfs_info();
1052 manifest.mutable_new_kernel_info();
1053 manifest.mutable_new_rootfs_info();
1054 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1055
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001056 DeltaPerformerTest::RunManifestValidation(manifest, false,
1057 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001058}
1059
1060TEST(DeltaPerformerTest, ValidateManifestFullUnsetMinorVersion) {
1061 // The Manifest we are validating.
1062 DeltaArchiveManifest manifest;
1063
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001064 DeltaPerformerTest::RunManifestValidation(manifest, true,
1065 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001066}
1067
1068TEST(DeltaPerformerTest, ValidateManifestDeltaUnsetMinorVersion) {
1069 // The Manifest we are validating.
1070 DeltaArchiveManifest manifest;
1071
1072 DeltaPerformerTest::RunManifestValidation(
1073 manifest, false,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001074 ErrorCode::kUnsupportedMinorPayloadVersion);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001075}
1076
1077TEST(DeltaPerformerTest, ValidateManifestFullOldKernelTest) {
1078 // The Manifest we are validating.
1079 DeltaArchiveManifest manifest;
1080 manifest.mutable_old_kernel_info();
1081 manifest.mutable_new_kernel_info();
1082 manifest.mutable_new_rootfs_info();
1083 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1084
1085 DeltaPerformerTest::RunManifestValidation(
1086 manifest, true,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001087 ErrorCode::kPayloadMismatchedType);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001088}
1089
1090TEST(DeltaPerformerTest, ValidateManifestFullOldRootfsTest) {
1091 // The Manifest we are validating.
1092 DeltaArchiveManifest manifest;
1093 manifest.mutable_old_rootfs_info();
1094 manifest.mutable_new_kernel_info();
1095 manifest.mutable_new_rootfs_info();
1096 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1097
1098 DeltaPerformerTest::RunManifestValidation(
1099 manifest, true,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001100 ErrorCode::kPayloadMismatchedType);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001101}
1102
1103TEST(DeltaPerformerTest, ValidateManifestBadMinorVersion) {
1104 // The Manifest we are validating.
1105 DeltaArchiveManifest manifest;
1106
1107 // Generate a bad version number.
1108 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion +
1109 10000);
1110
1111 DeltaPerformerTest::RunManifestValidation(
1112 manifest, false,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001113 ErrorCode::kUnsupportedMinorPayloadVersion);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001114}
1115
Darin Petkov68c10d12010-10-14 09:24:37 -07001116TEST(DeltaPerformerTest, RunAsRootSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001117 DoSmallImageTest(false, false, false, -1, kSignatureGenerator,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001118 false);
Darin Petkov8e447e02013-04-16 16:23:50 +02001119}
1120
Don Garrett2ae37872013-10-25 13:33:20 -07001121TEST(DeltaPerformerTest, RunAsRootSmallImageSignaturePlaceholderTest) {
1122 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedPlaceholder,
1123 false);
1124}
1125
1126TEST(DeltaPerformerTest, RunAsRootSmallImageSignaturePlaceholderMismatchTest) {
1127 DeltaState state;
1128 GenerateDeltaFile(false, false, false, -1,
1129 kSignatureGeneratedPlaceholderMismatch, &state);
1130}
1131
Darin Petkov8e447e02013-04-16 16:23:50 +02001132TEST(DeltaPerformerTest, RunAsRootSmallImageChunksTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001133 DoSmallImageTest(false, false, false, kBlockSize, kSignatureGenerator,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001134 false);
Darin Petkov68c10d12010-10-14 09:24:37 -07001135}
1136
1137TEST(DeltaPerformerTest, RunAsRootFullKernelSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001138 DoSmallImageTest(true, false, false, -1, kSignatureGenerator,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001139 false);
Darin Petkov7a22d792010-11-08 14:10:00 -08001140}
1141
1142TEST(DeltaPerformerTest, RunAsRootFullSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001143 DoSmallImageTest(true, true, false, -1, kSignatureGenerator,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001144 true);
Darin Petkov9fa7ec52010-10-18 11:45:23 -07001145}
1146
1147TEST(DeltaPerformerTest, RunAsRootNoopSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001148 DoSmallImageTest(false, false, true, -1, kSignatureGenerator,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001149 false);
Darin Petkov9574f7e2011-01-13 10:48:12 -08001150}
1151
Darin Petkovcbfb0172011-01-14 15:24:45 -08001152TEST(DeltaPerformerTest, RunAsRootSmallImageSignNoneTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001153 DoSmallImageTest(false, false, false, -1, kSignatureNone,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001154 false);
Darin Petkovcbfb0172011-01-14 15:24:45 -08001155}
1156
1157TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001158 DoSmallImageTest(false, false, false, -1, kSignatureGenerated,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001159 true);
Darin Petkovcbfb0172011-01-14 15:24:45 -08001160}
1161
1162TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001163 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShell,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001164 false);
Andrew de los Reyes27f7d372010-10-07 11:26:07 -07001165}
1166
Darin Petkov52dcaeb2011-01-14 15:33:06 -08001167TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellBadKeyTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001168 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellBadKey,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001169 false);
Darin Petkov52dcaeb2011-01-14 15:33:06 -08001170}
1171
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001172TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellRotateCl1Test) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001173 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellRotateCl1,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001174 false);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001175}
1176
1177TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellRotateCl2Test) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001178 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellRotateCl2,
Gilad Arnold434eb0c2013-07-22 06:47:52 -07001179 false);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001180}
1181
Darin Petkov934bb412010-11-18 11:21:35 -08001182TEST(DeltaPerformerTest, BadDeltaMagicTest) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001183 MockPrefs prefs;
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001184 InstallPlan install_plan;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001185 FakeSystemState fake_system_state;
1186 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Darin Petkov934bb412010-11-18 11:21:35 -08001187 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
1188 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
Don Garrette410e0f2011-11-10 15:39:01 -08001189 EXPECT_TRUE(performer.Write("junk", 4));
1190 EXPECT_TRUE(performer.Write("morejunk", 8));
1191 EXPECT_FALSE(performer.Write("morejunk", 8));
Darin Petkov934bb412010-11-18 11:21:35 -08001192 EXPECT_LT(performer.Close(), 0);
1193}
1194
Andrew de los Reyes353777c2010-10-08 10:34:30 -07001195TEST(DeltaPerformerTest, IsIdempotentOperationTest) {
1196 DeltaArchiveManifest_InstallOperation op;
1197 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1198 *(op.add_dst_extents()) = ExtentForRange(0, 5);
1199 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1200 *(op.add_src_extents()) = ExtentForRange(4, 1);
1201 EXPECT_FALSE(DeltaPerformer::IsIdempotentOperation(op));
1202 op.clear_src_extents();
1203 *(op.add_src_extents()) = ExtentForRange(5, 3);
1204 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1205 *(op.add_dst_extents()) = ExtentForRange(20, 6);
1206 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1207 *(op.add_src_extents()) = ExtentForRange(19, 2);
1208 EXPECT_FALSE(DeltaPerformer::IsIdempotentOperation(op));
1209}
1210
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001211TEST(DeltaPerformerTest, WriteUpdatesPayloadState) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001212 MockPrefs prefs;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001213 InstallPlan install_plan;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001214 FakeSystemState fake_system_state;
1215 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001216 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
1217 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
1218
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001219 EXPECT_CALL(*(fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001220 DownloadProgress(4)).Times(1);
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001221 EXPECT_CALL(*(fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001222 DownloadProgress(8)).Times(2);
1223
1224 EXPECT_TRUE(performer.Write("junk", 4));
1225 EXPECT_TRUE(performer.Write("morejunk", 8));
1226 EXPECT_FALSE(performer.Write("morejunk", 8));
1227 EXPECT_LT(performer.Close(), 0);
1228}
1229
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001230TEST(DeltaPerformerTest, MissingMandatoryMetadataSizeTest) {
1231 DoMetadataSizeTest(0, 75456, true);
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001232}
1233
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001234TEST(DeltaPerformerTest, MissingNonMandatoryMetadataSizeTest) {
1235 DoMetadataSizeTest(0, 123456, false);
1236}
1237
1238TEST(DeltaPerformerTest, InvalidMandatoryMetadataSizeTest) {
1239 DoMetadataSizeTest(13000, 140000, true);
1240}
1241
1242TEST(DeltaPerformerTest, InvalidNonMandatoryMetadataSizeTest) {
1243 DoMetadataSizeTest(40000, 50000, false);
1244}
1245
1246TEST(DeltaPerformerTest, ValidMandatoryMetadataSizeTest) {
1247 DoMetadataSizeTest(85376, 85376, true);
1248}
1249
1250TEST(DeltaPerformerTest, RunAsRootMandatoryEmptyMetadataSignatureTest) {
1251 DoMetadataSignatureTest(kEmptyMetadataSignature, kSignatureGenerated, true);
1252}
1253
1254TEST(DeltaPerformerTest, RunAsRootNonMandatoryEmptyMetadataSignatureTest) {
1255 DoMetadataSignatureTest(kEmptyMetadataSignature, kSignatureGenerated, false);
1256}
1257
1258TEST(DeltaPerformerTest, RunAsRootMandatoryInvalidMetadataSignatureTest) {
1259 DoMetadataSignatureTest(kInvalidMetadataSignature, kSignatureGenerated, true);
1260}
1261
1262TEST(DeltaPerformerTest, RunAsRootNonMandatoryInvalidMetadataSignatureTest) {
1263 DoMetadataSignatureTest(kInvalidMetadataSignature, kSignatureGenerated,
1264 false);
1265}
1266
1267TEST(DeltaPerformerTest, RunAsRootMandatoryValidMetadataSignature1Test) {
1268 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureNone, true);
1269}
1270
1271TEST(DeltaPerformerTest, RunAsRootMandatoryValidMetadataSignature2Test) {
1272 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureGenerated, true);
1273}
1274
1275TEST(DeltaPerformerTest, RunAsRootNonMandatoryValidMetadataSignatureTest) {
1276 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureGenerated, false);
1277}
1278
1279TEST(DeltaPerformerTest, RunAsRootMandatoryOperationHashMismatchTest) {
1280 DoOperationHashMismatchTest(kInvalidOperationData, true);
1281}
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001282
David Zeuthene7f89172013-10-31 10:21:04 -07001283TEST(DeltaPerformerTest, UsePublicKeyFromResponse) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001284 MockPrefs prefs;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001285 FakeSystemState fake_system_state;
David Zeuthene7f89172013-10-31 10:21:04 -07001286 InstallPlan install_plan;
1287 base::FilePath key_path;
1288
1289 // The result of the GetPublicKeyResponse() method is based on three things
1290 //
1291 // 1. Whether it's an official build; and
1292 // 2. Whether the Public RSA key to be used is in the root filesystem; and
Alex Vakulenko072359c2014-07-18 11:41:07 -07001293 // 3. Whether the response has a public key
David Zeuthene7f89172013-10-31 10:21:04 -07001294 //
1295 // We test all eight combinations to ensure that we only use the
1296 // public key in the response if
1297 //
1298 // a. it's not an official build; and
1299 // b. there is no key in the root filesystem.
1300
1301 DeltaPerformer *performer = new DeltaPerformer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001302 &fake_system_state,
David Zeuthene7f89172013-10-31 10:21:04 -07001303 &install_plan);
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001304 FakeHardware* fake_hardware = fake_system_state.fake_hardware();
David Zeuthene7f89172013-10-31 10:21:04 -07001305
1306 string temp_dir;
Gilad Arnolda6742b32014-01-11 00:18:34 -08001307 EXPECT_TRUE(utils::MakeTempDirectory("PublicKeyFromResponseTests.XXXXXX",
David Zeuthene7f89172013-10-31 10:21:04 -07001308 &temp_dir));
1309 string non_existing_file = temp_dir + "/non-existing";
1310 string existing_file = temp_dir + "/existing";
Alex Vakulenko75039d72014-03-25 12:36:28 -07001311 EXPECT_EQ(0, System(base::StringPrintf("touch %s", existing_file.c_str())));
David Zeuthene7f89172013-10-31 10:21:04 -07001312
1313 // Non-official build, non-existing public-key, key in response -> true
Don Garrett6646b442013-11-13 15:29:11 -08001314 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001315 performer->public_key_path_ = non_existing_file;
Alex Vakulenkod2779df2014-06-16 13:19:00 -07001316 install_plan.public_key_rsa = "VGVzdAo="; // result of 'echo "Test" | base64'
David Zeuthene7f89172013-10-31 10:21:04 -07001317 EXPECT_TRUE(performer->GetPublicKeyFromResponse(&key_path));
1318 EXPECT_FALSE(key_path.empty());
1319 EXPECT_EQ(unlink(key_path.value().c_str()), 0);
1320 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001321 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001322 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1323
1324 // Non-official build, existing public-key, key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001325 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001326 performer->public_key_path_ = existing_file;
Alex Vakulenkod2779df2014-06-16 13:19:00 -07001327 install_plan.public_key_rsa = "VGVzdAo="; // result of 'echo "Test" | base64'
David Zeuthene7f89172013-10-31 10:21:04 -07001328 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1329 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001330 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001331 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1332
1333 // Non-official build, non-existing public-key, no key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001334 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001335 performer->public_key_path_ = non_existing_file;
1336 install_plan.public_key_rsa = "";
1337 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1338 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001339 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001340 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1341
1342 // Non-official build, existing public-key, no key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001343 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001344 performer->public_key_path_ = existing_file;
1345 install_plan.public_key_rsa = "";
1346 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1347 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001348 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001349 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1350
1351 // Non-official build, non-existing public-key, key in response
1352 // but invalid base64 -> false
Don Garrett6646b442013-11-13 15:29:11 -08001353 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001354 performer->public_key_path_ = non_existing_file;
1355 install_plan.public_key_rsa = "not-valid-base64";
1356 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1357
1358 delete performer;
Alex Deymo10875d92014-11-10 21:52:57 -08001359 EXPECT_TRUE(test_utils::RecursiveUnlinkDir(temp_dir));
David Zeuthene7f89172013-10-31 10:21:04 -07001360}
1361
Allie Wood78750a42015-02-11 15:42:11 -08001362TEST(DeltaPerformerTest, MinorVersionsMatch) {
1363 // Test that the minor version in update_engine.conf that is installed to
1364 // the image matches the supported delta minor version in the update engine.
1365 uint32_t minor_version;
1366 base::FilePath conf_path("update_engine.conf");
1367 EXPECT_TRUE(utils::GetMinorVersion(conf_path, &minor_version));
1368 ASSERT_EQ(DeltaPerformer::kSupportedMinorPayloadVersion, minor_version);
1369}
1370
Andrew de los Reyes09e56d62010-04-23 13:45:53 -07001371} // namespace chromeos_update_engine