blob: d73b3da72a8d7c9560feae3cdd7fff722567765d [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2012 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
adlr@google.com3defe6a2009-12-04 20:57:17 +000016
Alex Deymo39910dc2015-11-09 17:04:30 -080017#include "update_engine/common/utils.h"
Alex Deymo2c0db7b2014-11-04 12:23:39 -080018
Alex Deymo5ba93ad2016-08-22 19:51:59 -070019#include <fcntl.h>
Ben Chan9abb7632014-08-07 00:10:53 -070020#include <stdint.h>
Alex Deymo5ba93ad2016-08-22 19:51:59 -070021#include <sys/mount.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000022#include <sys/stat.h>
23#include <sys/types.h>
Darin Petkov5c0a8af2010-08-24 13:39:13 -070024
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -070025#include <limits>
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -080026#include <string>
adlr@google.com3defe6a2009-12-04 20:57:17 +000027#include <vector>
Darin Petkov5c0a8af2010-08-24 13:39:13 -070028
Alex Deymoc1711e22014-08-08 13:16:23 -070029#include <base/files/file_path.h>
Alex Deymo2c0db7b2014-11-04 12:23:39 -080030#include <base/files/file_util.h>
Sen Jiang371615b2016-04-13 15:54:29 -070031#include <base/files/scoped_temp_dir.h>
Darin Petkovd3f8c892010-10-12 21:38:45 -070032#include <gtest/gtest.h>
33
Alex Deymo39910dc2015-11-09 17:04:30 -080034#include "update_engine/common/test_utils.h"
adlr@google.com3defe6a2009-12-04 20:57:17 +000035
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -070036using std::numeric_limits;
Andrew de los Reyes4fe15d02009-12-10 19:01:36 -080037using std::string;
adlr@google.com3defe6a2009-12-04 20:57:17 +000038using std::vector;
39
40namespace chromeos_update_engine {
41
Amin Hassanib2689592019-01-13 17:04:28 -080042class UtilsTest : public ::testing::Test {};
adlr@google.com3defe6a2009-12-04 20:57:17 +000043
Chris Sosac1972482013-04-30 22:31:10 -070044TEST(UtilsTest, CanParseECVersion) {
Chris Sosac1972482013-04-30 22:31:10 -070045 // Should be able to parse and valid key value line.
J. Richard Barnette63137e52013-10-28 10:57:29 -070046 EXPECT_EQ("12345", utils::ParseECVersion("fw_version=12345"));
Amin Hassanib2689592019-01-13 17:04:28 -080047 EXPECT_EQ("123456",
48 utils::ParseECVersion("b=1231a fw_version=123456 a=fasd2"));
J. Richard Barnette63137e52013-10-28 10:57:29 -070049 EXPECT_EQ("12345", utils::ParseECVersion("fw_version=12345"));
Amin Hassanib2689592019-01-13 17:04:28 -080050 EXPECT_EQ("00VFA616",
51 utils::ParseECVersion("vendor=\"sam\" fw_version=\"00VFA616\""));
Chris Sosac1972482013-04-30 22:31:10 -070052
53 // For invalid entries, should return the empty string.
J. Richard Barnette63137e52013-10-28 10:57:29 -070054 EXPECT_EQ("", utils::ParseECVersion("b=1231a fw_version a=fasd2"));
Chris Sosac1972482013-04-30 22:31:10 -070055}
56
Alex Deymo335516c2016-11-28 13:37:06 -080057TEST(UtilsTest, WriteFileOpenFailure) {
58 EXPECT_FALSE(utils::WriteFile("/this/doesn't/exist", "hello", 5));
59}
60
61TEST(UtilsTest, WriteFileReadFile) {
Sen Jiang0779a152018-07-02 17:34:56 -070062 test_utils::ScopedTempFile file;
63 EXPECT_TRUE(utils::WriteFile(file.path().c_str(), "hello", 5));
Alex Deymo335516c2016-11-28 13:37:06 -080064
65 brillo::Blob readback;
Sen Jiang0779a152018-07-02 17:34:56 -070066 EXPECT_TRUE(utils::ReadFile(file.path().c_str(), &readback));
Alex Deymo335516c2016-11-28 13:37:06 -080067 EXPECT_EQ("hello", string(readback.begin(), readback.end()));
68}
69
adlr@google.com3defe6a2009-12-04 20:57:17 +000070TEST(UtilsTest, ReadFileFailure) {
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070071 brillo::Blob empty;
adlr@google.com3defe6a2009-12-04 20:57:17 +000072 EXPECT_FALSE(utils::ReadFile("/this/doesn't/exist", &empty));
73}
74
Darin Petkov8e447e02013-04-16 16:23:50 +020075TEST(UtilsTest, ReadFileChunk) {
Sen Jiang0779a152018-07-02 17:34:56 -070076 test_utils::ScopedTempFile file;
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070077 brillo::Blob data;
Darin Petkov8e447e02013-04-16 16:23:50 +020078 const size_t kSize = 1024 * 1024;
79 for (size_t i = 0; i < kSize; i++) {
80 data.push_back(i % 255);
81 }
Sen Jiang0779a152018-07-02 17:34:56 -070082 EXPECT_TRUE(test_utils::WriteFileVector(file.path(), data));
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070083 brillo::Blob in_data;
Sen Jiang0779a152018-07-02 17:34:56 -070084 EXPECT_TRUE(utils::ReadFileChunk(file.path().c_str(), kSize, 10, &in_data));
Darin Petkov8e447e02013-04-16 16:23:50 +020085 EXPECT_TRUE(in_data.empty());
Sen Jiang0779a152018-07-02 17:34:56 -070086 EXPECT_TRUE(utils::ReadFileChunk(file.path().c_str(), 0, -1, &in_data));
87 EXPECT_EQ(data, in_data);
Darin Petkov8e447e02013-04-16 16:23:50 +020088 in_data.clear();
Sen Jiang0779a152018-07-02 17:34:56 -070089 EXPECT_TRUE(utils::ReadFileChunk(file.path().c_str(), 10, 20, &in_data));
90 EXPECT_EQ(brillo::Blob(data.begin() + 10, data.begin() + 10 + 20), in_data);
Darin Petkov8e447e02013-04-16 16:23:50 +020091}
92
adlr@google.com3defe6a2009-12-04 20:57:17 +000093TEST(UtilsTest, ErrnoNumberAsStringTest) {
94 EXPECT_EQ("No such file or directory", utils::ErrnoNumberAsString(ENOENT));
95}
96
Darin Petkov002b2fe2010-11-22 13:53:22 -080097TEST(UtilsTest, IsSymlinkTest) {
Sen Jiang371615b2016-04-13 15:54:29 -070098 base::ScopedTempDir temp_dir;
99 ASSERT_TRUE(temp_dir.CreateUniqueTempDir());
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900100 string temp_file = temp_dir.GetPath().Append("temp-file").value();
Darin Petkov002b2fe2010-11-22 13:53:22 -0800101 EXPECT_TRUE(utils::WriteFile(temp_file.c_str(), "", 0));
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900102 string temp_symlink = temp_dir.GetPath().Append("temp-symlink").value();
Darin Petkov002b2fe2010-11-22 13:53:22 -0800103 EXPECT_EQ(0, symlink(temp_file.c_str(), temp_symlink.c_str()));
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900104 EXPECT_FALSE(utils::IsSymlink(temp_dir.GetPath().value().c_str()));
Darin Petkov002b2fe2010-11-22 13:53:22 -0800105 EXPECT_FALSE(utils::IsSymlink(temp_file.c_str()));
106 EXPECT_TRUE(utils::IsSymlink(temp_symlink.c_str()));
107 EXPECT_FALSE(utils::IsSymlink("/non/existent/path"));
Darin Petkov002b2fe2010-11-22 13:53:22 -0800108}
109
Alex Deymo763e7db2015-08-27 21:08:08 -0700110TEST(UtilsTest, SplitPartitionNameTest) {
111 string disk;
112 int part_num;
Darin Petkovf74eb652010-08-04 12:08:38 -0700113
Alex Deymo763e7db2015-08-27 21:08:08 -0700114 EXPECT_TRUE(utils::SplitPartitionName("/dev/sda3", &disk, &part_num));
115 EXPECT_EQ("/dev/sda", disk);
116 EXPECT_EQ(3, part_num);
Darin Petkovf74eb652010-08-04 12:08:38 -0700117
Alex Deymo763e7db2015-08-27 21:08:08 -0700118 EXPECT_TRUE(utils::SplitPartitionName("/dev/sdp1234", &disk, &part_num));
119 EXPECT_EQ("/dev/sdp", disk);
120 EXPECT_EQ(1234, part_num);
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700121
Alex Deymo763e7db2015-08-27 21:08:08 -0700122 EXPECT_TRUE(utils::SplitPartitionName("/dev/mmcblk0p3", &disk, &part_num));
123 EXPECT_EQ("/dev/mmcblk0", disk);
124 EXPECT_EQ(3, part_num);
125
Alex Deymo763e7db2015-08-27 21:08:08 -0700126 EXPECT_TRUE(utils::SplitPartitionName("/dev/loop10", &disk, &part_num));
127 EXPECT_EQ("/dev/loop", disk);
128 EXPECT_EQ(10, part_num);
129
130 EXPECT_TRUE(utils::SplitPartitionName("/dev/loop28p11", &disk, &part_num));
131 EXPECT_EQ("/dev/loop28", disk);
132 EXPECT_EQ(11, part_num);
133
Alex Deymo763e7db2015-08-27 21:08:08 -0700134 EXPECT_FALSE(utils::SplitPartitionName("/dev/mmcblk0p", &disk, &part_num));
135 EXPECT_FALSE(utils::SplitPartitionName("/dev/sda", &disk, &part_num));
136 EXPECT_FALSE(utils::SplitPartitionName("/dev/foo/bar", &disk, &part_num));
137 EXPECT_FALSE(utils::SplitPartitionName("/", &disk, &part_num));
138 EXPECT_FALSE(utils::SplitPartitionName("", &disk, &part_num));
Andrew de los Reyesf9714432010-05-04 10:21:23 -0700139}
140
Alex Vakulenkof3f85bb2014-03-26 16:36:35 -0700141TEST(UtilsTest, MakePartitionNameTest) {
142 EXPECT_EQ("/dev/sda4", utils::MakePartitionName("/dev/sda", 4));
143 EXPECT_EQ("/dev/sda123", utils::MakePartitionName("/dev/sda", 123));
144 EXPECT_EQ("/dev/mmcblk2", utils::MakePartitionName("/dev/mmcblk", 2));
145 EXPECT_EQ("/dev/mmcblk0p2", utils::MakePartitionName("/dev/mmcblk0", 2));
146 EXPECT_EQ("/dev/loop8", utils::MakePartitionName("/dev/loop", 8));
147 EXPECT_EQ("/dev/loop12p2", utils::MakePartitionName("/dev/loop12", 2));
Alex Vakulenkof3f85bb2014-03-26 16:36:35 -0700148}
149
Darin Petkov5c0a8af2010-08-24 13:39:13 -0700150TEST(UtilsTest, FuzzIntTest) {
Amin Hassanib2689592019-01-13 17:04:28 -0800151 static const uint32_t kRanges[] = {0, 1, 2, 20};
Alex Deymo80f70ff2016-02-10 16:08:11 -0800152 for (uint32_t range : kRanges) {
Darin Petkov5c0a8af2010-08-24 13:39:13 -0700153 const int kValue = 50;
154 for (int tries = 0; tries < 100; ++tries) {
Alex Deymo80f70ff2016-02-10 16:08:11 -0800155 uint32_t value = utils::FuzzInt(kValue, range);
Darin Petkov5c0a8af2010-08-24 13:39:13 -0700156 EXPECT_GE(value, kValue - range / 2);
157 EXPECT_LE(value, kValue + range - range / 2);
158 }
159 }
160}
161
Andrew de los Reyes712b3ac2011-01-07 13:47:52 -0800162namespace {
Alex Deymo032e7722014-03-25 17:53:56 -0700163void GetFileFormatTester(const string& expected,
Ben Chan9abb7632014-08-07 00:10:53 -0700164 const vector<uint8_t>& contents) {
Alex Deymo10875d92014-11-10 21:52:57 -0800165 test_utils::ScopedTempFile file;
Alex Deymobffa0602016-02-12 17:16:29 -0800166 ASSERT_TRUE(utils::WriteFile(file.path().c_str(),
Alex Deymo032e7722014-03-25 17:53:56 -0700167 reinterpret_cast<const char*>(contents.data()),
168 contents.size()));
Alex Deymobffa0602016-02-12 17:16:29 -0800169 EXPECT_EQ(expected, utils::GetFileFormat(file.path()));
Alex Deymo032e7722014-03-25 17:53:56 -0700170}
Alex Deymo10875d92014-11-10 21:52:57 -0800171} // namespace
Alex Deymo032e7722014-03-25 17:53:56 -0700172
173TEST(UtilsTest, GetFileFormatTest) {
174 EXPECT_EQ("File not found.", utils::GetFileFormat("/path/to/nowhere"));
Ben Chan9abb7632014-08-07 00:10:53 -0700175 GetFileFormatTester("data", vector<uint8_t>{1, 2, 3, 4, 5, 6, 7, 8});
176 GetFileFormatTester("ELF", vector<uint8_t>{0x7f, 0x45, 0x4c, 0x46});
Alex Deymo032e7722014-03-25 17:53:56 -0700177
178 // Real tests from cros_installer on different boards.
179 // ELF 32-bit LSB executable, Intel 80386
180 GetFileFormatTester(
181 "ELF 32-bit little-endian x86",
Ben Chan9abb7632014-08-07 00:10:53 -0700182 vector<uint8_t>{0x7f, 0x45, 0x4c, 0x46, 0x01, 0x01, 0x01, 0x00,
183 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
184 0x02, 0x00, 0x03, 0x00, 0x01, 0x00, 0x00, 0x00,
185 0x90, 0x83, 0x04, 0x08, 0x34, 0x00, 0x00, 0x00});
Alex Deymo032e7722014-03-25 17:53:56 -0700186
Alex Deymoc1711e22014-08-08 13:16:23 -0700187 // ELF 32-bit LSB executable, MIPS
188 GetFileFormatTester(
189 "ELF 32-bit little-endian mips",
190 vector<uint8_t>{0x7f, 0x45, 0x4c, 0x46, 0x01, 0x01, 0x01, 0x00,
191 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
192 0x03, 0x00, 0x08, 0x00, 0x01, 0x00, 0x00, 0x00,
193 0xc0, 0x12, 0x00, 0x00, 0x34, 0x00, 0x00, 0x00});
194
Alex Deymo032e7722014-03-25 17:53:56 -0700195 // ELF 32-bit LSB executable, ARM
196 GetFileFormatTester(
197 "ELF 32-bit little-endian arm",
Ben Chan9abb7632014-08-07 00:10:53 -0700198 vector<uint8_t>{0x7f, 0x45, 0x4c, 0x46, 0x01, 0x01, 0x01, 0x00,
199 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
200 0x02, 0x00, 0x28, 0x00, 0x01, 0x00, 0x00, 0x00,
201 0x85, 0x8b, 0x00, 0x00, 0x34, 0x00, 0x00, 0x00});
Alex Deymo032e7722014-03-25 17:53:56 -0700202
203 // ELF 64-bit LSB executable, x86-64
204 GetFileFormatTester(
205 "ELF 64-bit little-endian x86-64",
Ben Chan9abb7632014-08-07 00:10:53 -0700206 vector<uint8_t>{0x7f, 0x45, 0x4c, 0x46, 0x02, 0x01, 0x01, 0x00,
207 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
208 0x02, 0x00, 0x3e, 0x00, 0x01, 0x00, 0x00, 0x00,
209 0xb0, 0x04, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00});
Alex Deymo032e7722014-03-25 17:53:56 -0700210}
211
David Zeuthen674c3182013-04-18 14:05:20 -0700212TEST(UtilsTest, FormatTimeDeltaTest) {
213 // utils::FormatTimeDelta() is not locale-aware (it's only used for logging
214 // which is not localized) so we only need to test the C locale
215 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromMilliseconds(100)),
216 "0.1s");
Amin Hassanib2689592019-01-13 17:04:28 -0800217 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(0)), "0s");
218 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1)), "1s");
219 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(59)), "59s");
220 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(60)), "1m0s");
221 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(61)), "1m1s");
222 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(90)), "1m30s");
David Zeuthen674c3182013-04-18 14:05:20 -0700223 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(1205)),
224 "20m5s");
225 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3600)),
226 "1h0m0s");
227 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3601)),
228 "1h0m1s");
229 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(3661)),
230 "1h1m1s");
231 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(7261)),
232 "2h1m1s");
233 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86400)),
234 "1d0h0m0s");
235 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(86401)),
236 "1d0h0m1s");
237 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000)),
238 "2d7h33m20s");
239 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(200000) +
240 base::TimeDelta::FromMilliseconds(1)),
241 "2d7h33m20.001s");
Amin Hassanib2689592019-01-13 17:04:28 -0800242 EXPECT_EQ(utils::FormatTimeDelta(base::TimeDelta::FromSeconds(-1)), "-1s");
David Zeuthen674c3182013-04-18 14:05:20 -0700243}
244
David Zeuthen639aa362014-02-03 16:23:44 -0800245TEST(UtilsTest, ConvertToOmahaInstallDate) {
246 // The Omaha Epoch starts at Jan 1, 2007 0:00 PST which is a
247 // Monday. In Unix time, this point in time is easily obtained via
248 // the date(1) command like this:
249 //
250 // $ date +"%s" --date="Jan 1, 2007 0:00 PST"
251 const time_t omaha_epoch = 1167638400;
252 int value;
253
254 // Points in time *on and after* the Omaha epoch should not fail.
255 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
256 base::Time::FromTimeT(omaha_epoch), &value));
257 EXPECT_GE(value, 0);
258
259 // Anything before the Omaha epoch should fail. We test it for two points.
260 EXPECT_FALSE(utils::ConvertToOmahaInstallDate(
261 base::Time::FromTimeT(omaha_epoch - 1), &value));
262 EXPECT_FALSE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800263 base::Time::FromTimeT(omaha_epoch - 100 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800264
265 // Check that we jump from 0 to 7 exactly on the one-week mark, e.g.
266 // on Jan 8, 2007 0:00 PST.
267 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800268 base::Time::FromTimeT(omaha_epoch + 7 * 24 * 3600 - 1), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800269 EXPECT_EQ(value, 0);
270 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800271 base::Time::FromTimeT(omaha_epoch + 7 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800272 EXPECT_EQ(value, 7);
273
274 // Check a couple of more values.
275 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800276 base::Time::FromTimeT(omaha_epoch + 10 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800277 EXPECT_EQ(value, 7);
278 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800279 base::Time::FromTimeT(omaha_epoch + 20 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800280 EXPECT_EQ(value, 14);
281 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800282 base::Time::FromTimeT(omaha_epoch + 26 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800283 EXPECT_EQ(value, 21);
284 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
Amin Hassanib2689592019-01-13 17:04:28 -0800285 base::Time::FromTimeT(omaha_epoch + 29 * 24 * 3600), &value));
David Zeuthen639aa362014-02-03 16:23:44 -0800286 EXPECT_EQ(value, 28);
287
288 // The date Jun 4, 2007 0:00 PDT is a Monday and is hence a point
289 // where the Omaha InstallDate jumps 7 days. Its unix time is
290 // 1180940400. Notably, this is a point in time where Daylight
291 // Savings Time (DST) was is in effect (e.g. it's PDT, not PST).
292 //
293 // Note that as utils::ConvertToOmahaInstallDate() _deliberately_
294 // ignores DST (as it's hard to implement in a thread-safe way using
295 // glibc, see comments in utils.h) we have to fudge by the DST
296 // offset which is one hour. Conveniently, if the function were
297 // someday modified to be DST aware, this test would have to be
298 // modified as well.
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700299 const time_t dst_time = 1180940400; // Jun 4, 2007 0:00 PDT.
David Zeuthen639aa362014-02-03 16:23:44 -0800300 const time_t fudge = 3600;
301 int value1, value2;
302 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
303 base::Time::FromTimeT(dst_time + fudge - 1), &value1));
304 EXPECT_TRUE(utils::ConvertToOmahaInstallDate(
305 base::Time::FromTimeT(dst_time + fudge), &value2));
306 EXPECT_EQ(value1, value2 - 7);
307}
308
Allie Wood78750a42015-02-11 15:42:11 -0800309TEST(UtilsTest, GetMinorVersion) {
310 // Test GetMinorVersion by verifying that it parses the conf file and returns
311 // the correct value.
Allie Wood78750a42015-02-11 15:42:11 -0800312 uint32_t minor_version;
313
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700314 brillo::KeyValueStore store;
Alex Deymob42b98d2015-07-06 17:42:38 -0700315 EXPECT_FALSE(utils::GetMinorVersion(store, &minor_version));
Allie Wood78750a42015-02-11 15:42:11 -0800316
Alex Deymob42b98d2015-07-06 17:42:38 -0700317 EXPECT_TRUE(store.LoadFromString("PAYLOAD_MINOR_VERSION=one-two-three\n"));
318 EXPECT_FALSE(utils::GetMinorVersion(store, &minor_version));
Allie Wood78750a42015-02-11 15:42:11 -0800319
Alex Deymob42b98d2015-07-06 17:42:38 -0700320 EXPECT_TRUE(store.LoadFromString("PAYLOAD_MINOR_VERSION=123\n"));
321 EXPECT_TRUE(utils::GetMinorVersion(store, &minor_version));
Alex Deymo80f70ff2016-02-10 16:08:11 -0800322 EXPECT_EQ(123U, minor_version);
Allie Wood78750a42015-02-11 15:42:11 -0800323}
324
Nam T. Nguyen2b67a592014-12-03 14:56:00 -0800325static bool BoolMacroTestHelper() {
326 int i = 1;
327 unsigned int ui = 1;
328 bool b = 1;
329 std::unique_ptr<char> cptr(new char);
330
331 TEST_AND_RETURN_FALSE(i);
332 TEST_AND_RETURN_FALSE(ui);
333 TEST_AND_RETURN_FALSE(b);
334 TEST_AND_RETURN_FALSE(cptr);
335
336 TEST_AND_RETURN_FALSE_ERRNO(i);
337 TEST_AND_RETURN_FALSE_ERRNO(ui);
338 TEST_AND_RETURN_FALSE_ERRNO(b);
339 TEST_AND_RETURN_FALSE_ERRNO(cptr);
340
341 return true;
342}
343
344static void VoidMacroTestHelper(bool* ret) {
345 int i = 1;
346 unsigned int ui = 1;
347 bool b = 1;
348 std::unique_ptr<char> cptr(new char);
349
350 *ret = false;
351
352 TEST_AND_RETURN(i);
353 TEST_AND_RETURN(ui);
354 TEST_AND_RETURN(b);
355 TEST_AND_RETURN(cptr);
356
357 TEST_AND_RETURN_ERRNO(i);
358 TEST_AND_RETURN_ERRNO(ui);
359 TEST_AND_RETURN_ERRNO(b);
360 TEST_AND_RETURN_ERRNO(cptr);
361
362 *ret = true;
363}
364
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -0700365static void ExpectParseRollbackKeyVersion(const string& version,
366 uint16_t expected_high,
367 uint16_t expected_low) {
368 uint16_t actual_high;
369 uint16_t actual_low;
370 utils::ParseRollbackKeyVersion(version, &actual_high, &actual_low);
371 EXPECT_EQ(expected_high, actual_high);
372 EXPECT_EQ(expected_low, actual_low);
373}
374
375static void ExpectInvalidParseRollbackKeyVersion(const string& version) {
376 ExpectParseRollbackKeyVersion(version,
377 numeric_limits<uint16_t>::max(),
378 numeric_limits<uint16_t>::max());
379}
380
Nam T. Nguyen2b67a592014-12-03 14:56:00 -0800381TEST(UtilsTest, TestMacros) {
382 bool void_test = false;
383 VoidMacroTestHelper(&void_test);
384 EXPECT_TRUE(void_test);
385
386 EXPECT_TRUE(BoolMacroTestHelper());
387}
388
Alex Deymoa2ea1c22016-08-24 17:26:19 -0700389TEST(UtilsTest, RunAsRootUnmountFilesystemFailureTest) {
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700390 EXPECT_FALSE(utils::UnmountFilesystem("/path/to/non-existing-dir"));
391}
392
Alex Deymoa2ea1c22016-08-24 17:26:19 -0700393TEST(UtilsTest, RunAsRootUnmountFilesystemBusyFailureTest) {
Sen Jiang0779a152018-07-02 17:34:56 -0700394 test_utils::ScopedTempFile tmp_image("img.XXXXXX");
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700395
396 EXPECT_TRUE(base::CopyFile(
397 test_utils::GetBuildArtifactsPath().Append("gen/disk_ext2_4k.img"),
Sen Jiang0779a152018-07-02 17:34:56 -0700398 base::FilePath(tmp_image.path())));
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700399
400 base::ScopedTempDir mnt_dir;
401 EXPECT_TRUE(mnt_dir.CreateUniqueTempDir());
402
403 string loop_dev;
404 test_utils::ScopedLoopbackDeviceBinder loop_binder(
Sen Jiang0779a152018-07-02 17:34:56 -0700405 tmp_image.path(), true, &loop_dev);
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700406
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900407 EXPECT_FALSE(utils::IsMountpoint(mnt_dir.GetPath().value()));
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700408 // This is the actual test part. While we hold a file descriptor open for the
409 // mounted filesystem, umount should still succeed.
410 EXPECT_TRUE(utils::MountFilesystem(
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900411 loop_dev, mnt_dir.GetPath().value(), MS_RDONLY, "ext4", ""));
Alex Deymof4116502017-03-22 17:00:31 -0700412 // Verify the directory is a mount point now.
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900413 EXPECT_TRUE(utils::IsMountpoint(mnt_dir.GetPath().value()));
Alex Deymof4116502017-03-22 17:00:31 -0700414
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900415 string target_file = mnt_dir.GetPath().Append("empty-file").value();
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700416 int fd = HANDLE_EINTR(open(target_file.c_str(), O_RDONLY));
417 EXPECT_GE(fd, 0);
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900418 EXPECT_TRUE(utils::UnmountFilesystem(mnt_dir.GetPath().value()));
Alex Deymof4116502017-03-22 17:00:31 -0700419 // The filesystem should be already unmounted at this point.
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900420 EXPECT_FALSE(utils::IsMountpoint(mnt_dir.GetPath().value()));
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700421 IGNORE_EINTR(close(fd));
422 // The filesystem was already unmounted so this call should fail.
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900423 EXPECT_FALSE(utils::UnmountFilesystem(mnt_dir.GetPath().value()));
Alex Deymo5ba93ad2016-08-22 19:51:59 -0700424}
425
Alex Deymof4116502017-03-22 17:00:31 -0700426TEST(UtilsTest, IsMountpointTest) {
427 EXPECT_TRUE(utils::IsMountpoint("/"));
428 EXPECT_FALSE(utils::IsMountpoint("/path/to/nowhere"));
429
430 base::ScopedTempDir mnt_dir;
431 EXPECT_TRUE(mnt_dir.CreateUniqueTempDir());
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900432 EXPECT_FALSE(utils::IsMountpoint(mnt_dir.GetPath().value()));
Alex Deymof4116502017-03-22 17:00:31 -0700433
Sen Jiang0779a152018-07-02 17:34:56 -0700434 test_utils::ScopedTempFile file;
435 EXPECT_FALSE(utils::IsMountpoint(file.path()));
Alex Deymof4116502017-03-22 17:00:31 -0700436}
437
Marton Hunyadya0302682018-05-16 18:52:13 +0200438TEST(UtilsTest, VersionPrefix) {
439 EXPECT_EQ(10575, utils::VersionPrefix("10575.39."));
440 EXPECT_EQ(10575, utils::VersionPrefix("10575.39"));
441 EXPECT_EQ(10575, utils::VersionPrefix("10575.x"));
442 EXPECT_EQ(10575, utils::VersionPrefix("10575."));
443 EXPECT_EQ(10575, utils::VersionPrefix("10575"));
444 EXPECT_EQ(0, utils::VersionPrefix(""));
445 EXPECT_EQ(-1, utils::VersionPrefix("x"));
446 EXPECT_EQ(-1, utils::VersionPrefix("1x"));
447 EXPECT_EQ(-1, utils::VersionPrefix("x.1"));
448}
449
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -0700450TEST(UtilsTest, ParseDottedVersion) {
451 // Valid case.
452 ExpectParseRollbackKeyVersion("2.3", 2, 3);
453 ExpectParseRollbackKeyVersion("65535.65535", 65535, 65535);
454
455 // Zero is technically allowed but never actually used.
456 ExpectParseRollbackKeyVersion("0.0", 0, 0);
457
458 // Invalid cases.
459 ExpectInvalidParseRollbackKeyVersion("");
460 ExpectInvalidParseRollbackKeyVersion("2");
461 ExpectInvalidParseRollbackKeyVersion("2.");
462 ExpectInvalidParseRollbackKeyVersion(".2");
463 ExpectInvalidParseRollbackKeyVersion("2.2.");
464 ExpectInvalidParseRollbackKeyVersion("2.2.3");
465 ExpectInvalidParseRollbackKeyVersion(".2.2");
466 ExpectInvalidParseRollbackKeyVersion("a.b");
467 ExpectInvalidParseRollbackKeyVersion("1.b");
468 ExpectInvalidParseRollbackKeyVersion("a.2");
469 ExpectInvalidParseRollbackKeyVersion("65536.65536");
470 ExpectInvalidParseRollbackKeyVersion("99999.99999");
471 ExpectInvalidParseRollbackKeyVersion("99999.1");
472 ExpectInvalidParseRollbackKeyVersion("1.99999");
473}
474
Kyeongkab.Nam500ca132019-06-26 13:48:07 +0900475TEST(UtilsTest, GetFilePathTest) {
476 test_utils::ScopedTempFile file;
477 int fd = HANDLE_EINTR(open(file.path().c_str(), O_RDONLY));
478 EXPECT_GE(fd, 0);
479 EXPECT_EQ(file.path(), utils::GetFilePath(fd));
480 EXPECT_EQ("not found", utils::GetFilePath(-1));
481 IGNORE_EINTR(close(fd));
482}
483
Kelvin Zhangd7191032020-08-11 10:48:16 -0400484TEST(UtilsTest, ValidatePerPartitionTimestamp) {
Yifan Hong87029332020-09-01 17:20:08 -0700485 ASSERT_EQ(ErrorCode::kPayloadTimestampError,
486 utils::IsTimestampNewer("10", "5"));
487 ASSERT_EQ(ErrorCode::kSuccess, utils::IsTimestampNewer("10", "11"));
488 ASSERT_EQ(ErrorCode::kDownloadManifestParseError,
489 utils::IsTimestampNewer("10", "lol"));
490 ASSERT_EQ(ErrorCode::kError, utils::IsTimestampNewer("lol", "ZZZ"));
491 ASSERT_EQ(ErrorCode::kSuccess, utils::IsTimestampNewer("10", ""));
Kelvin Zhangd7191032020-08-11 10:48:16 -0400492}
493
adlr@google.com3defe6a2009-12-04 20:57:17 +0000494} // namespace chromeos_update_engine