blob: 7105d3a5da036c16ece583495d305667b88360ab [file] [log] [blame]
Mike Frysinger8155d082012-04-06 15:23:18 -04001// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
adlr@google.com3defe6a2009-12-04 20:57:17 +00002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include <sys/stat.h>
6#include <sys/types.h>
7#include <errno.h>
Darin Petkov5c0a8af2010-08-24 13:39:13 -07008
Andrew de los Reyescc92cd32010-10-05 16:56:14 -07009#include <map>
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -080010#include <string>
adlr@google.com3defe6a2009-12-04 20:57:17 +000011#include <vector>
Darin Petkov5c0a8af2010-08-24 13:39:13 -070012
Alex Vakulenko75039d72014-03-25 12:36:28 -070013#include <base/files/file_path.h>
Darin Petkov8e447e02013-04-16 16:23:50 +020014#include <base/file_util.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070015#include <base/strings/string_util.h>
16#include <base/strings/stringprintf.h>
Darin Petkovd3f8c892010-10-12 21:38:45 -070017#include <gtest/gtest.h>
18
David Zeuthen33bae492014-02-25 16:16:18 -080019#include "update_engine/fake_clock.h"
20#include "update_engine/mock_system_state.h"
21#include "update_engine/prefs.h"
Darin Petkovd3f8c892010-10-12 21:38:45 -070022#include "update_engine/test_utils.h"
adlr@google.com3defe6a2009-12-04 20:57:17 +000023#include "update_engine/utils.h"
24
Andrew de los Reyescc92cd32010-10-05 16:56:14 -070025using std::map;
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -080026using std::string;
adlr@google.com3defe6a2009-12-04 20:57:17 +000027using std::vector;
28
29namespace chromeos_update_engine {
30
31class UtilsTest : public ::testing::Test { };
32
Chris Sosac1972482013-04-30 22:31:10 -070033TEST(UtilsTest, CanParseECVersion) {
Chris Sosac1972482013-04-30 22:31:10 -070034 // Should be able to parse and valid key value line.
J. Richard Barnette63137e52013-10-28 10:57:29 -070035 EXPECT_EQ("12345", utils::ParseECVersion("fw_version=12345"));
36 EXPECT_EQ("123456", utils::ParseECVersion(
37 "b=1231a fw_version=123456 a=fasd2"));
38 EXPECT_EQ("12345", utils::ParseECVersion("fw_version=12345"));
39 EXPECT_EQ("00VFA616", utils::ParseECVersion(
Chris Sosac1972482013-04-30 22:31:10 -070040 "vendor=\"sam\" fw_version=\"00VFA616\""));
41
42 // For invalid entries, should return the empty string.
J. Richard Barnette63137e52013-10-28 10:57:29 -070043 EXPECT_EQ("", utils::ParseECVersion("b=1231a fw_version a=fasd2"));
Chris Sosac1972482013-04-30 22:31:10 -070044}
45
J. Richard Barnette30842932013-10-28 15:04:23 -070046
47TEST(UtilsTest, KernelDeviceOfBootDevice) {
48 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("foo"));
49 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda0"));
50 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda1"));
51 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda2"));
52 EXPECT_EQ("/dev/sda2", utils::KernelDeviceOfBootDevice("/dev/sda3"));
53 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda4"));
54 EXPECT_EQ("/dev/sda4", utils::KernelDeviceOfBootDevice("/dev/sda5"));
55 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda6"));
56 EXPECT_EQ("/dev/sda6", utils::KernelDeviceOfBootDevice("/dev/sda7"));
57 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda8"));
58 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/sda9"));
59
60 EXPECT_EQ("/dev/mmcblk0p2",
61 utils::KernelDeviceOfBootDevice("/dev/mmcblk0p3"));
62 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/mmcblk0p4"));
63
64 EXPECT_EQ("/dev/ubi2", utils::KernelDeviceOfBootDevice("/dev/ubi3"));
65 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/ubi4"));
66
67 EXPECT_EQ("/dev/mtdblock2",
68 utils::KernelDeviceOfBootDevice("/dev/ubiblock3_0"));
69 EXPECT_EQ("/dev/mtdblock4",
70 utils::KernelDeviceOfBootDevice("/dev/ubiblock5_0"));
71 EXPECT_EQ("/dev/mtdblock6",
72 utils::KernelDeviceOfBootDevice("/dev/ubiblock7_0"));
73 EXPECT_EQ("", utils::KernelDeviceOfBootDevice("/dev/ubiblock4_0"));
74}
75
76
adlr@google.com3defe6a2009-12-04 20:57:17 +000077TEST(UtilsTest, NormalizePathTest) {
78 EXPECT_EQ("", utils::NormalizePath("", false));
79 EXPECT_EQ("", utils::NormalizePath("", true));
80 EXPECT_EQ("/", utils::NormalizePath("/", false));
81 EXPECT_EQ("", utils::NormalizePath("/", true));
82 EXPECT_EQ("/", utils::NormalizePath("//", false));
83 EXPECT_EQ("", utils::NormalizePath("//", true));
84 EXPECT_EQ("foo", utils::NormalizePath("foo", false));
85 EXPECT_EQ("foo", utils::NormalizePath("foo", true));
86 EXPECT_EQ("/foo/", utils::NormalizePath("/foo//", false));
87 EXPECT_EQ("/foo", utils::NormalizePath("/foo//", true));
88 EXPECT_EQ("bar/baz/foo/adlr", utils::NormalizePath("bar/baz//foo/adlr",
89 false));
90 EXPECT_EQ("bar/baz/foo/adlr", utils::NormalizePath("bar/baz//foo/adlr",
91 true));
92 EXPECT_EQ("/bar/baz/foo/adlr/", utils::NormalizePath("/bar/baz//foo/adlr/",
93 false));
94 EXPECT_EQ("/bar/baz/foo/adlr", utils::NormalizePath("/bar/baz//foo/adlr/",
95 true));
96 EXPECT_EQ("\\\\", utils::NormalizePath("\\\\", false));
97 EXPECT_EQ("\\\\", utils::NormalizePath("\\\\", true));
98 EXPECT_EQ("\\:/;$PATH\n\\", utils::NormalizePath("\\://;$PATH\n\\", false));
99 EXPECT_EQ("\\:/;$PATH\n\\", utils::NormalizePath("\\://;$PATH\n\\", true));
100 EXPECT_EQ("/spaces s/ ok/s / / /",
101 utils::NormalizePath("/spaces s/ ok/s / / /", false));
102 EXPECT_EQ("/spaces s/ ok/s / / ",
103 utils::NormalizePath("/spaces s/ ok/s / / /", true));
104}
105
106TEST(UtilsTest, ReadFileFailure) {
107 vector<char> empty;
108 EXPECT_FALSE(utils::ReadFile("/this/doesn't/exist", &empty));
109}
110
Darin Petkov8e447e02013-04-16 16:23:50 +0200111TEST(UtilsTest, ReadFileChunk) {
Alex Vakulenko75039d72014-03-25 12:36:28 -0700112 base::FilePath file;
113 EXPECT_TRUE(base::CreateTemporaryFile(&file));
Darin Petkov8e447e02013-04-16 16:23:50 +0200114 ScopedPathUnlinker unlinker(file.value());
115 vector<char> data;
116 const size_t kSize = 1024 * 1024;
117 for (size_t i = 0; i < kSize; i++) {
118 data.push_back(i % 255);
119 }
120 EXPECT_TRUE(utils::WriteFile(file.value().c_str(), &data[0], data.size()));
121 vector<char> in_data;
122 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), kSize, 10, &in_data));
123 EXPECT_TRUE(in_data.empty());
124 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), 0, -1, &in_data));
125 EXPECT_TRUE(data == in_data);
126 in_data.clear();
127 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), 10, 20, &in_data));
128 EXPECT_TRUE(vector<char>(data.begin() + 10, data.begin() + 10 + 20) ==
129 in_data);
130}
131
adlr@google.com3defe6a2009-12-04 20:57:17 +0000132TEST(UtilsTest, ErrnoNumberAsStringTest) {
133 EXPECT_EQ("No such file or directory", utils::ErrnoNumberAsString(ENOENT));
134}
135
136TEST(UtilsTest, StringHasSuffixTest) {
137 EXPECT_TRUE(utils::StringHasSuffix("foo", "foo"));
138 EXPECT_TRUE(utils::StringHasSuffix("foo", "o"));
139 EXPECT_TRUE(utils::StringHasSuffix("", ""));
140 EXPECT_TRUE(utils::StringHasSuffix("abcabc", "abc"));
141 EXPECT_TRUE(utils::StringHasSuffix("adlrwashere", "ere"));
142 EXPECT_TRUE(utils::StringHasSuffix("abcdefgh", "gh"));
143 EXPECT_TRUE(utils::StringHasSuffix("abcdefgh", ""));
144 EXPECT_FALSE(utils::StringHasSuffix("foo", "afoo"));
145 EXPECT_FALSE(utils::StringHasSuffix("", "x"));
146 EXPECT_FALSE(utils::StringHasSuffix("abcdefgh", "fg"));
147 EXPECT_FALSE(utils::StringHasSuffix("abcdefgh", "ab"));
148}
149
150TEST(UtilsTest, StringHasPrefixTest) {
151 EXPECT_TRUE(utils::StringHasPrefix("foo", "foo"));
152 EXPECT_TRUE(utils::StringHasPrefix("foo", "f"));
153 EXPECT_TRUE(utils::StringHasPrefix("", ""));
154 EXPECT_TRUE(utils::StringHasPrefix("abcabc", "abc"));
155 EXPECT_TRUE(utils::StringHasPrefix("adlrwashere", "adl"));
156 EXPECT_TRUE(utils::StringHasPrefix("abcdefgh", "ab"));
157 EXPECT_TRUE(utils::StringHasPrefix("abcdefgh", ""));
158 EXPECT_FALSE(utils::StringHasPrefix("foo", "fooa"));
159 EXPECT_FALSE(utils::StringHasPrefix("", "x"));
160 EXPECT_FALSE(utils::StringHasPrefix("abcdefgh", "bc"));
161 EXPECT_FALSE(utils::StringHasPrefix("abcdefgh", "gh"));
162}
163
adlr@google.com3defe6a2009-12-04 20:57:17 +0000164TEST(UtilsTest, RecursiveUnlinkDirTest) {
Gilad Arnoldb4346a22013-07-16 06:44:45 -0700165 string first_dir_name;
166 ASSERT_TRUE(utils::MakeTempDirectory("RecursiveUnlinkDirTest-a-XXXXXX",
167 &first_dir_name));
168 ASSERT_EQ(0, Chmod(first_dir_name, 0755));
169 string second_dir_name;
170 ASSERT_TRUE(utils::MakeTempDirectory("RecursiveUnlinkDirTest-b-XXXXXX",
171 &second_dir_name));
172 ASSERT_EQ(0, Chmod(second_dir_name, 0755));
173
174 EXPECT_EQ(0, Symlink(string("../") + first_dir_name,
175 second_dir_name + "/link"));
176 EXPECT_EQ(0, System(string("echo hi > ") + second_dir_name + "/file"));
177 EXPECT_EQ(0, Mkdir(second_dir_name + "/dir", 0755));
178 EXPECT_EQ(0, System(string("echo ok > ") + second_dir_name + "/dir/subfile"));
179 EXPECT_TRUE(utils::RecursiveUnlinkDir(second_dir_name));
180 EXPECT_TRUE(utils::FileExists(first_dir_name.c_str()));
181 EXPECT_EQ(0, System(string("rm -rf ") + first_dir_name));
182 EXPECT_FALSE(utils::FileExists(second_dir_name.c_str()));
adlr@google.com3defe6a2009-12-04 20:57:17 +0000183 EXPECT_TRUE(utils::RecursiveUnlinkDir("/something/that/doesnt/exist"));
184}
185
Darin Petkov002b2fe2010-11-22 13:53:22 -0800186TEST(UtilsTest, IsSymlinkTest) {
187 string temp_dir;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800188 EXPECT_TRUE(utils::MakeTempDirectory("symlink-test.XXXXXX", &temp_dir));
Darin Petkov002b2fe2010-11-22 13:53:22 -0800189 string temp_file = temp_dir + "temp-file";
190 EXPECT_TRUE(utils::WriteFile(temp_file.c_str(), "", 0));
191 string temp_symlink = temp_dir + "temp-symlink";
192 EXPECT_EQ(0, symlink(temp_file.c_str(), temp_symlink.c_str()));
193 EXPECT_FALSE(utils::IsSymlink(temp_dir.c_str()));
194 EXPECT_FALSE(utils::IsSymlink(temp_file.c_str()));
195 EXPECT_TRUE(utils::IsSymlink(temp_symlink.c_str()));
196 EXPECT_FALSE(utils::IsSymlink("/non/existent/path"));
197 EXPECT_TRUE(utils::RecursiveUnlinkDir(temp_dir));
198}
199
adlr@google.com3defe6a2009-12-04 20:57:17 +0000200TEST(UtilsTest, TempFilenameTest) {
201 const string original = "/foo.XXXXXX";
202 const string result = utils::TempFilename(original);
203 EXPECT_EQ(original.size(), result.size());
204 EXPECT_TRUE(utils::StringHasPrefix(result, "/foo."));
205 EXPECT_FALSE(utils::StringHasSuffix(result, "XXXXXX"));
206}
207
Alex Vakulenko4f5b1442014-02-21 12:19:44 -0800208TEST(UtilsTest, GetDiskNameTest) {
209 EXPECT_EQ("/dev/sda", utils::GetDiskName("/dev/sda3"));
210 EXPECT_EQ("/dev/sda", utils::GetDiskName("/dev/sda1234"));
211 EXPECT_EQ("/dev/mmcblk0", utils::GetDiskName("/dev/mmcblk0p3"));
212 EXPECT_EQ("", utils::GetDiskName("/dev/mmcblk0p"));
213 EXPECT_EQ("", utils::GetDiskName("/dev/sda"));
214 EXPECT_EQ("", utils::GetDiskName("/dev/ubiblock3_2"));
215 EXPECT_EQ("", utils::GetDiskName("/dev/foo/bar"));
216 EXPECT_EQ("", utils::GetDiskName("/"));
217 EXPECT_EQ("", utils::GetDiskName(""));
Darin Petkovf74eb652010-08-04 12:08:38 -0700218}
219
220TEST(UtilsTest, SysfsBlockDeviceTest) {
221 EXPECT_EQ("/sys/block/sda", utils::SysfsBlockDevice("/dev/sda"));
222 EXPECT_EQ("", utils::SysfsBlockDevice("/foo/sda"));
223 EXPECT_EQ("", utils::SysfsBlockDevice("/dev/foo/bar"));
224 EXPECT_EQ("", utils::SysfsBlockDevice("/"));
225 EXPECT_EQ("", utils::SysfsBlockDevice("./"));
226 EXPECT_EQ("", utils::SysfsBlockDevice(""));
227}
228
229TEST(UtilsTest, IsRemovableDeviceTest) {
230 EXPECT_FALSE(utils::IsRemovableDevice(""));
231 EXPECT_FALSE(utils::IsRemovableDevice("/dev/non-existent-device"));
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700232}
233
Alex Vakulenko4f5b1442014-02-21 12:19:44 -0800234TEST(UtilsTest, GetPartitionNumberTest) {
235 EXPECT_EQ(3, utils::GetPartitionNumber("/dev/sda3"));
236 EXPECT_EQ(123, utils::GetPartitionNumber("/dev/sda123"));
237 EXPECT_EQ(2, utils::GetPartitionNumber("/dev/mmcblk0p2"));
238 EXPECT_EQ(0, utils::GetPartitionNumber("/dev/mmcblk0p"));
239 EXPECT_EQ(0, utils::GetPartitionNumber("/dev/ubiblock3_2"));
240 EXPECT_EQ(0, utils::GetPartitionNumber(""));
241 EXPECT_EQ(0, utils::GetPartitionNumber("/"));
242 EXPECT_EQ(0, utils::GetPartitionNumber("/dev/"));
243 EXPECT_EQ(0, utils::GetPartitionNumber("/dev/sda"));
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700244}
245
Chris Sosa4f8ee272012-11-30 13:01:54 -0800246TEST(UtilsTest, CompareCpuSharesTest) {
247 EXPECT_LT(utils::CompareCpuShares(utils::kCpuSharesLow,
248 utils::kCpuSharesNormal), 0);
249 EXPECT_GT(utils::CompareCpuShares(utils::kCpuSharesNormal,
250 utils::kCpuSharesLow), 0);
251 EXPECT_EQ(utils::CompareCpuShares(utils::kCpuSharesNormal,
252 utils::kCpuSharesNormal), 0);
253 EXPECT_GT(utils::CompareCpuShares(utils::kCpuSharesHigh,
254 utils::kCpuSharesNormal), 0);
Darin Petkovc6c135c2010-08-11 13:36:18 -0700255}
256
Darin Petkov5c0a8af2010-08-24 13:39:13 -0700257TEST(UtilsTest, FuzzIntTest) {
258 static const unsigned int kRanges[] = { 0, 1, 2, 20 };
259 for (size_t r = 0; r < arraysize(kRanges); ++r) {
260 unsigned int range = kRanges[r];
261 const int kValue = 50;
262 for (int tries = 0; tries < 100; ++tries) {
263 int value = utils::FuzzInt(kValue, range);
264 EXPECT_GE(value, kValue - range / 2);
265 EXPECT_LE(value, kValue + range - range / 2);
266 }
267 }
268}
269
Andrew de los Reyescc92cd32010-10-05 16:56:14 -0700270TEST(UtilsTest, ApplyMapTest) {
271 int initial_values[] = {1, 2, 3, 4, 6};
272 vector<int> collection(&initial_values[0],
273 initial_values + arraysize(initial_values));
274 EXPECT_EQ(arraysize(initial_values), collection.size());
275 int expected_values[] = {1, 2, 5, 4, 8};
276 map<int, int> value_map;
277 value_map[3] = 5;
278 value_map[6] = 8;
279 value_map[5] = 10;
280
281 utils::ApplyMap(&collection, value_map);
282
283 size_t index = 0;
284 for (vector<int>::iterator it = collection.begin(), e = collection.end();
285 it != e; ++it) {
286 EXPECT_EQ(expected_values[index++], *it);
287 }
288}
289
Darin Petkovd3f8c892010-10-12 21:38:45 -0700290TEST(UtilsTest, RunAsRootGetFilesystemSizeTest) {
291 string img;
Gilad Arnolda6742b32014-01-11 00:18:34 -0800292 EXPECT_TRUE(utils::MakeTempFile("img.XXXXXX", &img, NULL));
Darin Petkovd3f8c892010-10-12 21:38:45 -0700293 ScopedPathUnlinker img_unlinker(img);
294 CreateExtImageAtPath(img, NULL);
295 // Extend the "partition" holding the file system from 10MiB to 20MiB.
296 EXPECT_EQ(0, System(base::StringPrintf(
297 "dd if=/dev/zero of=%s seek=20971519 bs=1 count=1",
298 img.c_str())));
299 EXPECT_EQ(20 * 1024 * 1024, utils::FileSize(img));
300 int block_count = 0;
301 int block_size = 0;
302 EXPECT_TRUE(utils::GetFilesystemSize(img, &block_count, &block_size));
303 EXPECT_EQ(4096, block_size);
304 EXPECT_EQ(10 * 1024 * 1024 / 4096, block_count);
305}
306
Chris Sosad317e402013-06-12 13:47:09 -0700307TEST(UtilsTest, GetInstallDevTest) {
308 string boot_dev = "/dev/sda5";
309 string install_dev;
310 EXPECT_TRUE(utils::GetInstallDev(boot_dev, &install_dev));
311 EXPECT_EQ(install_dev, "/dev/sda3");
312
313 boot_dev = "/dev/sda3";
314 EXPECT_TRUE(utils::GetInstallDev(boot_dev, &install_dev));
315 EXPECT_EQ(install_dev, "/dev/sda5");
316
317 boot_dev = "/dev/sda12";
318 EXPECT_FALSE(utils::GetInstallDev(boot_dev, &install_dev));
Liam McLoughlin049d1652013-07-31 18:47:46 -0700319
320 boot_dev = "/dev/ubiblock3_0";
321 EXPECT_TRUE(utils::GetInstallDev(boot_dev, &install_dev));
322 EXPECT_EQ(install_dev, "/dev/ubiblock5_0");
323
324 boot_dev = "/dev/ubiblock5_0";
325 EXPECT_TRUE(utils::GetInstallDev(boot_dev, &install_dev));
326 EXPECT_EQ(install_dev, "/dev/ubiblock3_0");
327
328 boot_dev = "/dev/ubiblock12_0";
329 EXPECT_FALSE(utils::GetInstallDev(boot_dev, &install_dev));
Chris Sosad317e402013-06-12 13:47:09 -0700330}
331
Andrew de los Reyes712b3ac2011-01-07 13:47:52 -0800332namespace {
Alex Deymo032e7722014-03-25 17:53:56 -0700333void GetFileFormatTester(const string& expected,
334 const vector<uint8>& contents) {
335 ScopedTempFile file;
336 ASSERT_TRUE(utils::WriteFile(file.GetPath().c_str(),
337 reinterpret_cast<const char*>(contents.data()),
338 contents.size()));
339 EXPECT_EQ(expected, utils::GetFileFormat(file.GetPath()));
340}
341}
342
343TEST(UtilsTest, GetFileFormatTest) {
344 EXPECT_EQ("File not found.", utils::GetFileFormat("/path/to/nowhere"));
345 GetFileFormatTester("data", vector<uint8>{1, 2, 3, 4, 5, 6, 7, 8});
346 GetFileFormatTester("ELF", vector<uint8>{0x7f, 0x45, 0x4c, 0x46});
347
348 // Real tests from cros_installer on different boards.
349 // ELF 32-bit LSB executable, Intel 80386
350 GetFileFormatTester(
351 "ELF 32-bit little-endian x86",
352 vector<uint8>{0x7f, 0x45, 0x4c, 0x46, 0x01, 0x01, 0x01, 0x00,
353 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
354 0x02, 0x00, 0x03, 0x00, 0x01, 0x00, 0x00, 0x00,
355 0x90, 0x83, 0x04, 0x08, 0x34, 0x00, 0x00, 0x00});
356
357 // ELF 32-bit LSB executable, ARM
358 GetFileFormatTester(
359 "ELF 32-bit little-endian arm",
360 vector<uint8>{0x7f, 0x45, 0x4c, 0x46, 0x01, 0x01, 0x01, 0x00,
361 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
362 0x02, 0x00, 0x28, 0x00, 0x01, 0x00, 0x00, 0x00,
363 0x85, 0x8b, 0x00, 0x00, 0x34, 0x00, 0x00, 0x00});
364
365 // ELF 64-bit LSB executable, x86-64
366 GetFileFormatTester(
367 "ELF 64-bit little-endian x86-64",
368 vector<uint8>{0x7f, 0x45, 0x4c, 0x46, 0x02, 0x01, 0x01, 0x00,
369 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
370 0x02, 0x00, 0x3e, 0x00, 0x01, 0x00, 0x00, 0x00,
371 0xb0, 0x04, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00});
372}
373
374namespace {
Andrew de los Reyes712b3ac2011-01-07 13:47:52 -0800375gboolean TerminateScheduleCrashReporterUploadTest(void* arg) {
376 GMainLoop* loop = reinterpret_cast<GMainLoop*>(arg);
377 g_main_loop_quit(loop);
378 return FALSE; // Don't call this callback again
379}
380} // namespace {}
381
382TEST(UtilsTest, ScheduleCrashReporterUploadTest) {
383 // Not much to test. At least this tests for memory leaks, crashes,
384 // log errors.
385 GMainLoop* loop = g_main_loop_new(g_main_context_default(), FALSE);
386 utils::ScheduleCrashReporterUpload();
387 g_timeout_add_seconds(1, &TerminateScheduleCrashReporterUploadTest, loop);
388 g_main_loop_run(loop);
389 g_main_loop_unref(loop);
390}
391
David Zeuthen674c3182013-04-18 14:05:20 -0700392TEST(UtilsTest, FormatTimeDeltaTest) {
393 // utils::FormatTimeDelta() is not locale-aware (it's only used for logging
394 // which is not localized) so we only need to test the C locale
395 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromMilliseconds(100)),
396 "0.1s");
397 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(0)),
398 "0s");
399 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1)),
400 "1s");
401 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(59)),
402 "59s");
403 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(60)),
404 "1m0s");
405 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(61)),
406 "1m1s");
407 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(90)),
408 "1m30s");
409 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1205)),
410 "20m5s");
411 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3600)),
412 "1h0m0s");
413 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3601)),
414 "1h0m1s");
415 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3661)),
416 "1h1m1s");
417 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(7261)),
418 "2h1m1s");
419 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86400)),
420 "1d0h0m0s");
421 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86401)),
422 "1d0h0m1s");
423 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000)),
424 "2d7h33m20s");
425 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000) +
426 base::TimeDelta::FromMilliseconds(1)),
427 "2d7h33m20.001s");
428}
429
David Zeuthen27a48bc2013-08-06 12:06:29 -0700430TEST(UtilsTest, TimeFromStructTimespecTest) {
431 struct timespec ts;
432
433 // Unix epoch (Thursday 00:00:00 UTC on Jan 1, 1970)
434 ts = (struct timespec) {.tv_sec = 0, .tv_nsec = 0};
435 EXPECT_EQ(base::Time::UnixEpoch(), utils::TimeFromStructTimespec(&ts));
436
437 // 42 ms after the Unix billennium (Sunday 01:46:40 UTC on September 9, 2001)
438 ts = (struct timespec) {.tv_sec = 1000 * 1000 * 1000,
439 .tv_nsec = 42 * 1000 * 1000};
440 base::Time::Exploded exploded = (base::Time::Exploded) {
441 .year = 2001, .month = 9, .day_of_week = 0, .day_of_month = 9,
442 .hour = 1, .minute = 46, .second = 40, .millisecond = 42};
443 EXPECT_EQ(base::Time::FromUTCExploded(exploded),
444 utils::TimeFromStructTimespec(&ts));
445}
446
David Zeuthene7f89172013-10-31 10:21:04 -0700447TEST(UtilsTest, DecodeAndStoreBase64String) {
448 base::FilePath path;
449
450 // Ensure we return false on empty strings or invalid base64.
451 EXPECT_FALSE(utils::DecodeAndStoreBase64String("", &path));
452 EXPECT_FALSE(utils::DecodeAndStoreBase64String("not valid base64", &path));
453
454 // Pass known base64 and check that it matches. This string was generated
455 // the following way:
456 //
457 // $ echo "Update Engine" | base64
458 // VXBkYXRlIEVuZ2luZQo=
459 EXPECT_TRUE(utils::DecodeAndStoreBase64String("VXBkYXRlIEVuZ2luZQo=",
460 &path));
461 ScopedPathUnlinker unlinker(path.value());
462 string expected_contents = "Update Engine\n";
463 string contents;
464 EXPECT_TRUE(utils::ReadFile(path.value(), &contents));
465 EXPECT_EQ(contents, expected_contents);
466 EXPECT_EQ(utils::FileSize(path.value()), expected_contents.size());
467}
468
David Zeuthen639aa362014-02-03 16:23:44 -0800469TEST(UtilsTest, ConvertToOmahaInstallDate) {
470 // The Omaha Epoch starts at Jan 1, 2007 0:00 PST which is a
471 // Monday. In Unix time, this point in time is easily obtained via
472 // the date(1) command like this:
473 //
474 // $ date +"%s" --date="Jan 1, 2007 0:00 PST"
475 const time_t omaha_epoch = 1167638400;
476 int value;
477
478 // Points in time *on and after* the Omaha epoch should not fail.
479 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
480 base::Time::FromTimeT(omaha_epoch), &value));
481 EXPECT_GE(value, 0);
482
483 // Anything before the Omaha epoch should fail. We test it for two points.
484 EXPECT_FALSE(utils::ConvertToOmahaInstallDate(
485 base::Time::FromTimeT(omaha_epoch - 1), &value));
486 EXPECT_FALSE(utils::ConvertToOmahaInstallDate(
487 base::Time::FromTimeT(omaha_epoch - 100*24*3600), &value));
488
489 // Check that we jump from 0 to 7 exactly on the one-week mark, e.g.
490 // on Jan 8, 2007 0:00 PST.
491 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
492 base::Time::FromTimeT(omaha_epoch + 7*24*3600 - 1), &value));
493 EXPECT_EQ(value, 0);
494 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
495 base::Time::FromTimeT(omaha_epoch + 7*24*3600), &value));
496 EXPECT_EQ(value, 7);
497
498 // Check a couple of more values.
499 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
500 base::Time::FromTimeT(omaha_epoch + 10*24*3600), &value));
501 EXPECT_EQ(value, 7);
502 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
503 base::Time::FromTimeT(omaha_epoch + 20*24*3600), &value));
504 EXPECT_EQ(value, 14);
505 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
506 base::Time::FromTimeT(omaha_epoch + 26*24*3600), &value));
507 EXPECT_EQ(value, 21);
508 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
509 base::Time::FromTimeT(omaha_epoch + 29*24*3600), &value));
510 EXPECT_EQ(value, 28);
511
512 // The date Jun 4, 2007 0:00 PDT is a Monday and is hence a point
513 // where the Omaha InstallDate jumps 7 days. Its unix time is
514 // 1180940400. Notably, this is a point in time where Daylight
515 // Savings Time (DST) was is in effect (e.g. it's PDT, not PST).
516 //
517 // Note that as utils::ConvertToOmahaInstallDate() _deliberately_
518 // ignores DST (as it's hard to implement in a thread-safe way using
519 // glibc, see comments in utils.h) we have to fudge by the DST
520 // offset which is one hour. Conveniently, if the function were
521 // someday modified to be DST aware, this test would have to be
522 // modified as well.
523 const time_t dst_time = 1180940400; // Jun 4, 2007 0:00 PDT.
524 const time_t fudge = 3600;
525 int value1, value2;
526 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
527 base::Time::FromTimeT(dst_time + fudge - 1), &value1));
528 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
529 base::Time::FromTimeT(dst_time + fudge), &value2));
530 EXPECT_EQ(value1, value2 - 7);
531}
532
David Zeuthen33bae492014-02-25 16:16:18 -0800533TEST(UtilsTest, WallclockDurationHelper) {
534 MockSystemState mock_system_state;
535 FakeClock fake_clock;
536 base::TimeDelta duration;
537 string state_variable_key = "test-prefs";
538 string temp_dir;
539 Prefs fake_prefs;
540
541 EXPECT_TRUE(utils::MakeTempDirectory("DurationPrefs.XXXXXX", &temp_dir));
542 fake_prefs.Init(base::FilePath(temp_dir));
543
544 mock_system_state.set_clock(&fake_clock);
545 mock_system_state.set_prefs(&fake_prefs);
546
547 // Initialize wallclock to 1 sec.
548 fake_clock.SetWallclockTime(base::Time::FromInternalValue(1000000));
549
550 // First time called so no previous measurement available.
551 EXPECT_FALSE(utils::WallclockDurationHelper(&mock_system_state,
552 state_variable_key,
553 &duration));
554
555 // Next time, we should get zero since the clock didn't advance.
556 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
557 state_variable_key,
558 &duration));
559 EXPECT_EQ(duration.InSeconds(), 0);
560
561 // We can also call it as many times as we want with it being
562 // considered a failure.
563 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
564 state_variable_key,
565 &duration));
566 EXPECT_EQ(duration.InSeconds(), 0);
567 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
568 state_variable_key,
569 &duration));
570 EXPECT_EQ(duration.InSeconds(), 0);
571
572 // Advance the clock one second, then we should get 1 sec on the
573 // next call and 0 sec on the subsequent call.
574 fake_clock.SetWallclockTime(base::Time::FromInternalValue(2000000));
575 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
576 state_variable_key,
577 &duration));
578 EXPECT_EQ(duration.InSeconds(), 1);
579 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
580 state_variable_key,
581 &duration));
582 EXPECT_EQ(duration.InSeconds(), 0);
583
584 // Advance clock two seconds and we should get 2 sec and then 0 sec.
585 fake_clock.SetWallclockTime(base::Time::FromInternalValue(4000000));
586 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
587 state_variable_key,
588 &duration));
589 EXPECT_EQ(duration.InSeconds(), 2);
590 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
591 state_variable_key,
592 &duration));
593 EXPECT_EQ(duration.InSeconds(), 0);
594
595 // There's a possibility that the wallclock can go backwards (NTP
596 // adjustments, for example) so check that we properly handle this
597 // case.
598 fake_clock.SetWallclockTime(base::Time::FromInternalValue(3000000));
599 EXPECT_FALSE(utils::WallclockDurationHelper(&mock_system_state,
600 state_variable_key,
601 &duration));
602 fake_clock.SetWallclockTime(base::Time::FromInternalValue(4000000));
603 EXPECT_TRUE(utils::WallclockDurationHelper(&mock_system_state,
604 state_variable_key,
605 &duration));
606 EXPECT_EQ(duration.InSeconds(), 1);
607
608 EXPECT_TRUE(utils::RecursiveUnlinkDir(temp_dir));
609}
610
611TEST(UtilsTest, MonotonicDurationHelper) {
612 int64_t storage = 0;
613 MockSystemState mock_system_state;
614 FakeClock fake_clock;
615 base::TimeDelta duration;
616
617 mock_system_state.set_clock(&fake_clock);
618
619 // Initialize monotonic clock to 1 sec.
620 fake_clock.SetMonotonicTime(base::Time::FromInternalValue(1000000));
621
622 // First time called so no previous measurement available.
623 EXPECT_FALSE(utils::MonotonicDurationHelper(&mock_system_state,
624 &storage,
625 &duration));
626
627 // Next time, we should get zero since the clock didn't advance.
628 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
629 &storage,
630 &duration));
631 EXPECT_EQ(duration.InSeconds(), 0);
632
633 // We can also call it as many times as we want with it being
634 // considered a failure.
635 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
636 &storage,
637 &duration));
638 EXPECT_EQ(duration.InSeconds(), 0);
639 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
640 &storage,
641 &duration));
642 EXPECT_EQ(duration.InSeconds(), 0);
643
644 // Advance the clock one second, then we should get 1 sec on the
645 // next call and 0 sec on the subsequent call.
646 fake_clock.SetMonotonicTime(base::Time::FromInternalValue(2000000));
647 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
648 &storage,
649 &duration));
650 EXPECT_EQ(duration.InSeconds(), 1);
651 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
652 &storage,
653 &duration));
654 EXPECT_EQ(duration.InSeconds(), 0);
655
656 // Advance clock two seconds and we should get 2 sec and then 0 sec.
657 fake_clock.SetMonotonicTime(base::Time::FromInternalValue(4000000));
658 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
659 &storage,
660 &duration));
661 EXPECT_EQ(duration.InSeconds(), 2);
662 EXPECT_TRUE(utils::MonotonicDurationHelper(&mock_system_state,
663 &storage,
664 &duration));
665 EXPECT_EQ(duration.InSeconds(), 0);
666}
667
adlr@google.com3defe6a2009-12-04 20:57:17 +0000668} // namespace chromeos_update_engine