blob: 4d9dea4dcf9c4fe2ea45ef6272cf4e22a94006ac [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
Darin Petkov8e447e02013-04-16 16:23:50 +020013#include <base/file_path.h>
14#include <base/file_util.h>
Darin Petkovd3f8c892010-10-12 21:38:45 -070015#include <base/string_util.h>
Mike Frysinger8155d082012-04-06 15:23:18 -040016#include <base/stringprintf.h>
Darin Petkovd3f8c892010-10-12 21:38:45 -070017#include <gtest/gtest.h>
18
19#include "update_engine/test_utils.h"
adlr@google.com3defe6a2009-12-04 20:57:17 +000020#include "update_engine/utils.h"
21
Andrew de los Reyescc92cd32010-10-05 16:56:14 -070022using std::map;
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -080023using std::string;
adlr@google.com3defe6a2009-12-04 20:57:17 +000024using std::vector;
25
26namespace chromeos_update_engine {
27
28class UtilsTest : public ::testing::Test { };
29
Darin Petkov33d30642010-08-04 10:18:57 -070030TEST(UtilsTest, IsOfficialBuild) {
31 // Pretty lame test...
32 EXPECT_TRUE(utils::IsOfficialBuild());
33}
34
Darin Petkovc91dd6b2011-01-10 12:31:34 -080035TEST(UtilsTest, IsNormalBootMode) {
36 // Pretty lame test...
Darin Petkov44d98d92011-03-21 16:08:11 -070037 EXPECT_TRUE(utils::IsNormalBootMode());
Darin Petkovc91dd6b2011-01-10 12:31:34 -080038}
39
Chris Sosac1972482013-04-30 22:31:10 -070040TEST(UtilsTest, CanParseECVersion) {
41 // Chroot won't have an ec version.
42 EXPECT_EQ("", utils::GetECVersion(NULL));
43
44 // Should be able to parse and valid key value line.
45 EXPECT_EQ("12345", utils::GetECVersion("fw_version=12345"));
46 EXPECT_EQ("123456", utils::GetECVersion("b=1231a fw_version=123456 a=fasd2"));
47 EXPECT_EQ("12345", utils::GetECVersion("fw_version=12345"));
48 EXPECT_EQ("00VFA616", utils::GetECVersion(
49 "vendor=\"sam\" fw_version=\"00VFA616\""));
50
51 // For invalid entries, should return the empty string.
52 EXPECT_EQ("", utils::GetECVersion("b=1231a fw_version a=fasd2"));
53}
54
adlr@google.com3defe6a2009-12-04 20:57:17 +000055TEST(UtilsTest, NormalizePathTest) {
56 EXPECT_EQ("", utils::NormalizePath("", false));
57 EXPECT_EQ("", utils::NormalizePath("", true));
58 EXPECT_EQ("/", utils::NormalizePath("/", false));
59 EXPECT_EQ("", utils::NormalizePath("/", true));
60 EXPECT_EQ("/", utils::NormalizePath("//", false));
61 EXPECT_EQ("", utils::NormalizePath("//", true));
62 EXPECT_EQ("foo", utils::NormalizePath("foo", false));
63 EXPECT_EQ("foo", utils::NormalizePath("foo", true));
64 EXPECT_EQ("/foo/", utils::NormalizePath("/foo//", false));
65 EXPECT_EQ("/foo", utils::NormalizePath("/foo//", true));
66 EXPECT_EQ("bar/baz/foo/adlr", utils::NormalizePath("bar/baz//foo/adlr",
67 false));
68 EXPECT_EQ("bar/baz/foo/adlr", utils::NormalizePath("bar/baz//foo/adlr",
69 true));
70 EXPECT_EQ("/bar/baz/foo/adlr/", utils::NormalizePath("/bar/baz//foo/adlr/",
71 false));
72 EXPECT_EQ("/bar/baz/foo/adlr", utils::NormalizePath("/bar/baz//foo/adlr/",
73 true));
74 EXPECT_EQ("\\\\", utils::NormalizePath("\\\\", false));
75 EXPECT_EQ("\\\\", utils::NormalizePath("\\\\", true));
76 EXPECT_EQ("\\:/;$PATH\n\\", utils::NormalizePath("\\://;$PATH\n\\", false));
77 EXPECT_EQ("\\:/;$PATH\n\\", utils::NormalizePath("\\://;$PATH\n\\", true));
78 EXPECT_EQ("/spaces s/ ok/s / / /",
79 utils::NormalizePath("/spaces s/ ok/s / / /", false));
80 EXPECT_EQ("/spaces s/ ok/s / / ",
81 utils::NormalizePath("/spaces s/ ok/s / / /", true));
82}
83
84TEST(UtilsTest, ReadFileFailure) {
85 vector<char> empty;
86 EXPECT_FALSE(utils::ReadFile("/this/doesn't/exist", &empty));
87}
88
Darin Petkov8e447e02013-04-16 16:23:50 +020089TEST(UtilsTest, ReadFileChunk) {
90 FilePath file;
91 EXPECT_TRUE(file_util::CreateTemporaryFile(&file));
92 ScopedPathUnlinker unlinker(file.value());
93 vector<char> data;
94 const size_t kSize = 1024 * 1024;
95 for (size_t i = 0; i < kSize; i++) {
96 data.push_back(i % 255);
97 }
98 EXPECT_TRUE(utils::WriteFile(file.value().c_str(), &data[0], data.size()));
99 vector<char> in_data;
100 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), kSize, 10, &in_data));
101 EXPECT_TRUE(in_data.empty());
102 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), 0, -1, &in_data));
103 EXPECT_TRUE(data == in_data);
104 in_data.clear();
105 EXPECT_TRUE(utils::ReadFileChunk(file.value().c_str(), 10, 20, &in_data));
106 EXPECT_TRUE(vector<char>(data.begin() + 10, data.begin() + 10 + 20) ==
107 in_data);
108}
109
adlr@google.com3defe6a2009-12-04 20:57:17 +0000110TEST(UtilsTest, ErrnoNumberAsStringTest) {
111 EXPECT_EQ("No such file or directory", utils::ErrnoNumberAsString(ENOENT));
112}
113
114TEST(UtilsTest, StringHasSuffixTest) {
115 EXPECT_TRUE(utils::StringHasSuffix("foo", "foo"));
116 EXPECT_TRUE(utils::StringHasSuffix("foo", "o"));
117 EXPECT_TRUE(utils::StringHasSuffix("", ""));
118 EXPECT_TRUE(utils::StringHasSuffix("abcabc", "abc"));
119 EXPECT_TRUE(utils::StringHasSuffix("adlrwashere", "ere"));
120 EXPECT_TRUE(utils::StringHasSuffix("abcdefgh", "gh"));
121 EXPECT_TRUE(utils::StringHasSuffix("abcdefgh", ""));
122 EXPECT_FALSE(utils::StringHasSuffix("foo", "afoo"));
123 EXPECT_FALSE(utils::StringHasSuffix("", "x"));
124 EXPECT_FALSE(utils::StringHasSuffix("abcdefgh", "fg"));
125 EXPECT_FALSE(utils::StringHasSuffix("abcdefgh", "ab"));
126}
127
128TEST(UtilsTest, StringHasPrefixTest) {
129 EXPECT_TRUE(utils::StringHasPrefix("foo", "foo"));
130 EXPECT_TRUE(utils::StringHasPrefix("foo", "f"));
131 EXPECT_TRUE(utils::StringHasPrefix("", ""));
132 EXPECT_TRUE(utils::StringHasPrefix("abcabc", "abc"));
133 EXPECT_TRUE(utils::StringHasPrefix("adlrwashere", "adl"));
134 EXPECT_TRUE(utils::StringHasPrefix("abcdefgh", "ab"));
135 EXPECT_TRUE(utils::StringHasPrefix("abcdefgh", ""));
136 EXPECT_FALSE(utils::StringHasPrefix("foo", "fooa"));
137 EXPECT_FALSE(utils::StringHasPrefix("", "x"));
138 EXPECT_FALSE(utils::StringHasPrefix("abcdefgh", "bc"));
139 EXPECT_FALSE(utils::StringHasPrefix("abcdefgh", "gh"));
140}
141
142TEST(UtilsTest, BootDeviceTest) {
143 // Pretty lame test...
144 EXPECT_FALSE(utils::BootDevice().empty());
145}
146
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700147TEST(UtilsTest, BootKernelDeviceTest) {
148 EXPECT_EQ("", utils::BootKernelDevice("foo"));
149 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda0"));
150 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda1"));
151 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda2"));
152 EXPECT_EQ("/dev/sda2", utils::BootKernelDevice("/dev/sda3"));
153 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda4"));
154 EXPECT_EQ("/dev/sda4", utils::BootKernelDevice("/dev/sda5"));
155 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda6"));
156 EXPECT_EQ("/dev/sda6", utils::BootKernelDevice("/dev/sda7"));
157 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda8"));
158 EXPECT_EQ("", utils::BootKernelDevice("/dev/sda9"));
159}
160
adlr@google.com3defe6a2009-12-04 20:57:17 +0000161TEST(UtilsTest, RecursiveUnlinkDirTest) {
162 EXPECT_EQ(0, mkdir("RecursiveUnlinkDirTest-a", 0755));
163 EXPECT_EQ(0, mkdir("RecursiveUnlinkDirTest-b", 0755));
164 EXPECT_EQ(0, symlink("../RecursiveUnlinkDirTest-a",
165 "RecursiveUnlinkDirTest-b/link"));
166 EXPECT_EQ(0, system("echo hi > RecursiveUnlinkDirTest-b/file"));
167 EXPECT_EQ(0, mkdir("RecursiveUnlinkDirTest-b/dir", 0755));
168 EXPECT_EQ(0, system("echo ok > RecursiveUnlinkDirTest-b/dir/subfile"));
169 EXPECT_TRUE(utils::RecursiveUnlinkDir("RecursiveUnlinkDirTest-b"));
170 EXPECT_TRUE(utils::FileExists("RecursiveUnlinkDirTest-a"));
171 EXPECT_EQ(0, system("rm -rf RecursiveUnlinkDirTest-a"));
172 EXPECT_FALSE(utils::FileExists("RecursiveUnlinkDirTest-b"));
173 EXPECT_TRUE(utils::RecursiveUnlinkDir("/something/that/doesnt/exist"));
174}
175
Darin Petkov002b2fe2010-11-22 13:53:22 -0800176TEST(UtilsTest, IsSymlinkTest) {
177 string temp_dir;
178 EXPECT_TRUE(utils::MakeTempDirectory("/tmp/symlink-test.XXXXXX", &temp_dir));
179 string temp_file = temp_dir + "temp-file";
180 EXPECT_TRUE(utils::WriteFile(temp_file.c_str(), "", 0));
181 string temp_symlink = temp_dir + "temp-symlink";
182 EXPECT_EQ(0, symlink(temp_file.c_str(), temp_symlink.c_str()));
183 EXPECT_FALSE(utils::IsSymlink(temp_dir.c_str()));
184 EXPECT_FALSE(utils::IsSymlink(temp_file.c_str()));
185 EXPECT_TRUE(utils::IsSymlink(temp_symlink.c_str()));
186 EXPECT_FALSE(utils::IsSymlink("/non/existent/path"));
187 EXPECT_TRUE(utils::RecursiveUnlinkDir(temp_dir));
188}
189
adlr@google.com3defe6a2009-12-04 20:57:17 +0000190TEST(UtilsTest, TempFilenameTest) {
191 const string original = "/foo.XXXXXX";
192 const string result = utils::TempFilename(original);
193 EXPECT_EQ(original.size(), result.size());
194 EXPECT_TRUE(utils::StringHasPrefix(result, "/foo."));
195 EXPECT_FALSE(utils::StringHasSuffix(result, "XXXXXX"));
196}
197
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700198TEST(UtilsTest, RootDeviceTest) {
199 EXPECT_EQ("/dev/sda", utils::RootDevice("/dev/sda3"));
200 EXPECT_EQ("/dev/mmc0", utils::RootDevice("/dev/mmc0p3"));
Darin Petkovf74eb652010-08-04 12:08:38 -0700201 EXPECT_EQ("", utils::RootDevice("/dev/foo/bar"));
202 EXPECT_EQ("", utils::RootDevice("/"));
203 EXPECT_EQ("", utils::RootDevice(""));
204}
205
206TEST(UtilsTest, SysfsBlockDeviceTest) {
207 EXPECT_EQ("/sys/block/sda", utils::SysfsBlockDevice("/dev/sda"));
208 EXPECT_EQ("", utils::SysfsBlockDevice("/foo/sda"));
209 EXPECT_EQ("", utils::SysfsBlockDevice("/dev/foo/bar"));
210 EXPECT_EQ("", utils::SysfsBlockDevice("/"));
211 EXPECT_EQ("", utils::SysfsBlockDevice("./"));
212 EXPECT_EQ("", utils::SysfsBlockDevice(""));
213}
214
215TEST(UtilsTest, IsRemovableDeviceTest) {
216 EXPECT_FALSE(utils::IsRemovableDevice(""));
217 EXPECT_FALSE(utils::IsRemovableDevice("/dev/non-existent-device"));
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700218}
219
220TEST(UtilsTest, PartitionNumberTest) {
221 EXPECT_EQ("3", utils::PartitionNumber("/dev/sda3"));
222 EXPECT_EQ("3", utils::PartitionNumber("/dev/mmc0p3"));
223}
224
Chris Sosa4f8ee272012-11-30 13:01:54 -0800225TEST(UtilsTest, CompareCpuSharesTest) {
226 EXPECT_LT(utils::CompareCpuShares(utils::kCpuSharesLow,
227 utils::kCpuSharesNormal), 0);
228 EXPECT_GT(utils::CompareCpuShares(utils::kCpuSharesNormal,
229 utils::kCpuSharesLow), 0);
230 EXPECT_EQ(utils::CompareCpuShares(utils::kCpuSharesNormal,
231 utils::kCpuSharesNormal), 0);
232 EXPECT_GT(utils::CompareCpuShares(utils::kCpuSharesHigh,
233 utils::kCpuSharesNormal), 0);
Darin Petkovc6c135c2010-08-11 13:36:18 -0700234}
235
Darin Petkov5c0a8af2010-08-24 13:39:13 -0700236TEST(UtilsTest, FuzzIntTest) {
237 static const unsigned int kRanges[] = { 0, 1, 2, 20 };
238 for (size_t r = 0; r < arraysize(kRanges); ++r) {
239 unsigned int range = kRanges[r];
240 const int kValue = 50;
241 for (int tries = 0; tries < 100; ++tries) {
242 int value = utils::FuzzInt(kValue, range);
243 EXPECT_GE(value, kValue - range / 2);
244 EXPECT_LE(value, kValue + range - range / 2);
245 }
246 }
247}
248
Andrew de los Reyescc92cd32010-10-05 16:56:14 -0700249TEST(UtilsTest, ApplyMapTest) {
250 int initial_values[] = {1, 2, 3, 4, 6};
251 vector<int> collection(&initial_values[0],
252 initial_values + arraysize(initial_values));
253 EXPECT_EQ(arraysize(initial_values), collection.size());
254 int expected_values[] = {1, 2, 5, 4, 8};
255 map<int, int> value_map;
256 value_map[3] = 5;
257 value_map[6] = 8;
258 value_map[5] = 10;
259
260 utils::ApplyMap(&collection, value_map);
261
262 size_t index = 0;
263 for (vector<int>::iterator it = collection.begin(), e = collection.end();
264 it != e; ++it) {
265 EXPECT_EQ(expected_values[index++], *it);
266 }
267}
268
Darin Petkovd3f8c892010-10-12 21:38:45 -0700269TEST(UtilsTest, RunAsRootGetFilesystemSizeTest) {
270 string img;
271 EXPECT_TRUE(utils::MakeTempFile("/tmp/img.XXXXXX", &img, NULL));
272 ScopedPathUnlinker img_unlinker(img);
273 CreateExtImageAtPath(img, NULL);
274 // Extend the "partition" holding the file system from 10MiB to 20MiB.
275 EXPECT_EQ(0, System(base::StringPrintf(
276 "dd if=/dev/zero of=%s seek=20971519 bs=1 count=1",
277 img.c_str())));
278 EXPECT_EQ(20 * 1024 * 1024, utils::FileSize(img));
279 int block_count = 0;
280 int block_size = 0;
281 EXPECT_TRUE(utils::GetFilesystemSize(img, &block_count, &block_size));
282 EXPECT_EQ(4096, block_size);
283 EXPECT_EQ(10 * 1024 * 1024 / 4096, block_count);
284}
285
Andrew de los Reyes712b3ac2011-01-07 13:47:52 -0800286namespace {
287gboolean TerminateScheduleCrashReporterUploadTest(void* arg) {
288 GMainLoop* loop = reinterpret_cast<GMainLoop*>(arg);
289 g_main_loop_quit(loop);
290 return FALSE; // Don't call this callback again
291}
292} // namespace {}
293
294TEST(UtilsTest, ScheduleCrashReporterUploadTest) {
295 // Not much to test. At least this tests for memory leaks, crashes,
296 // log errors.
297 GMainLoop* loop = g_main_loop_new(g_main_context_default(), FALSE);
298 utils::ScheduleCrashReporterUpload();
299 g_timeout_add_seconds(1, &TerminateScheduleCrashReporterUploadTest, loop);
300 g_main_loop_run(loop);
301 g_main_loop_unref(loop);
302}
303
David Zeuthen674c3182013-04-18 14:05:20 -0700304TEST(UtilsTest, FormatTimeDeltaTest) {
305 // utils::FormatTimeDelta() is not locale-aware (it's only used for logging
306 // which is not localized) so we only need to test the C locale
307 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromMilliseconds(100)),
308 "0.1s");
309 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(0)),
310 "0s");
311 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1)),
312 "1s");
313 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(59)),
314 "59s");
315 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(60)),
316 "1m0s");
317 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(61)),
318 "1m1s");
319 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(90)),
320 "1m30s");
321 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1205)),
322 "20m5s");
323 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3600)),
324 "1h0m0s");
325 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3601)),
326 "1h0m1s");
327 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3661)),
328 "1h1m1s");
329 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(7261)),
330 "2h1m1s");
331 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86400)),
332 "1d0h0m0s");
333 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86401)),
334 "1d0h0m1s");
335 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000)),
336 "2d7h33m20s");
337 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000) +
338 base::TimeDelta::FromMilliseconds(1)),
339 "2d7h33m20.001s");
340}
341
adlr@google.com3defe6a2009-12-04 20:57:17 +0000342} // namespace chromeos_update_engine