blob: e969c577d2759ad074915b8246e9cc305330c917 [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;
Allie Wood9f6f0a52015-03-30 11:25:47 -070065 string result_img;
Jay Srinivasan738fdf32012-12-07 17:40:54 -080066 int image_size;
Andrew de los Reyes27f7d372010-10-07 11:26:07 -070067
Jay Srinivasan738fdf32012-12-07 17:40:54 -080068 string delta_path;
69 uint64_t metadata_size;
Andrew de los Reyes27f7d372010-10-07 11:26:07 -070070
Jay Srinivasan738fdf32012-12-07 17:40:54 -080071 string old_kernel;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080072 chromeos::Blob old_kernel_data;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070073
Jay Srinivasan738fdf32012-12-07 17:40:54 -080074 string new_kernel;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080075 chromeos::Blob new_kernel_data;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070076
Allie Wood9f6f0a52015-03-30 11:25:47 -070077 string result_kernel;
78 chromeos::Blob result_kernel_data;
79
Jay Srinivasan738fdf32012-12-07 17:40:54 -080080 // The in-memory copy of delta file.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -080081 chromeos::Blob delta;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -080082
83 // The mock system state object with which we initialize the
84 // delta performer.
Gilad Arnold5bb4c902014-04-10 12:32:13 -070085 FakeSystemState fake_system_state;
Jay Srinivasan738fdf32012-12-07 17:40:54 -080086};
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070087
Jay Srinivasan738fdf32012-12-07 17:40:54 -080088enum SignatureTest {
89 kSignatureNone, // No payload signing.
90 kSignatureGenerator, // Sign the payload at generation time.
91 kSignatureGenerated, // Sign the payload after it's generated.
Don Garrett2ae37872013-10-25 13:33:20 -070092 kSignatureGeneratedPlaceholder, // Insert placeholder signatures, then real.
Alex Vakulenkod2779df2014-06-16 13:19:00 -070093 kSignatureGeneratedPlaceholderMismatch, // Insert a wrong sized placeholder.
Jay Srinivasan738fdf32012-12-07 17:40:54 -080094 kSignatureGeneratedShell, // Sign the generated payload through shell cmds.
95 kSignatureGeneratedShellBadKey, // Sign with a bad key through shell cmds.
96 kSignatureGeneratedShellRotateCl1, // Rotate key, test client v1
97 kSignatureGeneratedShellRotateCl2, // Rotate key, test client v2
98};
Andrew de los Reyes09e56d62010-04-23 13:45:53 -070099
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800100// Different options that determine what we should fill into the
101// install_plan.metadata_signature to simulate the contents received in the
102// Omaha response.
103enum MetadataSignatureTest {
104 kEmptyMetadataSignature,
105 kInvalidMetadataSignature,
106 kValidMetadataSignature,
107};
108
109enum OperationHashTest {
110 kInvalidOperationData,
111 kValidOperationData,
112};
113
Allie Woodfdf00512015-03-02 13:34:55 -0800114// Chuck size used for full payloads during test.
115size_t kDefaultFullChunkSize = 1024 * 1024;
116
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700117} // namespace
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800118
Allie Woodfdf00512015-03-02 13:34:55 -0800119class DeltaPerformerTest : public ::testing::Test {
120 public:
121 // Test helper placed where it can easily be friended from DeltaPerformer.
122 static void RunManifestValidation(const DeltaArchiveManifest& manifest,
123 bool full_payload,
124 ErrorCode expected) {
125 MockPrefs prefs;
126 InstallPlan install_plan;
127 FakeSystemState fake_system_state;
128 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
129
130 // The install plan is for Full or Delta.
131 install_plan.is_full_update = full_payload;
132
133 // The Manifest we are validating.
134 performer.manifest_.CopyFrom(manifest);
135
136 EXPECT_EQ(expected, performer.ValidateManifest());
137 }
138
139 static void SetSupportedVersion(DeltaPerformer* performer,
140 uint64_t minor_version) {
141 performer->supported_minor_version_ = minor_version;
142 }
143};
144
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800145static void CompareFilesByBlock(const string& a_file, const string& b_file) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800146 chromeos::Blob a_data, b_data;
Andrew de los Reyes3270f742010-07-15 22:28:14 -0700147 EXPECT_TRUE(utils::ReadFile(a_file, &a_data)) << "file failed: " << a_file;
148 EXPECT_TRUE(utils::ReadFile(b_file, &b_data)) << "file failed: " << b_file;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700149
150 EXPECT_EQ(a_data.size(), b_data.size());
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700151 EXPECT_EQ(0, a_data.size() % kBlockSize);
152 for (size_t i = 0; i < a_data.size(); i += kBlockSize) {
153 EXPECT_EQ(0, i % kBlockSize);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800154 chromeos::Blob a_sub(&a_data[i], &a_data[i + kBlockSize]);
155 chromeos::Blob b_sub(&b_data[i], &b_data[i + kBlockSize]);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700156 EXPECT_TRUE(a_sub == b_sub) << "Block " << (i/kBlockSize) << " differs";
157 }
158}
159
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800160static bool WriteSparseFile(const string& path, off_t size) {
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700161 int fd = open(path.c_str(), O_CREAT | O_TRUNC | O_WRONLY, 0644);
162 TEST_AND_RETURN_FALSE_ERRNO(fd >= 0);
163 ScopedFdCloser fd_closer(&fd);
164 off_t rc = lseek(fd, size + 1, SEEK_SET);
165 TEST_AND_RETURN_FALSE_ERRNO(rc != static_cast<off_t>(-1));
166 int return_code = ftruncate(fd, size);
167 TEST_AND_RETURN_FALSE_ERRNO(return_code == 0);
168 return true;
169}
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700170
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800171static size_t GetSignatureSize(const string& private_key_path) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800172 const chromeos::Blob data(1, 'x');
173 chromeos::Blob hash;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800174 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(data, &hash));
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800175 chromeos::Blob signature;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800176 EXPECT_TRUE(PayloadSigner::SignHash(hash,
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800177 private_key_path,
Darin Petkovcbfb0172011-01-14 15:24:45 -0800178 &signature));
179 return signature.size();
180}
181
Don Garrett2ae37872013-10-25 13:33:20 -0700182static bool InsertSignaturePlaceholder(int signature_size,
183 const string& payload_path,
184 uint64_t* out_metadata_size) {
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800185 vector<chromeos::Blob> signatures;
186 signatures.push_back(chromeos::Blob(signature_size, 0));
Don Garrett2ae37872013-10-25 13:33:20 -0700187
188 return PayloadSigner::AddSignatureToPayload(
189 payload_path,
190 signatures,
191 payload_path,
192 out_metadata_size);
193}
194
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800195static void SignGeneratedPayload(const string& payload_path,
196 uint64_t* out_metadata_size) {
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800197 int signature_size = GetSignatureSize(kUnittestPrivateKeyPath);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800198 chromeos::Blob hash;
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700199 ASSERT_TRUE(PayloadSigner::HashPayloadForSigning(
200 payload_path,
201 vector<int>(1, signature_size),
202 &hash));
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800203 chromeos::Blob signature;
Darin Petkovcbfb0172011-01-14 15:24:45 -0800204 ASSERT_TRUE(PayloadSigner::SignHash(hash,
205 kUnittestPrivateKeyPath,
206 &signature));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700207 ASSERT_TRUE(PayloadSigner::AddSignatureToPayload(
208 payload_path,
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800209 vector<chromeos::Blob>(1, signature),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800210 payload_path,
211 out_metadata_size));
Alex Deymo923d8fa2014-07-15 17:58:51 -0700212 EXPECT_TRUE(PayloadVerifier::VerifySignedPayload(
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700213 payload_path,
214 kUnittestPublicKeyPath,
215 kSignatureMessageOriginalVersion));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800216}
217
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800218static void SignGeneratedShellPayload(SignatureTest signature_test,
219 const string& payload_path) {
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800220 string private_key_path = kUnittestPrivateKeyPath;
221 if (signature_test == kSignatureGeneratedShellBadKey) {
Gilad Arnolda6742b32014-01-11 00:18:34 -0800222 ASSERT_TRUE(utils::MakeTempFile("key.XXXXXX",
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800223 &private_key_path,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700224 nullptr));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800225 } else {
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700226 ASSERT_TRUE(signature_test == kSignatureGeneratedShell ||
227 signature_test == kSignatureGeneratedShellRotateCl1 ||
228 signature_test == kSignatureGeneratedShellRotateCl2);
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800229 }
230 ScopedPathUnlinker key_unlinker(private_key_path);
231 key_unlinker.set_should_remove(signature_test ==
232 kSignatureGeneratedShellBadKey);
233 // Generates a new private key that will not match the public key.
234 if (signature_test == kSignatureGeneratedShellBadKey) {
235 LOG(INFO) << "Generating a mismatched private key.";
Alex Deymo719bfff2014-07-11 12:12:32 -0700236 ASSERT_EQ(0, System(base::StringPrintf(
237 "%s genrsa -out %s 2048",
238 utils::GetPathOnBoard("openssl").c_str(), private_key_path.c_str())));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800239 }
240 int signature_size = GetSignatureSize(private_key_path);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800241 string hash_file;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700242 ASSERT_TRUE(utils::MakeTempFile("hash.XXXXXX", &hash_file, nullptr));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800243 ScopedPathUnlinker hash_unlinker(hash_file);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700244 string signature_size_string;
245 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
246 signature_test == kSignatureGeneratedShellRotateCl2)
Alex Vakulenko75039d72014-03-25 12:36:28 -0700247 signature_size_string = base::StringPrintf("%d:%d",
248 signature_size, signature_size);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700249 else
Alex Vakulenko75039d72014-03-25 12:36:28 -0700250 signature_size_string = base::StringPrintf("%d", signature_size);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800251 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700252 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700253 "./delta_generator -in_file=%s -signature_size=%s "
254 "-out_hash_file=%s",
Darin Petkovcbfb0172011-01-14 15:24:45 -0800255 payload_path.c_str(),
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700256 signature_size_string.c_str(),
Darin Petkovcbfb0172011-01-14 15:24:45 -0800257 hash_file.c_str())));
258
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700259 // Pad the hash
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800260 chromeos::Blob hash;
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700261 ASSERT_TRUE(utils::ReadFile(hash_file, &hash));
Alex Deymo923d8fa2014-07-15 17:58:51 -0700262 ASSERT_TRUE(PayloadVerifier::PadRSA2048SHA256Hash(&hash));
Alex Deymo10875d92014-11-10 21:52:57 -0800263 ASSERT_TRUE(test_utils::WriteFileVector(hash_file, hash));
Andrew de los Reyesbdfaaf02011-03-30 10:35:12 -0700264
Darin Petkovcbfb0172011-01-14 15:24:45 -0800265 string sig_file;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700266 ASSERT_TRUE(utils::MakeTempFile("signature.XXXXXX", &sig_file, nullptr));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800267 ScopedPathUnlinker sig_unlinker(sig_file);
268 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700269 System(base::StringPrintf(
Alex Deymo719bfff2014-07-11 12:12:32 -0700270 "%s rsautl -raw -sign -inkey %s -in %s -out %s",
271 utils::GetPathOnBoard("openssl").c_str(),
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800272 private_key_path.c_str(),
Darin Petkovcbfb0172011-01-14 15:24:45 -0800273 hash_file.c_str(),
274 sig_file.c_str())));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700275 string sig_file2;
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700276 ASSERT_TRUE(utils::MakeTempFile("signature.XXXXXX", &sig_file2, nullptr));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700277 ScopedPathUnlinker sig2_unlinker(sig_file2);
278 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
279 signature_test == kSignatureGeneratedShellRotateCl2) {
280 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700281 System(base::StringPrintf(
Alex Deymo719bfff2014-07-11 12:12:32 -0700282 "%s rsautl -raw -sign -inkey %s -in %s -out %s",
283 utils::GetPathOnBoard("openssl").c_str(),
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700284 kUnittestPrivateKey2Path,
285 hash_file.c_str(),
286 sig_file2.c_str())));
287 // Append second sig file to first path
288 sig_file += ":" + sig_file2;
289 }
290
Darin Petkovcbfb0172011-01-14 15:24:45 -0800291 ASSERT_EQ(0,
Alex Vakulenko75039d72014-03-25 12:36:28 -0700292 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700293 "./delta_generator -in_file=%s -signature_file=%s "
294 "-out_file=%s",
Darin Petkovcbfb0172011-01-14 15:24:45 -0800295 payload_path.c_str(),
296 sig_file.c_str(),
297 payload_path.c_str())));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800298 int verify_result =
Alex Vakulenko75039d72014-03-25 12:36:28 -0700299 System(base::StringPrintf(
Steve Fung97b6f5a2014-10-07 12:39:51 -0700300 "./delta_generator -in_file=%s -public_key=%s -public_key_version=%d",
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700301 payload_path.c_str(),
302 signature_test == kSignatureGeneratedShellRotateCl2 ?
303 kUnittestPublicKey2Path : kUnittestPublicKeyPath,
304 signature_test == kSignatureGeneratedShellRotateCl2 ? 2 : 1));
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800305 if (signature_test == kSignatureGeneratedShellBadKey) {
306 ASSERT_NE(0, verify_result);
307 } else {
308 ASSERT_EQ(0, verify_result);
309 }
Darin Petkovcbfb0172011-01-14 15:24:45 -0800310}
311
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800312static void GenerateDeltaFile(bool full_kernel,
313 bool full_rootfs,
314 bool noop,
Darin Petkov8e447e02013-04-16 16:23:50 +0200315 off_t chunk_size,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800316 SignatureTest signature_test,
Allie Woodfdf00512015-03-02 13:34:55 -0800317 DeltaState *state,
318 uint32_t minor_version) {
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700319 EXPECT_TRUE(utils::MakeTempFile("a_img.XXXXXX", &state->a_img, nullptr));
320 EXPECT_TRUE(utils::MakeTempFile("b_img.XXXXXX", &state->b_img, nullptr));
Allie Wood9f6f0a52015-03-30 11:25:47 -0700321
322 // result_img is used in minor version 2. Instead of applying the update
323 // in-place on A, we apply it to a new image, result_img.
324 EXPECT_TRUE(
325 utils::MakeTempFile("result_img.XXXXXX", &state->result_img, nullptr));
Alex Deymo10875d92014-11-10 21:52:57 -0800326 test_utils::CreateExtImageAtPath(state->a_img, nullptr);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700327
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800328 state->image_size = static_cast<int>(utils::FileSize(state->a_img));
Darin Petkov7ea32332010-10-13 10:46:11 -0700329
330 // Extend the "partitions" holding the file system a bit.
331 EXPECT_EQ(0, System(base::StringPrintf(
Alex Deymo1f93d032015-03-10 18:58:32 -0700332 "dd if=/dev/zero of=%s seek=%d bs=1 count=1 status=none",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800333 state->a_img.c_str(),
334 state->image_size + 1024 * 1024 - 1)));
335 EXPECT_EQ(state->image_size + 1024 * 1024, utils::FileSize(state->a_img));
Darin Petkov7ea32332010-10-13 10:46:11 -0700336
Don Garrett0dd39852013-04-03 16:55:42 -0700337 // Create ImageInfo A & B
338 ImageInfo old_image_info;
339 ImageInfo new_image_info;
340
341 if (!full_rootfs) {
342 old_image_info.set_channel("src-channel");
343 old_image_info.set_board("src-board");
344 old_image_info.set_version("src-version");
345 old_image_info.set_key("src-key");
346 old_image_info.set_build_channel("src-build-channel");
347 old_image_info.set_build_version("src-build-version");
348 }
349
350 new_image_info.set_channel("test-channel");
351 new_image_info.set_board("test-board");
352 new_image_info.set_version("test-version");
353 new_image_info.set_key("test-key");
354 new_image_info.set_build_channel("test-build-channel");
355 new_image_info.set_build_version("test-build-version");
356
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700357 // Make some changes to the A image.
358 {
359 string a_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800360 ScopedLoopMounter b_mounter(state->a_img, &a_mnt, 0);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700361
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800362 chromeos::Blob hardtocompress;
Darin Petkov8e447e02013-04-16 16:23:50 +0200363 while (hardtocompress.size() < 3 * kBlockSize) {
364 hardtocompress.insert(hardtocompress.end(),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800365 std::begin(kRandomString), std::end(kRandomString));
Darin Petkov8e447e02013-04-16 16:23:50 +0200366 }
Alex Vakulenko75039d72014-03-25 12:36:28 -0700367 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/hardtocompress",
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800368 a_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700369 hardtocompress.data(),
Darin Petkov8e447e02013-04-16 16:23:50 +0200370 hardtocompress.size()));
371
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800372 chromeos::Blob zeros(16 * 1024, 0);
Darin Petkov8a075a72013-04-25 14:46:09 +0200373 EXPECT_EQ(zeros.size(),
Ben Chan736fcb52014-05-21 18:28:22 -0700374 base::WriteFile(base::FilePath(base::StringPrintf(
375 "%s/move-to-sparse", a_mnt.c_str())),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800376 reinterpret_cast<const char*>(zeros.data()),
377 zeros.size()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200378
379 EXPECT_TRUE(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700380 WriteSparseFile(base::StringPrintf("%s/move-from-sparse",
381 a_mnt.c_str()), 16 * 1024));
Darin Petkov8a075a72013-04-25 14:46:09 +0200382
Alex Vakulenko75039d72014-03-25 12:36:28 -0700383 EXPECT_EQ(0,
Alex Deymo10875d92014-11-10 21:52:57 -0800384 System(base::StringPrintf("dd if=/dev/zero of=%s/move-semi-sparse"
Alex Deymo1f93d032015-03-10 18:58:32 -0700385 " bs=1 seek=4096 count=1 status=none",
Alex Deymo10875d92014-11-10 21:52:57 -0800386 a_mnt.c_str()).c_str()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200387
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700388 // Write 1 MiB of 0xff to try to catch the case where writing a bsdiff
389 // patch fails to zero out the final block.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800390 chromeos::Blob ones(1024 * 1024, 0xff);
Alex Vakulenko75039d72014-03-25 12:36:28 -0700391 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/ones",
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800392 a_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700393 ones.data(),
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700394 ones.size()));
395 }
396
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700397 if (noop) {
Alex Vakulenko75039d72014-03-25 12:36:28 -0700398 EXPECT_TRUE(base::CopyFile(base::FilePath(state->a_img),
399 base::FilePath(state->b_img)));
Don Garrett0dd39852013-04-03 16:55:42 -0700400 old_image_info = new_image_info;
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700401 } else {
Allie Wood9f6f0a52015-03-30 11:25:47 -0700402 if (minor_version == kSourceMinorPayloadVersion) {
403 // Create a result image with image_size bytes of garbage, followed by
404 // zeroes after the rootfs, like image A and B have.
405 chromeos::Blob ones(state->image_size, 0xff);
406 ones.insert(ones.end(), 1024 * 1024, 0);
407 EXPECT_TRUE(utils::WriteFile(state->result_img.c_str(),
408 ones.data(),
409 ones.size()));
410 EXPECT_EQ(utils::FileSize(state->a_img),
411 utils::FileSize(state->result_img));
412 }
413
Alex Deymo10875d92014-11-10 21:52:57 -0800414 test_utils::CreateExtImageAtPath(state->b_img, nullptr);
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700415 EXPECT_EQ(0, System(base::StringPrintf(
Alex Deymo1f93d032015-03-10 18:58:32 -0700416 "dd if=/dev/zero of=%s seek=%d bs=1 count=1 status=none",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800417 state->b_img.c_str(),
418 state->image_size + 1024 * 1024 - 1)));
419 EXPECT_EQ(state->image_size + 1024 * 1024, utils::FileSize(state->b_img));
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700420
421 // Make some changes to the B image.
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700422 string b_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800423 ScopedLoopMounter b_mounter(state->b_img, &b_mnt, 0);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700424
Alex Deymo10875d92014-11-10 21:52:57 -0800425 EXPECT_EQ(0, System(base::StringPrintf("cp %s/hello %s/hello2",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700426 b_mnt.c_str(),
427 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800428 EXPECT_EQ(0, System(base::StringPrintf("rm %s/hello",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700429 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800430 EXPECT_EQ(0, System(base::StringPrintf("mv %s/hello2 %s/hello",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700431 b_mnt.c_str(),
432 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800433 EXPECT_EQ(0, System(base::StringPrintf("echo foo > %s/foo",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700434 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800435 EXPECT_EQ(0, System(base::StringPrintf("touch %s/emptyfile",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700436 b_mnt.c_str()).c_str()));
437 EXPECT_TRUE(WriteSparseFile(base::StringPrintf("%s/fullsparse",
438 b_mnt.c_str()),
439 1024 * 1024));
Darin Petkov8a075a72013-04-25 14:46:09 +0200440
441 EXPECT_TRUE(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700442 WriteSparseFile(base::StringPrintf("%s/move-to-sparse", b_mnt.c_str()),
Darin Petkov8a075a72013-04-25 14:46:09 +0200443 16 * 1024));
444
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800445 chromeos::Blob zeros(16 * 1024, 0);
Darin Petkov8a075a72013-04-25 14:46:09 +0200446 EXPECT_EQ(zeros.size(),
Ben Chan736fcb52014-05-21 18:28:22 -0700447 base::WriteFile(base::FilePath(base::StringPrintf(
448 "%s/move-from-sparse", b_mnt.c_str())),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800449 reinterpret_cast<const char*>(zeros.data()),
450 zeros.size()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200451
Alex Deymo10875d92014-11-10 21:52:57 -0800452 EXPECT_EQ(0, System(base::StringPrintf("dd if=/dev/zero "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700453 "of=%s/move-semi-sparse "
Alex Deymo1f93d032015-03-10 18:58:32 -0700454 "bs=1 seek=4096 count=1 status=none",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700455 b_mnt.c_str()).c_str()));
Darin Petkov8a075a72013-04-25 14:46:09 +0200456
Alex Deymo10875d92014-11-10 21:52:57 -0800457 EXPECT_EQ(0, System(base::StringPrintf("dd if=/dev/zero "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700458 "of=%s/partsparse bs=1 "
Alex Deymo1f93d032015-03-10 18:58:32 -0700459 "seek=4096 count=1 status=none",
Alex Vakulenko75039d72014-03-25 12:36:28 -0700460 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800461 EXPECT_EQ(0, System(base::StringPrintf("cp %s/srchardlink0 %s/tmp && "
Alex Vakulenko75039d72014-03-25 12:36:28 -0700462 "mv %s/tmp %s/srchardlink1",
463 b_mnt.c_str(),
464 b_mnt.c_str(),
465 b_mnt.c_str(),
466 b_mnt.c_str()).c_str()));
Alex Deymo10875d92014-11-10 21:52:57 -0800467 EXPECT_EQ(0, System(
468 base::StringPrintf("rm %s/boguslink && echo foobar > %s/boguslink",
469 b_mnt.c_str(), b_mnt.c_str()).c_str()));
Darin Petkov8e447e02013-04-16 16:23:50 +0200470
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800471 chromeos::Blob hardtocompress;
Darin Petkov8e447e02013-04-16 16:23:50 +0200472 while (hardtocompress.size() < 3 * kBlockSize) {
473 hardtocompress.insert(hardtocompress.end(),
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800474 std::begin(kRandomString), std::end(kRandomString));
Darin Petkov8e447e02013-04-16 16:23:50 +0200475 }
Alex Vakulenko75039d72014-03-25 12:36:28 -0700476 EXPECT_TRUE(utils::WriteFile(base::StringPrintf("%s/hardtocompress",
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700477 b_mnt.c_str()).c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700478 hardtocompress.data(),
Darin Petkov8e447e02013-04-16 16:23:50 +0200479 hardtocompress.size()));
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700480 }
481
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700482 string old_kernel;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800483 EXPECT_TRUE(utils::MakeTempFile("old_kernel.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800484 &state->old_kernel,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700485 nullptr));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700486
487 string new_kernel;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800488 EXPECT_TRUE(utils::MakeTempFile("new_kernel.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800489 &state->new_kernel,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700490 nullptr));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700491
Allie Wood9f6f0a52015-03-30 11:25:47 -0700492 string result_kernel;
493 EXPECT_TRUE(utils::MakeTempFile("result_kernel.XXXXXX",
494 &state->result_kernel,
495 nullptr));
496
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800497 state->old_kernel_data.resize(kDefaultKernelSize);
498 state->new_kernel_data.resize(state->old_kernel_data.size());
Allie Wood9f6f0a52015-03-30 11:25:47 -0700499 state->result_kernel_data.resize(state->old_kernel_data.size());
Alex Deymo10875d92014-11-10 21:52:57 -0800500 test_utils::FillWithData(&state->old_kernel_data);
501 test_utils::FillWithData(&state->new_kernel_data);
Allie Wood9f6f0a52015-03-30 11:25:47 -0700502 test_utils::FillWithData(&state->result_kernel_data);
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700503
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700504 // change the new kernel data
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800505 std::copy(std::begin(kNewData), std::end(kNewData),
506 state->new_kernel_data.begin());
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700507
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700508 if (noop) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800509 state->old_kernel_data = state->new_kernel_data;
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700510 }
511
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700512 // Write kernels to disk
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800513 EXPECT_TRUE(utils::WriteFile(state->old_kernel.c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700514 state->old_kernel_data.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800515 state->old_kernel_data.size()));
516 EXPECT_TRUE(utils::WriteFile(state->new_kernel.c_str(),
Alex Vakulenko75039d72014-03-25 12:36:28 -0700517 state->new_kernel_data.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800518 state->new_kernel_data.size()));
Allie Wood9f6f0a52015-03-30 11:25:47 -0700519 EXPECT_TRUE(utils::WriteFile(state->result_kernel.c_str(),
520 state->result_kernel_data.data(),
521 state->result_kernel_data.size()));
Andrew de los Reyesf4c7ef12010-04-30 10:37:00 -0700522
Gilad Arnolda6742b32014-01-11 00:18:34 -0800523 EXPECT_TRUE(utils::MakeTempFile("delta.XXXXXX",
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800524 &state->delta_path,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700525 nullptr));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800526 LOG(INFO) << "delta path: " << state->delta_path;
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700527 {
528 string a_mnt, b_mnt;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800529 ScopedLoopMounter a_mounter(state->a_img, &a_mnt, MS_RDONLY);
530 ScopedLoopMounter b_mounter(state->b_img, &b_mnt, MS_RDONLY);
Darin Petkovcbfb0172011-01-14 15:24:45 -0800531 const string private_key =
532 signature_test == kSignatureGenerator ? kUnittestPrivateKeyPath : "";
Alex Deymof1cbe172015-03-05 15:58:37 -0800533
534 PayloadGenerationConfig payload_config;
535 payload_config.is_delta = !full_rootfs;
536 payload_config.chunk_size = chunk_size;
Alex Deymo9b244df2015-03-11 21:51:18 -0700537 payload_config.rootfs_partition_size = kRootFSPartitionSize;
Allie Woodfdf00512015-03-02 13:34:55 -0800538 payload_config.minor_version = minor_version;
Alex Deymof1cbe172015-03-05 15:58:37 -0800539 if (!full_rootfs) {
540 payload_config.source.rootfs_part = state->a_img;
541 payload_config.source.rootfs_mountpt = a_mnt;
542 if (!full_kernel)
543 payload_config.source.kernel_part = state->old_kernel;
544 payload_config.source.image_info = old_image_info;
545 EXPECT_TRUE(payload_config.source.LoadImageSize());
546
Alex Deymof1cbe172015-03-05 15:58:37 -0800547 } else {
Allie Woodfdf00512015-03-02 13:34:55 -0800548 if (payload_config.chunk_size == -1)
549 payload_config.chunk_size = kDefaultFullChunkSize;
Alex Deymof1cbe172015-03-05 15:58:37 -0800550 }
551 payload_config.target.rootfs_part = state->b_img;
552 payload_config.target.rootfs_mountpt = b_mnt;
553 payload_config.target.kernel_part = state->new_kernel;
554 payload_config.target.image_info = new_image_info;
555 EXPECT_TRUE(payload_config.target.LoadImageSize());
556
557 EXPECT_TRUE(payload_config.Validate());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700558 EXPECT_TRUE(
Alex Deymo477aec22015-03-24 23:40:48 -0700559 GenerateUpdatePayloadFile(
Alex Deymof1cbe172015-03-05 15:58:37 -0800560 payload_config,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800561 state->delta_path,
562 private_key,
563 &state->metadata_size));
Darin Petkov9574f7e2011-01-13 10:48:12 -0800564 }
565
Don Garrett2ae37872013-10-25 13:33:20 -0700566 if (signature_test == kSignatureGeneratedPlaceholder ||
567 signature_test == kSignatureGeneratedPlaceholderMismatch) {
Don Garrett2ae37872013-10-25 13:33:20 -0700568 int signature_size = GetSignatureSize(kUnittestPrivateKeyPath);
569 LOG(INFO) << "Inserting placeholder signature.";
570 ASSERT_TRUE(InsertSignaturePlaceholder(signature_size, state->delta_path,
571 &state->metadata_size));
572
573 if (signature_test == kSignatureGeneratedPlaceholderMismatch) {
574 signature_size -= 1;
575 LOG(INFO) << "Inserting mismatched placeholder signature.";
576 ASSERT_FALSE(InsertSignaturePlaceholder(signature_size, state->delta_path,
577 &state->metadata_size));
578 return;
579 }
580 }
581
582 if (signature_test == kSignatureGenerated ||
583 signature_test == kSignatureGeneratedPlaceholder ||
584 signature_test == kSignatureGeneratedPlaceholderMismatch) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800585 // Generate the signed payload and update the metadata size in state to
586 // reflect the new size after adding the signature operation to the
587 // manifest.
Don Garrett2ae37872013-10-25 13:33:20 -0700588 LOG(INFO) << "Signing payload.";
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800589 SignGeneratedPayload(state->delta_path, &state->metadata_size);
Darin Petkov52dcaeb2011-01-14 15:33:06 -0800590 } else if (signature_test == kSignatureGeneratedShell ||
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700591 signature_test == kSignatureGeneratedShellBadKey ||
592 signature_test == kSignatureGeneratedShellRotateCl1 ||
593 signature_test == kSignatureGeneratedShellRotateCl2) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800594 SignGeneratedShellPayload(signature_test, state->delta_path);
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700595 }
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800596}
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700597
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800598static void ApplyDeltaFile(bool full_kernel, bool full_rootfs, bool noop,
599 SignatureTest signature_test, DeltaState* state,
600 bool hash_checks_mandatory,
601 OperationHashTest op_hash_test,
Allie Woodfdf00512015-03-02 13:34:55 -0800602 DeltaPerformer** performer,
603 uint32_t minor_version) {
Darin Petkov36a58222010-10-07 22:00:09 -0700604 // Check the metadata.
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700605 {
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700606 DeltaArchiveManifest manifest;
Alex Deymo923d8fa2014-07-15 17:58:51 -0700607 EXPECT_TRUE(PayloadVerifier::LoadPayload(state->delta_path,
608 &state->delta,
609 &manifest,
610 &state->metadata_size));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800611 LOG(INFO) << "Metadata size: " << state->metadata_size;
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700612
Don Garrett0dd39852013-04-03 16:55:42 -0700613
614
Darin Petkovcbfb0172011-01-14 15:24:45 -0800615 if (signature_test == kSignatureNone) {
616 EXPECT_FALSE(manifest.has_signatures_offset());
617 EXPECT_FALSE(manifest.has_signatures_size());
618 } else {
619 EXPECT_TRUE(manifest.has_signatures_offset());
620 EXPECT_TRUE(manifest.has_signatures_size());
621 Signatures sigs_message;
622 EXPECT_TRUE(sigs_message.ParseFromArray(
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800623 &state->delta[state->metadata_size + manifest.signatures_offset()],
Darin Petkovcbfb0172011-01-14 15:24:45 -0800624 manifest.signatures_size()));
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700625 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
626 signature_test == kSignatureGeneratedShellRotateCl2)
627 EXPECT_EQ(2, sigs_message.signatures_size());
628 else
629 EXPECT_EQ(1, sigs_message.signatures_size());
Darin Petkovcbfb0172011-01-14 15:24:45 -0800630 const Signatures_Signature& signature = sigs_message.signatures(0);
631 EXPECT_EQ(1, signature.version());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700632
Darin Petkovcbfb0172011-01-14 15:24:45 -0800633 uint64_t expected_sig_data_length = 0;
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700634 vector<string> key_paths{kUnittestPrivateKeyPath};
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700635 if (signature_test == kSignatureGeneratedShellRotateCl1 ||
636 signature_test == kSignatureGeneratedShellRotateCl2) {
637 key_paths.push_back(kUnittestPrivateKey2Path);
638 }
Darin Petkovcbfb0172011-01-14 15:24:45 -0800639 EXPECT_TRUE(PayloadSigner::SignatureBlobLength(
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -0700640 key_paths,
641 &expected_sig_data_length));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800642 EXPECT_EQ(expected_sig_data_length, manifest.signatures_size());
643 EXPECT_FALSE(signature.data().empty());
644 }
Darin Petkov36a58222010-10-07 22:00:09 -0700645
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700646 if (noop) {
Gilad Arnold063181c2015-02-05 12:24:49 -0800647 EXPECT_EQ(0, manifest.install_operations_size());
Darin Petkov9fa7ec52010-10-18 11:45:23 -0700648 EXPECT_EQ(1, manifest.kernel_install_operations_size());
649 }
650
Darin Petkovd43d6902010-10-14 11:17:50 -0700651 if (full_kernel) {
652 EXPECT_FALSE(manifest.has_old_kernel_info());
653 } else {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800654 EXPECT_EQ(state->old_kernel_data.size(),
655 manifest.old_kernel_info().size());
Darin Petkovd43d6902010-10-14 11:17:50 -0700656 EXPECT_FALSE(manifest.old_kernel_info().hash().empty());
657 }
Darin Petkov698d0412010-10-13 10:59:44 -0700658
Don Garrett0dd39852013-04-03 16:55:42 -0700659 EXPECT_EQ(manifest.new_image_info().channel(), "test-channel");
660 EXPECT_EQ(manifest.new_image_info().board(), "test-board");
661 EXPECT_EQ(manifest.new_image_info().version(), "test-version");
662 EXPECT_EQ(manifest.new_image_info().key(), "test-key");
663 EXPECT_EQ(manifest.new_image_info().build_channel(), "test-build-channel");
664 EXPECT_EQ(manifest.new_image_info().build_version(), "test-build-version");
665
666 if (!full_rootfs) {
Don Garrett0dd39852013-04-03 16:55:42 -0700667 if (noop) {
668 EXPECT_EQ(manifest.old_image_info().channel(), "test-channel");
669 EXPECT_EQ(manifest.old_image_info().board(), "test-board");
670 EXPECT_EQ(manifest.old_image_info().version(), "test-version");
671 EXPECT_EQ(manifest.old_image_info().key(), "test-key");
672 EXPECT_EQ(manifest.old_image_info().build_channel(),
673 "test-build-channel");
674 EXPECT_EQ(manifest.old_image_info().build_version(),
675 "test-build-version");
676 } else {
677 EXPECT_EQ(manifest.old_image_info().channel(), "src-channel");
678 EXPECT_EQ(manifest.old_image_info().board(), "src-board");
679 EXPECT_EQ(manifest.old_image_info().version(), "src-version");
680 EXPECT_EQ(manifest.old_image_info().key(), "src-key");
681 EXPECT_EQ(manifest.old_image_info().build_channel(),
682 "src-build-channel");
683 EXPECT_EQ(manifest.old_image_info().build_version(),
684 "src-build-version");
685 }
686 }
687
688
Darin Petkov7a22d792010-11-08 14:10:00 -0800689 if (full_rootfs) {
690 EXPECT_FALSE(manifest.has_old_rootfs_info());
Don Garrett0dd39852013-04-03 16:55:42 -0700691 EXPECT_FALSE(manifest.has_old_image_info());
692 EXPECT_TRUE(manifest.has_new_image_info());
Darin Petkov7a22d792010-11-08 14:10:00 -0800693 } else {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800694 EXPECT_EQ(state->image_size, manifest.old_rootfs_info().size());
Darin Petkov7a22d792010-11-08 14:10:00 -0800695 EXPECT_FALSE(manifest.old_rootfs_info().hash().empty());
696 }
697
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800698 EXPECT_EQ(state->new_kernel_data.size(), manifest.new_kernel_info().size());
699 EXPECT_EQ(state->image_size, manifest.new_rootfs_info().size());
Darin Petkov36a58222010-10-07 22:00:09 -0700700
Darin Petkov36a58222010-10-07 22:00:09 -0700701 EXPECT_FALSE(manifest.new_kernel_info().hash().empty());
Darin Petkov36a58222010-10-07 22:00:09 -0700702 EXPECT_FALSE(manifest.new_rootfs_info().hash().empty());
Andrew de los Reyes932bc4c2010-08-23 18:14:09 -0700703 }
704
Alex Deymo8427b4a2014-11-05 14:00:32 -0800705 MockPrefs prefs;
Darin Petkov73058b42010-10-06 16:32:19 -0700706 EXPECT_CALL(prefs, SetInt64(kPrefsManifestMetadataSize,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800707 state->metadata_size)).WillOnce(Return(true));
Darin Petkov73058b42010-10-06 16:32:19 -0700708 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextOperation, _))
709 .WillRepeatedly(Return(true));
Darin Petkov9b230572010-10-08 10:20:09 -0700710 EXPECT_CALL(prefs, GetInt64(kPrefsUpdateStateNextOperation, _))
711 .WillOnce(Return(false));
Darin Petkov73058b42010-10-06 16:32:19 -0700712 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextDataOffset, _))
713 .WillRepeatedly(Return(true));
David Zeuthen41996ad2013-09-24 15:43:24 -0700714 EXPECT_CALL(prefs, SetInt64(kPrefsUpdateStateNextDataLength, _))
715 .WillRepeatedly(Return(true));
Darin Petkov437adc42010-10-07 13:12:24 -0700716 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSHA256Context, _))
Darin Petkov73058b42010-10-06 16:32:19 -0700717 .WillRepeatedly(Return(true));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800718 if (op_hash_test == kValidOperationData && signature_test != kSignatureNone) {
Darin Petkovcbfb0172011-01-14 15:24:45 -0800719 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSignedSHA256Context, _))
720 .WillOnce(Return(true));
Darin Petkov4f0a07b2011-05-25 16:47:20 -0700721 EXPECT_CALL(prefs, SetString(kPrefsUpdateStateSignatureBlob, _))
722 .WillOnce(Return(true));
Darin Petkovcbfb0172011-01-14 15:24:45 -0800723 }
Darin Petkov73058b42010-10-06 16:32:19 -0700724
Andrew de los Reyes09e56d62010-04-23 13:45:53 -0700725 // Update the A image in place.
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700726 InstallPlan install_plan;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800727 install_plan.hash_checks_mandatory = hash_checks_mandatory;
728 install_plan.metadata_size = state->metadata_size;
Don Garrettb8dd1d92013-11-22 17:40:02 -0800729 install_plan.is_full_update = full_kernel && full_rootfs;
Allie Woodfdf00512015-03-02 13:34:55 -0800730 install_plan.source_path = state->a_img.c_str();
731 install_plan.kernel_source_path = state->old_kernel.c_str();
Don Garrettb8dd1d92013-11-22 17:40:02 -0800732
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800733 LOG(INFO) << "Setting payload metadata size in Omaha = "
734 << state->metadata_size;
Jay Srinivasanf4318702012-09-24 11:56:24 -0700735 ASSERT_TRUE(PayloadSigner::GetMetadataSignature(
Alex Vakulenko75039d72014-03-25 12:36:28 -0700736 state->delta.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800737 state->metadata_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700738 kUnittestPrivateKeyPath,
Jay Srinivasanf4318702012-09-24 11:56:24 -0700739 &install_plan.metadata_signature));
740 EXPECT_FALSE(install_plan.metadata_signature.empty());
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700741
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800742 *performer = new DeltaPerformer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700743 &state->fake_system_state,
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800744 &install_plan);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700745 EXPECT_TRUE(utils::FileExists(kUnittestPublicKeyPath));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800746 (*performer)->set_public_key_path(kUnittestPublicKeyPath);
Allie Woodfdf00512015-03-02 13:34:55 -0800747 DeltaPerformerTest::SetSupportedVersion(*performer, minor_version);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700748
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800749 EXPECT_EQ(state->image_size,
750 OmahaHashCalculator::RawHashOfFile(state->a_img,
751 state->image_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700752 &install_plan.rootfs_hash));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800753 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(state->old_kernel_data,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700754 &install_plan.kernel_hash));
755
Allie Wood9f6f0a52015-03-30 11:25:47 -0700756 // With minor version 2, we want the target to be the new image, result_img,
757 // but with version 1, we want to update A in place.
758 if (minor_version == kSourceMinorPayloadVersion) {
759 EXPECT_EQ(0, (*performer)->Open(state->result_img.c_str(), 0, 0));
760 EXPECT_TRUE((*performer)->OpenKernel(state->result_kernel.c_str()));
761 } else {
762 EXPECT_EQ(0, (*performer)->Open(state->a_img.c_str(), 0, 0));
763 EXPECT_TRUE((*performer)->OpenKernel(state->old_kernel.c_str()));
764 }
765
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800766
David Zeuthena99981f2013-04-29 13:42:47 -0700767 ErrorCode expected_error, actual_error;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800768 bool continue_writing;
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700769 switch (op_hash_test) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800770 case kInvalidOperationData: {
771 // Muck with some random offset post the metadata size so that
772 // some operation hash will result in a mismatch.
773 int some_offset = state->metadata_size + 300;
774 LOG(INFO) << "Tampered value at offset: " << some_offset;
775 state->delta[some_offset]++;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700776 expected_error = ErrorCode::kDownloadOperationHashMismatch;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800777 continue_writing = false;
778 break;
779 }
780
781 case kValidOperationData:
782 default:
783 // no change.
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700784 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800785 continue_writing = true;
786 break;
787 }
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700788
789 // Write at some number of bytes per operation. Arbitrarily chose 5.
790 const size_t kBytesPerWrite = 5;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800791 for (size_t i = 0; i < state->delta.size(); i += kBytesPerWrite) {
Alex Deymof329b932014-10-30 01:37:48 -0700792 size_t count = std::min(state->delta.size() - i, kBytesPerWrite);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800793 bool write_succeeded = ((*performer)->Write(&state->delta[i],
794 count,
795 &actual_error));
796 // Normally write_succeeded should be true every time and
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700797 // actual_error should be ErrorCode::kSuccess. If so, continue the loop.
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800798 // But if we seeded an operation hash error above, then write_succeeded
799 // will be false. The failure may happen at any operation n. So, all
800 // Writes until n-1 should succeed and the nth operation will fail with
801 // actual_error. In this case, we should bail out of the loop because
802 // we cannot proceed applying the delta.
803 if (!write_succeeded) {
804 LOG(INFO) << "Write failed. Checking if it failed with expected error";
805 EXPECT_EQ(expected_error, actual_error);
806 if (!continue_writing) {
807 LOG(INFO) << "Cannot continue writing. Bailing out.";
808 break;
809 }
810 }
811
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700812 EXPECT_EQ(ErrorCode::kSuccess, actual_error);
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700813 }
814
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800815 // If we had continued all the way through, Close should succeed.
816 // Otherwise, it should fail. Check appropriately.
817 bool close_result = (*performer)->Close();
818 if (continue_writing)
819 EXPECT_EQ(0, close_result);
820 else
821 EXPECT_LE(0, close_result);
822}
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700823
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800824void VerifyPayloadResult(DeltaPerformer* performer,
825 DeltaState* state,
Allie Wood9f6f0a52015-03-30 11:25:47 -0700826 ErrorCode expected_result,
827 uint32_t minor_version) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800828 if (!performer) {
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700829 EXPECT_TRUE(!"Skipping payload verification since performer is null.");
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800830 return;
831 }
832
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700833 int expected_times = (expected_result == ErrorCode::kSuccess) ? 1 : 0;
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700834 EXPECT_CALL(*(state->fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -0800835 DownloadComplete()).Times(expected_times);
836
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800837 LOG(INFO) << "Verifying payload for expected result "
838 << expected_result;
839 EXPECT_EQ(expected_result, performer->VerifyPayload(
840 OmahaHashCalculator::OmahaHashOfData(state->delta),
841 state->delta.size()));
842 LOG(INFO) << "Verified payload.";
843
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700844 if (expected_result != ErrorCode::kSuccess) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800845 // no need to verify new partition if VerifyPayload failed.
846 return;
847 }
848
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800849 chromeos::Blob updated_kernel_partition;
Allie Wood9f6f0a52015-03-30 11:25:47 -0700850 if (minor_version == kSourceMinorPayloadVersion) {
851 CompareFilesByBlock(state->result_kernel, state->new_kernel);
852 CompareFilesByBlock(state->result_img, state->b_img);
853 EXPECT_TRUE(utils::ReadFile(state->result_kernel,
854 &updated_kernel_partition));
855 } else {
856 CompareFilesByBlock(state->old_kernel, state->new_kernel);
857 CompareFilesByBlock(state->a_img, state->b_img);
858 EXPECT_TRUE(utils::ReadFile(state->old_kernel, &updated_kernel_partition));
859 }
860
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800861 ASSERT_GE(updated_kernel_partition.size(), arraysize(kNewData));
862 EXPECT_TRUE(std::equal(std::begin(kNewData), std::end(kNewData),
863 updated_kernel_partition.begin()));
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700864
865 uint64_t new_kernel_size;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800866 chromeos::Blob new_kernel_hash;
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700867 uint64_t new_rootfs_size;
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800868 chromeos::Blob new_rootfs_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800869 EXPECT_TRUE(performer->GetNewPartitionInfo(&new_kernel_size,
Don Garrett0dd39852013-04-03 16:55:42 -0700870 &new_kernel_hash,
871 &new_rootfs_size,
872 &new_rootfs_hash));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800873 EXPECT_EQ(kDefaultKernelSize, new_kernel_size);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800874 chromeos::Blob expected_new_kernel_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800875 EXPECT_TRUE(OmahaHashCalculator::RawHashOfData(state->new_kernel_data,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700876 &expected_new_kernel_hash));
877 EXPECT_TRUE(expected_new_kernel_hash == new_kernel_hash);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800878 EXPECT_EQ(state->image_size, new_rootfs_size);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800879 chromeos::Blob expected_new_rootfs_hash;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800880 EXPECT_EQ(state->image_size,
881 OmahaHashCalculator::RawHashOfFile(state->b_img,
882 state->image_size,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700883 &expected_new_rootfs_hash));
884 EXPECT_TRUE(expected_new_rootfs_hash == new_rootfs_hash);
885}
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800886
887void VerifyPayload(DeltaPerformer* performer,
888 DeltaState* state,
Allie Wood9f6f0a52015-03-30 11:25:47 -0700889 SignatureTest signature_test,
890 uint32_t minor_version) {
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700891 ErrorCode expected_result = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800892 switch (signature_test) {
893 case kSignatureNone:
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700894 expected_result = ErrorCode::kSignedDeltaPayloadExpectedError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800895 break;
896 case kSignatureGeneratedShellBadKey:
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700897 expected_result = ErrorCode::kDownloadPayloadPubKeyVerificationError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800898 break;
899 default: break; // appease gcc
900 }
901
Allie Wood9f6f0a52015-03-30 11:25:47 -0700902 VerifyPayloadResult(performer, state, expected_result, minor_version);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800903}
904
905void DoSmallImageTest(bool full_kernel, bool full_rootfs, bool noop,
Darin Petkov8e447e02013-04-16 16:23:50 +0200906 off_t chunk_size,
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800907 SignatureTest signature_test,
Allie Woodfdf00512015-03-02 13:34:55 -0800908 bool hash_checks_mandatory, uint32_t minor_version) {
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800909 DeltaState state;
Alex Deymo52146ce2014-05-29 11:09:45 -0700910 DeltaPerformer *performer = nullptr;
Darin Petkov8e447e02013-04-16 16:23:50 +0200911 GenerateDeltaFile(full_kernel, full_rootfs, noop, chunk_size,
Allie Woodfdf00512015-03-02 13:34:55 -0800912 signature_test, &state, minor_version);
Don Garrett0dd39852013-04-03 16:55:42 -0700913
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800914 ScopedPathUnlinker a_img_unlinker(state.a_img);
915 ScopedPathUnlinker b_img_unlinker(state.b_img);
Allie Wood9f6f0a52015-03-30 11:25:47 -0700916 ScopedPathUnlinker new_img_unlinker(state.result_img);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800917 ScopedPathUnlinker delta_unlinker(state.delta_path);
918 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
919 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
Allie Wood9f6f0a52015-03-30 11:25:47 -0700920 ScopedPathUnlinker result_kernel_unlinker(state.result_kernel);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800921 ApplyDeltaFile(full_kernel, full_rootfs, noop, signature_test,
922 &state, hash_checks_mandatory, kValidOperationData,
Allie Woodfdf00512015-03-02 13:34:55 -0800923 &performer, minor_version);
Allie Wood9f6f0a52015-03-30 11:25:47 -0700924 VerifyPayload(performer, &state, signature_test, minor_version);
Alex Deymo52146ce2014-05-29 11:09:45 -0700925 delete performer;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800926}
927
928// Calls delta performer's Write method by pretending to pass in bytes from a
929// delta file whose metadata size is actual_metadata_size and tests if all
930// checks are correctly performed if the install plan contains
931// expected_metadata_size and that the result of the parsing are as per
932// hash_checks_mandatory flag.
933void DoMetadataSizeTest(uint64_t expected_metadata_size,
934 uint64_t actual_metadata_size,
935 bool hash_checks_mandatory) {
Alex Deymo8427b4a2014-11-05 14:00:32 -0800936 MockPrefs prefs;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800937 InstallPlan install_plan;
938 install_plan.hash_checks_mandatory = hash_checks_mandatory;
Gilad Arnold5bb4c902014-04-10 12:32:13 -0700939 FakeSystemState fake_system_state;
940 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800941 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
942 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
943
944 // Set a valid magic string and version number 1.
945 EXPECT_TRUE(performer.Write("CrAU", 4));
946 uint64_t version = htobe64(1);
947 EXPECT_TRUE(performer.Write(&version, 8));
948
949 install_plan.metadata_size = expected_metadata_size;
David Zeuthena99981f2013-04-29 13:42:47 -0700950 ErrorCode error_code;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800951 // When filling in size in manifest, exclude the size of the 20-byte header.
952 uint64_t size_in_manifest = htobe64(actual_metadata_size - 20);
953 bool result = performer.Write(&size_in_manifest, 8, &error_code);
954 if (expected_metadata_size == actual_metadata_size ||
955 !hash_checks_mandatory) {
956 EXPECT_TRUE(result);
957 } else {
958 EXPECT_FALSE(result);
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -0700959 EXPECT_EQ(ErrorCode::kDownloadInvalidMetadataSize, error_code);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800960 }
961
962 EXPECT_LT(performer.Close(), 0);
963}
964
965// Generates a valid delta file but tests the delta performer by suppling
966// different metadata signatures as per omaha_metadata_signature flag and
967// sees if the result of the parsing are as per hash_checks_mandatory flag.
968void DoMetadataSignatureTest(MetadataSignatureTest metadata_signature_test,
969 SignatureTest signature_test,
970 bool hash_checks_mandatory) {
971 DeltaState state;
972
973 // Using kSignatureNone since it doesn't affect the results of our test.
974 // If we've to use other signature options, then we'd have to get the
975 // metadata size again after adding the signing operation to the manifest.
Allie Woodfdf00512015-03-02 13:34:55 -0800976 GenerateDeltaFile(true, true, false, -1, signature_test, &state,
977 DeltaPerformer::kFullPayloadMinorVersion);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800978
979 ScopedPathUnlinker a_img_unlinker(state.a_img);
980 ScopedPathUnlinker b_img_unlinker(state.b_img);
981 ScopedPathUnlinker delta_unlinker(state.delta_path);
982 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
983 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
984
985 // Loads the payload and parses the manifest.
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800986 chromeos::Blob payload;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800987 EXPECT_TRUE(utils::ReadFile(state.delta_path, &payload));
988 LOG(INFO) << "Payload size: " << payload.size();
989
990 InstallPlan install_plan;
991 install_plan.hash_checks_mandatory = hash_checks_mandatory;
992 install_plan.metadata_size = state.metadata_size;
993
994 DeltaPerformer::MetadataParseResult expected_result, actual_result;
David Zeuthena99981f2013-04-29 13:42:47 -0700995 ErrorCode expected_error, actual_error;
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800996
997 // Fill up the metadata signature in install plan according to the test.
998 switch (metadata_signature_test) {
999 case kEmptyMetadataSignature:
1000 install_plan.metadata_signature.clear();
1001 expected_result = DeltaPerformer::kMetadataParseError;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001002 expected_error = ErrorCode::kDownloadMetadataSignatureMissingError;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001003 break;
1004
1005 case kInvalidMetadataSignature:
1006 install_plan.metadata_signature = kBogusMetadataSignature1;
1007 expected_result = DeltaPerformer::kMetadataParseError;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001008 expected_error = ErrorCode::kDownloadMetadataSignatureMismatch;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001009 break;
1010
1011 case kValidMetadataSignature:
1012 default:
1013 // Set the install plan's metadata size to be the same as the one
1014 // in the manifest so that we pass the metadata size checks. Only
1015 // then we can get to manifest signature checks.
1016 ASSERT_TRUE(PayloadSigner::GetMetadataSignature(
Alex Vakulenko75039d72014-03-25 12:36:28 -07001017 payload.data(),
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001018 state.metadata_size,
1019 kUnittestPrivateKeyPath,
1020 &install_plan.metadata_signature));
1021 EXPECT_FALSE(install_plan.metadata_signature.empty());
1022 expected_result = DeltaPerformer::kMetadataParseSuccess;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001023 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001024 break;
1025 }
1026
1027 // Ignore the expected result/error if hash checks are not mandatory.
1028 if (!hash_checks_mandatory) {
1029 expected_result = DeltaPerformer::kMetadataParseSuccess;
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001030 expected_error = ErrorCode::kSuccess;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001031 }
1032
1033 // Create the delta performer object.
Alex Deymo8427b4a2014-11-05 14:00:32 -08001034 MockPrefs prefs;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001035 DeltaPerformer delta_performer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001036 &state.fake_system_state,
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001037 &install_plan);
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001038
1039 // Use the public key corresponding to the private key used above to
1040 // sign the metadata.
1041 EXPECT_TRUE(utils::FileExists(kUnittestPublicKeyPath));
1042 delta_performer.set_public_key_path(kUnittestPublicKeyPath);
1043
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001044 // Init actual_error with an invalid value so that we make sure
1045 // ParsePayloadMetadata properly populates it in all cases.
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001046 actual_error = ErrorCode::kUmaReportedMax;
Gilad Arnolddaa27402014-01-23 11:56:17 -08001047 actual_result = delta_performer.ParsePayloadMetadata(payload, &actual_error);
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001048
1049 EXPECT_EQ(expected_result, actual_result);
1050 EXPECT_EQ(expected_error, actual_error);
1051
1052 // Check that the parsed metadata size is what's expected. This test
1053 // implicitly confirms that the metadata signature is valid, if required.
Gilad Arnoldfe133932014-01-14 12:25:50 -08001054 EXPECT_EQ(state.metadata_size, delta_performer.GetMetadataSize());
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001055}
1056
1057void DoOperationHashMismatchTest(OperationHashTest op_hash_test,
1058 bool hash_checks_mandatory) {
1059 DeltaState state;
Allie Woodfdf00512015-03-02 13:34:55 -08001060 uint64_t minor_version = DeltaPerformer::kFullPayloadMinorVersion;
1061 GenerateDeltaFile(true, true, false, -1, kSignatureGenerated, &state,
1062 minor_version);
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001063 ScopedPathUnlinker a_img_unlinker(state.a_img);
1064 ScopedPathUnlinker b_img_unlinker(state.b_img);
1065 ScopedPathUnlinker delta_unlinker(state.delta_path);
1066 ScopedPathUnlinker old_kernel_unlinker(state.old_kernel);
1067 ScopedPathUnlinker new_kernel_unlinker(state.new_kernel);
Alex Deymo52146ce2014-05-29 11:09:45 -07001068 DeltaPerformer *performer = nullptr;
Allie Woodfdf00512015-03-02 13:34:55 -08001069 ApplyDeltaFile(true, true, false, kSignatureGenerated, &state,
1070 hash_checks_mandatory, op_hash_test, &performer,
1071 minor_version);
Alex Deymo52146ce2014-05-29 11:09:45 -07001072 delete performer;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001073}
1074
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001075TEST(DeltaPerformerTest, ExtentsToByteStringTest) {
Allie Wood56873452015-03-27 17:48:40 -07001076 uint64_t test[] = {1, 1, 4, 2, 0, 1};
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001077 COMPILE_ASSERT(arraysize(test) % 2 == 0, array_size_uneven);
1078 const uint64_t block_size = 4096;
Allie Wood56873452015-03-27 17:48:40 -07001079 const uint64_t file_length = 4 * block_size - 13;
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001080
1081 google::protobuf::RepeatedPtrField<Extent> extents;
1082 for (size_t i = 0; i < arraysize(test); i += 2) {
1083 Extent* extent = extents.Add();
1084 extent->set_start_block(test[i]);
1085 extent->set_num_blocks(test[i + 1]);
1086 }
1087
Allie Wood56873452015-03-27 17:48:40 -07001088 string expected_output = "4096:4096,16384:8192,0:4083";
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001089 string actual_output;
1090 EXPECT_TRUE(DeltaPerformer::ExtentsToBsdiffPositionsString(extents,
1091 block_size,
1092 file_length,
1093 &actual_output));
1094 EXPECT_EQ(expected_output, actual_output);
1095}
Darin Petkov68c10d12010-10-14 09:24:37 -07001096
Don Garrettb8dd1d92013-11-22 17:40:02 -08001097TEST(DeltaPerformerTest, ValidateManifestFullGoodTest) {
1098 // The Manifest we are validating.
1099 DeltaArchiveManifest manifest;
1100 manifest.mutable_new_kernel_info();
1101 manifest.mutable_new_rootfs_info();
1102 manifest.set_minor_version(DeltaPerformer::kFullPayloadMinorVersion);
1103
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001104 DeltaPerformerTest::RunManifestValidation(manifest, true,
1105 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001106}
1107
1108TEST(DeltaPerformerTest, ValidateManifestDeltaGoodTest) {
1109 // The Manifest we are validating.
1110 DeltaArchiveManifest manifest;
1111 manifest.mutable_old_kernel_info();
1112 manifest.mutable_old_rootfs_info();
1113 manifest.mutable_new_kernel_info();
1114 manifest.mutable_new_rootfs_info();
1115 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1116
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001117 DeltaPerformerTest::RunManifestValidation(manifest, false,
1118 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001119}
1120
1121TEST(DeltaPerformerTest, ValidateManifestFullUnsetMinorVersion) {
1122 // The Manifest we are validating.
1123 DeltaArchiveManifest manifest;
1124
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001125 DeltaPerformerTest::RunManifestValidation(manifest, true,
1126 ErrorCode::kSuccess);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001127}
1128
1129TEST(DeltaPerformerTest, ValidateManifestDeltaUnsetMinorVersion) {
1130 // The Manifest we are validating.
1131 DeltaArchiveManifest manifest;
1132
1133 DeltaPerformerTest::RunManifestValidation(
1134 manifest, false,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001135 ErrorCode::kUnsupportedMinorPayloadVersion);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001136}
1137
1138TEST(DeltaPerformerTest, ValidateManifestFullOldKernelTest) {
1139 // The Manifest we are validating.
1140 DeltaArchiveManifest manifest;
1141 manifest.mutable_old_kernel_info();
1142 manifest.mutable_new_kernel_info();
1143 manifest.mutable_new_rootfs_info();
1144 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1145
1146 DeltaPerformerTest::RunManifestValidation(
1147 manifest, true,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001148 ErrorCode::kPayloadMismatchedType);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001149}
1150
1151TEST(DeltaPerformerTest, ValidateManifestFullOldRootfsTest) {
1152 // The Manifest we are validating.
1153 DeltaArchiveManifest manifest;
1154 manifest.mutable_old_rootfs_info();
1155 manifest.mutable_new_kernel_info();
1156 manifest.mutable_new_rootfs_info();
1157 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion);
1158
1159 DeltaPerformerTest::RunManifestValidation(
1160 manifest, true,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001161 ErrorCode::kPayloadMismatchedType);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001162}
1163
1164TEST(DeltaPerformerTest, ValidateManifestBadMinorVersion) {
1165 // The Manifest we are validating.
1166 DeltaArchiveManifest manifest;
1167
1168 // Generate a bad version number.
1169 manifest.set_minor_version(DeltaPerformer::kSupportedMinorPayloadVersion +
1170 10000);
1171
1172 DeltaPerformerTest::RunManifestValidation(
1173 manifest, false,
Gilad Arnoldd1c4d2d2014-06-05 14:07:53 -07001174 ErrorCode::kUnsupportedMinorPayloadVersion);
Don Garrettb8dd1d92013-11-22 17:40:02 -08001175}
1176
Darin Petkov68c10d12010-10-14 09:24:37 -07001177TEST(DeltaPerformerTest, RunAsRootSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001178 DoSmallImageTest(false, false, false, -1, kSignatureGenerator,
Allie Woodfdf00512015-03-02 13:34:55 -08001179 false, kInPlaceMinorPayloadVersion);
Darin Petkov8e447e02013-04-16 16:23:50 +02001180}
1181
Don Garrett2ae37872013-10-25 13:33:20 -07001182TEST(DeltaPerformerTest, RunAsRootSmallImageSignaturePlaceholderTest) {
1183 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedPlaceholder,
Allie Woodfdf00512015-03-02 13:34:55 -08001184 false, kInPlaceMinorPayloadVersion);
Don Garrett2ae37872013-10-25 13:33:20 -07001185}
1186
1187TEST(DeltaPerformerTest, RunAsRootSmallImageSignaturePlaceholderMismatchTest) {
1188 DeltaState state;
1189 GenerateDeltaFile(false, false, false, -1,
Allie Woodfdf00512015-03-02 13:34:55 -08001190 kSignatureGeneratedPlaceholderMismatch, &state,
1191 kInPlaceMinorPayloadVersion);
Don Garrett2ae37872013-10-25 13:33:20 -07001192}
1193
Darin Petkov8e447e02013-04-16 16:23:50 +02001194TEST(DeltaPerformerTest, RunAsRootSmallImageChunksTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001195 DoSmallImageTest(false, false, false, kBlockSize, kSignatureGenerator,
Allie Woodfdf00512015-03-02 13:34:55 -08001196 false, kInPlaceMinorPayloadVersion);
Darin Petkov68c10d12010-10-14 09:24:37 -07001197}
1198
1199TEST(DeltaPerformerTest, RunAsRootFullKernelSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001200 DoSmallImageTest(true, false, false, -1, kSignatureGenerator,
Allie Woodfdf00512015-03-02 13:34:55 -08001201 false, kInPlaceMinorPayloadVersion);
Darin Petkov7a22d792010-11-08 14:10:00 -08001202}
1203
1204TEST(DeltaPerformerTest, RunAsRootFullSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001205 DoSmallImageTest(true, true, false, -1, kSignatureGenerator,
Allie Woodfdf00512015-03-02 13:34:55 -08001206 true, DeltaPerformer::kFullPayloadMinorVersion);
Darin Petkov9fa7ec52010-10-18 11:45:23 -07001207}
1208
1209TEST(DeltaPerformerTest, RunAsRootNoopSmallImageTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001210 DoSmallImageTest(false, false, true, -1, kSignatureGenerator,
Allie Woodfdf00512015-03-02 13:34:55 -08001211 false, kInPlaceMinorPayloadVersion);
Darin Petkov9574f7e2011-01-13 10:48:12 -08001212}
1213
Darin Petkovcbfb0172011-01-14 15:24:45 -08001214TEST(DeltaPerformerTest, RunAsRootSmallImageSignNoneTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001215 DoSmallImageTest(false, false, false, -1, kSignatureNone,
Allie Woodfdf00512015-03-02 13:34:55 -08001216 false, kInPlaceMinorPayloadVersion);
Darin Petkovcbfb0172011-01-14 15:24:45 -08001217}
1218
1219TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001220 DoSmallImageTest(false, false, false, -1, kSignatureGenerated,
Allie Woodfdf00512015-03-02 13:34:55 -08001221 true, kInPlaceMinorPayloadVersion);
Darin Petkovcbfb0172011-01-14 15:24:45 -08001222}
1223
1224TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001225 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShell,
Allie Woodfdf00512015-03-02 13:34:55 -08001226 false, kInPlaceMinorPayloadVersion);
Andrew de los Reyes27f7d372010-10-07 11:26:07 -07001227}
1228
Darin Petkov52dcaeb2011-01-14 15:33:06 -08001229TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellBadKeyTest) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001230 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellBadKey,
Allie Woodfdf00512015-03-02 13:34:55 -08001231 false, kInPlaceMinorPayloadVersion);
Darin Petkov52dcaeb2011-01-14 15:33:06 -08001232}
1233
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001234TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellRotateCl1Test) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001235 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellRotateCl1,
Allie Woodfdf00512015-03-02 13:34:55 -08001236 false, kInPlaceMinorPayloadVersion);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001237}
1238
1239TEST(DeltaPerformerTest, RunAsRootSmallImageSignGeneratedShellRotateCl2Test) {
Darin Petkov8e447e02013-04-16 16:23:50 +02001240 DoSmallImageTest(false, false, false, -1, kSignatureGeneratedShellRotateCl2,
Allie Woodfdf00512015-03-02 13:34:55 -08001241 false, kInPlaceMinorPayloadVersion);
Andrew de los Reyesc24e3f32011-08-30 15:45:20 -07001242}
1243
Allie Wood9f6f0a52015-03-30 11:25:47 -07001244TEST(DeltaPerformerTest, RunAsRootSmallImageSourceOpsTest) {
1245 DoSmallImageTest(false, false, false, -1, kSignatureGenerator,
1246 false, kSourceMinorPayloadVersion);
1247}
1248
Darin Petkov934bb412010-11-18 11:21:35 -08001249TEST(DeltaPerformerTest, BadDeltaMagicTest) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001250 MockPrefs prefs;
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001251 InstallPlan install_plan;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001252 FakeSystemState fake_system_state;
1253 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Darin Petkov934bb412010-11-18 11:21:35 -08001254 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
1255 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
Don Garrette410e0f2011-11-10 15:39:01 -08001256 EXPECT_TRUE(performer.Write("junk", 4));
1257 EXPECT_TRUE(performer.Write("morejunk", 8));
1258 EXPECT_FALSE(performer.Write("morejunk", 8));
Darin Petkov934bb412010-11-18 11:21:35 -08001259 EXPECT_LT(performer.Close(), 0);
1260}
1261
Andrew de los Reyes353777c2010-10-08 10:34:30 -07001262TEST(DeltaPerformerTest, IsIdempotentOperationTest) {
1263 DeltaArchiveManifest_InstallOperation op;
1264 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1265 *(op.add_dst_extents()) = ExtentForRange(0, 5);
1266 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1267 *(op.add_src_extents()) = ExtentForRange(4, 1);
1268 EXPECT_FALSE(DeltaPerformer::IsIdempotentOperation(op));
1269 op.clear_src_extents();
1270 *(op.add_src_extents()) = ExtentForRange(5, 3);
1271 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1272 *(op.add_dst_extents()) = ExtentForRange(20, 6);
1273 EXPECT_TRUE(DeltaPerformer::IsIdempotentOperation(op));
1274 *(op.add_src_extents()) = ExtentForRange(19, 2);
1275 EXPECT_FALSE(DeltaPerformer::IsIdempotentOperation(op));
1276}
1277
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001278TEST(DeltaPerformerTest, WriteUpdatesPayloadState) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001279 MockPrefs prefs;
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001280 InstallPlan install_plan;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001281 FakeSystemState fake_system_state;
1282 DeltaPerformer performer(&prefs, &fake_system_state, &install_plan);
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001283 EXPECT_EQ(0, performer.Open("/dev/null", 0, 0));
1284 EXPECT_TRUE(performer.OpenKernel("/dev/null"));
1285
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001286 EXPECT_CALL(*(fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001287 DownloadProgress(4)).Times(1);
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001288 EXPECT_CALL(*(fake_system_state.mock_payload_state()),
Jay Srinivasan2b5a0f02012-12-19 17:25:56 -08001289 DownloadProgress(8)).Times(2);
1290
1291 EXPECT_TRUE(performer.Write("junk", 4));
1292 EXPECT_TRUE(performer.Write("morejunk", 8));
1293 EXPECT_FALSE(performer.Write("morejunk", 8));
1294 EXPECT_LT(performer.Close(), 0);
1295}
1296
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001297TEST(DeltaPerformerTest, MissingMandatoryMetadataSizeTest) {
1298 DoMetadataSizeTest(0, 75456, true);
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001299}
1300
Jay Srinivasan738fdf32012-12-07 17:40:54 -08001301TEST(DeltaPerformerTest, MissingNonMandatoryMetadataSizeTest) {
1302 DoMetadataSizeTest(0, 123456, false);
1303}
1304
1305TEST(DeltaPerformerTest, InvalidMandatoryMetadataSizeTest) {
1306 DoMetadataSizeTest(13000, 140000, true);
1307}
1308
1309TEST(DeltaPerformerTest, InvalidNonMandatoryMetadataSizeTest) {
1310 DoMetadataSizeTest(40000, 50000, false);
1311}
1312
1313TEST(DeltaPerformerTest, ValidMandatoryMetadataSizeTest) {
1314 DoMetadataSizeTest(85376, 85376, true);
1315}
1316
1317TEST(DeltaPerformerTest, RunAsRootMandatoryEmptyMetadataSignatureTest) {
1318 DoMetadataSignatureTest(kEmptyMetadataSignature, kSignatureGenerated, true);
1319}
1320
1321TEST(DeltaPerformerTest, RunAsRootNonMandatoryEmptyMetadataSignatureTest) {
1322 DoMetadataSignatureTest(kEmptyMetadataSignature, kSignatureGenerated, false);
1323}
1324
1325TEST(DeltaPerformerTest, RunAsRootMandatoryInvalidMetadataSignatureTest) {
1326 DoMetadataSignatureTest(kInvalidMetadataSignature, kSignatureGenerated, true);
1327}
1328
1329TEST(DeltaPerformerTest, RunAsRootNonMandatoryInvalidMetadataSignatureTest) {
1330 DoMetadataSignatureTest(kInvalidMetadataSignature, kSignatureGenerated,
1331 false);
1332}
1333
1334TEST(DeltaPerformerTest, RunAsRootMandatoryValidMetadataSignature1Test) {
1335 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureNone, true);
1336}
1337
1338TEST(DeltaPerformerTest, RunAsRootMandatoryValidMetadataSignature2Test) {
1339 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureGenerated, true);
1340}
1341
1342TEST(DeltaPerformerTest, RunAsRootNonMandatoryValidMetadataSignatureTest) {
1343 DoMetadataSignatureTest(kValidMetadataSignature, kSignatureGenerated, false);
1344}
1345
1346TEST(DeltaPerformerTest, RunAsRootMandatoryOperationHashMismatchTest) {
1347 DoOperationHashMismatchTest(kInvalidOperationData, true);
1348}
Jay Srinivasan51dcf262012-09-13 17:24:32 -07001349
David Zeuthene7f89172013-10-31 10:21:04 -07001350TEST(DeltaPerformerTest, UsePublicKeyFromResponse) {
Alex Deymo8427b4a2014-11-05 14:00:32 -08001351 MockPrefs prefs;
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001352 FakeSystemState fake_system_state;
David Zeuthene7f89172013-10-31 10:21:04 -07001353 InstallPlan install_plan;
1354 base::FilePath key_path;
1355
1356 // The result of the GetPublicKeyResponse() method is based on three things
1357 //
1358 // 1. Whether it's an official build; and
1359 // 2. Whether the Public RSA key to be used is in the root filesystem; and
Alex Vakulenko072359c2014-07-18 11:41:07 -07001360 // 3. Whether the response has a public key
David Zeuthene7f89172013-10-31 10:21:04 -07001361 //
1362 // We test all eight combinations to ensure that we only use the
1363 // public key in the response if
1364 //
1365 // a. it's not an official build; and
1366 // b. there is no key in the root filesystem.
1367
1368 DeltaPerformer *performer = new DeltaPerformer(&prefs,
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001369 &fake_system_state,
David Zeuthene7f89172013-10-31 10:21:04 -07001370 &install_plan);
Gilad Arnold5bb4c902014-04-10 12:32:13 -07001371 FakeHardware* fake_hardware = fake_system_state.fake_hardware();
David Zeuthene7f89172013-10-31 10:21:04 -07001372
1373 string temp_dir;
Gilad Arnolda6742b32014-01-11 00:18:34 -08001374 EXPECT_TRUE(utils::MakeTempDirectory("PublicKeyFromResponseTests.XXXXXX",
David Zeuthene7f89172013-10-31 10:21:04 -07001375 &temp_dir));
1376 string non_existing_file = temp_dir + "/non-existing";
1377 string existing_file = temp_dir + "/existing";
Alex Vakulenko75039d72014-03-25 12:36:28 -07001378 EXPECT_EQ(0, System(base::StringPrintf("touch %s", existing_file.c_str())));
David Zeuthene7f89172013-10-31 10:21:04 -07001379
1380 // Non-official build, non-existing public-key, key in response -> true
Don Garrett6646b442013-11-13 15:29:11 -08001381 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001382 performer->public_key_path_ = non_existing_file;
Alex Vakulenkod2779df2014-06-16 13:19:00 -07001383 install_plan.public_key_rsa = "VGVzdAo="; // result of 'echo "Test" | base64'
David Zeuthene7f89172013-10-31 10:21:04 -07001384 EXPECT_TRUE(performer->GetPublicKeyFromResponse(&key_path));
1385 EXPECT_FALSE(key_path.empty());
1386 EXPECT_EQ(unlink(key_path.value().c_str()), 0);
1387 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001388 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001389 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1390
1391 // Non-official build, existing public-key, key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001392 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001393 performer->public_key_path_ = existing_file;
Alex Vakulenkod2779df2014-06-16 13:19:00 -07001394 install_plan.public_key_rsa = "VGVzdAo="; // result of 'echo "Test" | base64'
David Zeuthene7f89172013-10-31 10:21:04 -07001395 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1396 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001397 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001398 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1399
1400 // Non-official build, non-existing public-key, no key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001401 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001402 performer->public_key_path_ = non_existing_file;
1403 install_plan.public_key_rsa = "";
1404 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1405 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001406 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001407 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1408
1409 // Non-official build, existing public-key, no key in response -> false
Don Garrett6646b442013-11-13 15:29:11 -08001410 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001411 performer->public_key_path_ = existing_file;
1412 install_plan.public_key_rsa = "";
1413 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1414 // Same with official build -> false
Don Garrett6646b442013-11-13 15:29:11 -08001415 fake_hardware->SetIsOfficialBuild(true);
David Zeuthene7f89172013-10-31 10:21:04 -07001416 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1417
1418 // Non-official build, non-existing public-key, key in response
1419 // but invalid base64 -> false
Don Garrett6646b442013-11-13 15:29:11 -08001420 fake_hardware->SetIsOfficialBuild(false);
David Zeuthene7f89172013-10-31 10:21:04 -07001421 performer->public_key_path_ = non_existing_file;
1422 install_plan.public_key_rsa = "not-valid-base64";
1423 EXPECT_FALSE(performer->GetPublicKeyFromResponse(&key_path));
1424
1425 delete performer;
Alex Deymo10875d92014-11-10 21:52:57 -08001426 EXPECT_TRUE(test_utils::RecursiveUnlinkDir(temp_dir));
David Zeuthene7f89172013-10-31 10:21:04 -07001427}
1428
Allie Wood78750a42015-02-11 15:42:11 -08001429TEST(DeltaPerformerTest, MinorVersionsMatch) {
1430 // Test that the minor version in update_engine.conf that is installed to
1431 // the image matches the supported delta minor version in the update engine.
1432 uint32_t minor_version;
1433 base::FilePath conf_path("update_engine.conf");
1434 EXPECT_TRUE(utils::GetMinorVersion(conf_path, &minor_version));
1435 ASSERT_EQ(DeltaPerformer::kSupportedMinorPayloadVersion, minor_version);
1436}
1437
Andrew de los Reyes09e56d62010-04-23 13:45:53 -07001438} // namespace chromeos_update_engine