blob: 02a8409c0feeea11a1c5199a8fbd89a2d35d830f [file] [log] [blame]
Lei YUf77189f2019-08-07 14:26:30 +08001#include "item_updater.hpp"
2#include "mocked_utils.hpp"
3
4#include <sdbusplus/test/sdbus_mock.hpp>
5
6#include <gmock/gmock.h>
7#include <gtest/gtest.h>
8
9using namespace phosphor::software::updater;
10using ::testing::_;
Lei YU1517f5f2019-10-14 16:44:42 +080011using ::testing::ContainerEq;
Lei YU58c26e32019-09-27 17:52:06 +080012using ::testing::Pointee;
Lei YUf77189f2019-08-07 14:26:30 +080013using ::testing::Return;
14using ::testing::ReturnArg;
15using ::testing::StrEq;
16
Patrick Williamsb5f9b822022-06-16 17:26:17 -050017using std::any;
Lei YUf77189f2019-08-07 14:26:30 +080018
19class TestItemUpdater : public ::testing::Test
20{
21 public:
Lei YUa2c2cd72019-08-09 15:54:10 +080022 using Properties = ItemUpdater::Properties;
23 using PropertyType = utils::UtilsInterface::PropertyType;
24
George Liu66a54ad2024-08-23 13:53:39 +080025 TestItemUpdater(const TestItemUpdater&) = delete;
26 TestItemUpdater& operator=(const TestItemUpdater&) = delete;
27 TestItemUpdater(TestItemUpdater&&) = delete;
28 TestItemUpdater& operator=(TestItemUpdater&&) = delete;
29
Lei YUf77189f2019-08-07 14:26:30 +080030 TestItemUpdater() :
31 mockedUtils(
32 reinterpret_cast<const utils::MockedUtils&>(utils::getUtils()))
33 {
34 ON_CALL(mockedUtils, getVersionId(_)).WillByDefault(ReturnArg<0>());
Lei YUff83c2a2019-09-12 13:55:18 +080035 ON_CALL(mockedUtils, getPropertyImpl(_, _, _, _, StrEq(PRESENT)))
36 .WillByDefault(Return(any(PropertyType(true))));
Lei YUf77189f2019-08-07 14:26:30 +080037 }
38
George Liu047d9942024-08-23 13:44:31 +080039 ~TestItemUpdater() override
Lei YUf77189f2019-08-07 14:26:30 +080040 {
Lei YUc09155b2019-10-11 17:30:48 +080041 utils::freeUtils();
Lei YUf77189f2019-08-07 14:26:30 +080042 }
43
George Liu71ae5352024-08-23 15:21:05 +080044 auto& GetActivations() const
Lei YUf77189f2019-08-07 14:26:30 +080045 {
46 return itemUpdater->activations;
47 }
48
George Liu80c2daa2024-08-23 15:46:12 +080049 static std::string getObjPath(const std::string& versionId)
Lei YUf77189f2019-08-07 14:26:30 +080050 {
51 return std::string(dBusPath) + "/" + versionId;
52 }
53
Lei YUa2c2cd72019-08-09 15:54:10 +080054 void onPsuInventoryChanged(const std::string& psuPath,
George Liu71ae5352024-08-23 15:21:05 +080055 const Properties& properties) const
Lei YUa2c2cd72019-08-09 15:54:10 +080056 {
57 itemUpdater->onPsuInventoryChanged(psuPath, properties);
58 }
59
George Liu71ae5352024-08-23 15:21:05 +080060 void scanDirectory(const fs::path& p) const
Lei YU58c26e32019-09-27 17:52:06 +080061 {
62 itemUpdater->scanDirectory(p);
63 }
64
Lei YUf77189f2019-08-07 14:26:30 +080065 static constexpr auto dBusPath = SOFTWARE_OBJPATH;
66 sdbusplus::SdBusMock sdbusMock;
Patrick Williams374fae52022-07-22 19:26:55 -050067 sdbusplus::bus_t mockedBus = sdbusplus::get_mocked_new(&sdbusMock);
Lei YUf77189f2019-08-07 14:26:30 +080068 const utils::MockedUtils& mockedUtils;
69 std::unique_ptr<ItemUpdater> itemUpdater;
Lei YU1517f5f2019-10-14 16:44:42 +080070 Properties propAdded{{PRESENT, PropertyType(true)}};
71 Properties propRemoved{{PRESENT, PropertyType(false)}};
Lei YUf77189f2019-08-07 14:26:30 +080072};
73
74TEST_F(TestItemUpdater, ctordtor)
75{
Lei YU65207482019-10-11 16:39:36 +080076 EXPECT_CALL(mockedUtils, getLatestVersion(_)).Times(1);
Lei YUf77189f2019-08-07 14:26:30 +080077 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
78}
79
80TEST_F(TestItemUpdater, NotCreateObjectOnNotPresent)
81{
82 constexpr auto psuPath = "/com/example/inventory/psu0";
83 constexpr auto service = "com.example.Software.Psu";
84 constexpr auto version = "version0";
85 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -060086 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUf77189f2019-08-07 14:26:30 +080087 .WillOnce(Return(std::vector<std::string>({psuPath})));
88 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
89 .WillOnce(Return(service));
Lei YUf77189f2019-08-07 14:26:30 +080090 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
91 _, StrEq(PRESENT)))
92 .WillOnce(Return(any(PropertyType(false)))); // not present
93
94 // The item updater itself
95 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
96 .Times(1);
97
98 // No activation/version objects are created
99 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
100 .Times(0);
101 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
102}
103
104TEST_F(TestItemUpdater, CreateOnePSUOnPresent)
105{
106 constexpr auto psuPath = "/com/example/inventory/psu0";
107 constexpr auto service = "com.example.Software.Psu";
108 constexpr auto version = "version0";
109 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600110 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUf77189f2019-08-07 14:26:30 +0800111 .WillOnce(Return(std::vector<std::string>({psuPath})));
112 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
113 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800114 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
115 .WillOnce(Return(std::string(version)));
Lei YUf77189f2019-08-07 14:26:30 +0800116 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
117 _, StrEq(PRESENT)))
118 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600119 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
120 .WillOnce(Return(std::string("dummyModel")));
Lei YUf77189f2019-08-07 14:26:30 +0800121
122 // The item updater itself
123 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
124 .Times(1);
125
126 // activation and version object will be added
127 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
128 .Times(2);
129 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
130}
131
132TEST_F(TestItemUpdater, CreateTwoPSUsWithSameVersion)
133{
134 constexpr auto psu0 = "/com/example/inventory/psu0";
135 constexpr auto psu1 = "/com/example/inventory/psu1";
136 constexpr auto service = "com.example.Software.Psu";
137 auto version0 = std::string("version0");
138 auto version1 = std::string("version0");
139 auto objPath0 = getObjPath(version0);
140 auto objPath1 = getObjPath(version1);
141
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600142 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUf77189f2019-08-07 14:26:30 +0800143 .WillOnce(Return(std::vector<std::string>({psu0, psu1})));
144 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu0), _))
145 .WillOnce(Return(service));
146 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu1), _))
147 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800148 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
149 .WillOnce(Return(std::string(version0)));
Lei YUf77189f2019-08-07 14:26:30 +0800150 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu0), _,
151 StrEq(PRESENT)))
152 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600153 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
154 .WillOnce(Return(std::string("dummyModel0")));
Lei YU5f3584d2019-08-27 16:28:53 +0800155 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
156 .WillOnce(Return(std::string(version1)));
Lei YUf77189f2019-08-07 14:26:30 +0800157 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu1), _,
158 StrEq(PRESENT)))
159 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600160 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
161 .WillOnce(Return(std::string("dummyModel1")));
Lei YUf77189f2019-08-07 14:26:30 +0800162
163 // The item updater itself
164 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
165 .Times(1);
166
167 // activation and version object will be added
168 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath0)))
169 .Times(2);
170 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
171
172 // Verify there is only one activation and it has two associations
173 const auto& activations = GetActivations();
George Liu22c2fbd2024-08-23 15:23:21 +0800174 EXPECT_EQ(1U, activations.size());
Lei YUf77189f2019-08-07 14:26:30 +0800175 const auto& activation = activations.find(version0)->second;
176 const auto& assocs = activation->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800177 EXPECT_EQ(2U, assocs.size());
Lei YUf77189f2019-08-07 14:26:30 +0800178 EXPECT_EQ(psu0, std::get<2>(assocs[0]));
179 EXPECT_EQ(psu1, std::get<2>(assocs[1]));
180}
181
182TEST_F(TestItemUpdater, CreateTwoPSUsWithDifferentVersion)
183{
184 constexpr auto psu0 = "/com/example/inventory/psu0";
185 constexpr auto psu1 = "/com/example/inventory/psu1";
186 constexpr auto service = "com.example.Software.Psu";
187 auto version0 = std::string("version0");
188 auto version1 = std::string("version1");
189 auto objPath0 = getObjPath(version0);
190 auto objPath1 = getObjPath(version1);
191
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600192 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUf77189f2019-08-07 14:26:30 +0800193 .WillOnce(Return(std::vector<std::string>({psu0, psu1})));
194 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu0), _))
195 .WillOnce(Return(service));
196 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu1), _))
197 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800198 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
199 .WillOnce(Return(std::string(version0)));
Lei YUf77189f2019-08-07 14:26:30 +0800200 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu0), _,
201 StrEq(PRESENT)))
202 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600203 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
204 .WillOnce(Return(std::string("dummyModel0")));
Lei YU5f3584d2019-08-27 16:28:53 +0800205 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
206 .WillOnce(Return(std::string(version1)));
Lei YUf77189f2019-08-07 14:26:30 +0800207 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu1), _,
208 StrEq(PRESENT)))
209 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600210 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
211 .WillOnce(Return(std::string("dummyModel1")));
Lei YUf77189f2019-08-07 14:26:30 +0800212
213 // The item updater itself
214 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
215 .Times(1);
216
217 // two new activation and version objects will be added
218 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath0)))
219 .Times(2);
220 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath1)))
221 .Times(2);
222 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
223
224 // Verify there are two activations and each with one association
225 const auto& activations = GetActivations();
George Liu22c2fbd2024-08-23 15:23:21 +0800226 EXPECT_EQ(2U, activations.size());
Lei YUf77189f2019-08-07 14:26:30 +0800227 const auto& activation0 = activations.find(version0)->second;
228 const auto& assocs0 = activation0->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800229 EXPECT_EQ(1U, assocs0.size());
Lei YUf77189f2019-08-07 14:26:30 +0800230 EXPECT_EQ(psu0, std::get<2>(assocs0[0]));
231
232 const auto& activation1 = activations.find(version1)->second;
233 const auto& assocs1 = activation1->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800234 EXPECT_EQ(1U, assocs1.size());
Lei YUf77189f2019-08-07 14:26:30 +0800235 EXPECT_EQ(psu1, std::get<2>(assocs1[0]));
236}
Lei YUa2c2cd72019-08-09 15:54:10 +0800237
238TEST_F(TestItemUpdater, OnOnePSURemoved)
239{
240 constexpr auto psuPath = "/com/example/inventory/psu0";
241 constexpr auto service = "com.example.Software.Psu";
242 constexpr auto version = "version0";
243 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600244 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUa2c2cd72019-08-09 15:54:10 +0800245 .WillOnce(Return(std::vector<std::string>({psuPath})));
246 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
247 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800248 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
249 .WillOnce(Return(std::string(version)));
Lei YUa2c2cd72019-08-09 15:54:10 +0800250 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
251 _, StrEq(PRESENT)))
252 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600253 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
254 .WillOnce(Return(std::string("dummyModel")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800255
256 // The item updater itself
257 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
258 .Times(1);
259
260 // activation and version object will be added
261 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
262 .Times(2);
263 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
264
265 // the activation and version object will be removed
266 Properties p{{PRESENT, PropertyType(false)}};
267 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath)))
268 .Times(2);
269 onPsuInventoryChanged(psuPath, p);
270
271 // on exit, item updater is removed
272 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(dBusPath)))
273 .Times(1);
274}
275
276TEST_F(TestItemUpdater, OnOnePSUAdded)
277{
278 constexpr auto psuPath = "/com/example/inventory/psu0";
279 constexpr auto service = "com.example.Software.Psu";
280 constexpr auto version = "version0";
281 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600282 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUa2c2cd72019-08-09 15:54:10 +0800283 .WillOnce(Return(std::vector<std::string>({psuPath})));
284 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
285 .WillOnce(Return(service));
Lei YUa2c2cd72019-08-09 15:54:10 +0800286 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
287 _, StrEq(PRESENT)))
288 .WillOnce(Return(any(PropertyType(false)))); // not present
289
290 // The item updater itself
291 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
292 .Times(1);
293
294 // No activation/version objects are created
295 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
296 .Times(0);
297 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
298
299 // The PSU is present and version is added in a single call
Shawn McCarney783406e2024-11-17 21:49:37 -0600300 Properties propAdded{{PRESENT, PropertyType(true)}};
Lei YUdcaf8932019-09-09 16:09:35 +0800301 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
302 .WillOnce(Return(std::string(version)));
Shawn McCarney783406e2024-11-17 21:49:37 -0600303 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
304 .WillOnce(Return(std::string("testModel")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800305 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
306 .Times(2);
Shawn McCarney783406e2024-11-17 21:49:37 -0600307 onPsuInventoryChanged(psuPath, propAdded);
Lei YUa2c2cd72019-08-09 15:54:10 +0800308}
309
Lei YU1517f5f2019-10-14 16:44:42 +0800310TEST_F(TestItemUpdater, OnOnePSURemovedAndAddedWithLatestVersion)
Lei YUa2c2cd72019-08-09 15:54:10 +0800311{
312 constexpr auto psuPath = "/com/example/inventory/psu0";
313 constexpr auto service = "com.example.Software.Psu";
314 constexpr auto version = "version0";
315 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600316 ON_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YU1517f5f2019-10-14 16:44:42 +0800317 .WillByDefault(Return(std::vector<std::string>({psuPath})));
Lei YUa2c2cd72019-08-09 15:54:10 +0800318 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
319 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800320 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
321 .WillOnce(Return(std::string(version)));
Lei YUa2c2cd72019-08-09 15:54:10 +0800322 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
323 _, StrEq(PRESENT)))
324 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600325 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
326 .WillOnce(Return(std::string("dummyModel")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800327
328 // The item updater itself
329 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
330 .Times(1);
331
332 // activation and version object will be added
333 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
334 .Times(2);
335 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
336
337 // the activation and version object will be removed
Lei YUa2c2cd72019-08-09 15:54:10 +0800338 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath)))
339 .Times(2);
340 onPsuInventoryChanged(psuPath, propRemoved);
341
Lei YUdcaf8932019-09-09 16:09:35 +0800342 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
343 .WillOnce(Return(std::string(version)));
Shawn McCarney783406e2024-11-17 21:49:37 -0600344 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
345 .WillOnce(Return(std::string("dummyModel")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800346 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
347 .Times(2);
Lei YU1517f5f2019-10-14 16:44:42 +0800348
349 // On PSU inserted, it shall check if it's the latest version
350 std::set<std::string> expectedVersions = {version};
351 EXPECT_CALL(mockedUtils, getLatestVersion(ContainerEq(expectedVersions)))
352 .WillOnce(Return(version));
353 EXPECT_CALL(mockedUtils, isAssociated(StrEq(psuPath), _))
354 .WillOnce(Return(true));
355 EXPECT_CALL(sdbusMock, sd_bus_message_new_method_call(_, _, _, _, _,
356 StrEq("StartUnit")))
357 .Times(0);
Lei YUa2c2cd72019-08-09 15:54:10 +0800358 onPsuInventoryChanged(psuPath, propAdded);
Lei YUa2c2cd72019-08-09 15:54:10 +0800359
360 // on exit, objects are removed
361 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath)))
362 .Times(2);
363 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(dBusPath)))
364 .Times(1);
365}
366
367TEST_F(TestItemUpdater,
368 TwoPSUsWithSameVersionRemovedAndAddedWithDifferntVersion)
369{
370 constexpr auto psu0 = "/com/example/inventory/psu0";
371 constexpr auto psu1 = "/com/example/inventory/psu1";
372 constexpr auto service = "com.example.Software.Psu";
373 auto version0 = std::string("version0");
374 auto version1 = std::string("version0");
375 auto objPath0 = getObjPath(version0);
376 auto objPath1 = getObjPath(version1);
377
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600378 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUa2c2cd72019-08-09 15:54:10 +0800379 .WillOnce(Return(std::vector<std::string>({psu0, psu1})));
380 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu0), _))
381 .WillOnce(Return(service));
382 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu1), _))
383 .WillOnce(Return(service));
Lei YU5f3584d2019-08-27 16:28:53 +0800384 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
385 .WillOnce(Return(std::string(version0)));
Lei YUa2c2cd72019-08-09 15:54:10 +0800386 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu0), _,
387 StrEq(PRESENT)))
388 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600389 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
390 .WillOnce(Return(std::string("dummyModel0")));
Lei YU5f3584d2019-08-27 16:28:53 +0800391 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
392 .WillOnce(Return(std::string(version1)));
Lei YUa2c2cd72019-08-09 15:54:10 +0800393 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu1), _,
394 StrEq(PRESENT)))
395 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600396 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
397 .WillOnce(Return(std::string("dummyModel1")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800398
399 // The item updater itself
400 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
401 .Times(1);
402
403 // activation and version object will be added
404 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath0)))
405 .Times(2);
406 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
407
408 // Verify there is only one activation and it has two associations
409 const auto& activations = GetActivations();
George Liu22c2fbd2024-08-23 15:23:21 +0800410 EXPECT_EQ(1U, activations.size());
Lei YUa2c2cd72019-08-09 15:54:10 +0800411 const auto& activation = activations.find(version0)->second;
412 auto assocs = activation->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800413 EXPECT_EQ(2U, assocs.size());
Lei YUa2c2cd72019-08-09 15:54:10 +0800414 EXPECT_EQ(psu0, std::get<2>(assocs[0]));
415 EXPECT_EQ(psu1, std::get<2>(assocs[1]));
416
417 // PSU0 is removed, only associations shall be updated
Lei YUa2c2cd72019-08-09 15:54:10 +0800418 onPsuInventoryChanged(psu0, propRemoved);
419 assocs = activation->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800420 EXPECT_EQ(1U, assocs.size());
Lei YUa2c2cd72019-08-09 15:54:10 +0800421 EXPECT_EQ(psu1, std::get<2>(assocs[0]));
422
423 // PSU1 is removed, the activation and version object shall be removed
424 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath0)))
425 .Times(2);
426 onPsuInventoryChanged(psu1, propRemoved);
427
428 // Add PSU0 and PSU1 back, but PSU1 with a different version
429 version1 = "version1";
430 objPath1 = getObjPath(version1);
Lei YUdcaf8932019-09-09 16:09:35 +0800431 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
432 .WillOnce(Return(std::string(version0)));
Shawn McCarney783406e2024-11-17 21:49:37 -0600433 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
434 .WillOnce(Return(std::string("dummyModel0")));
Lei YUdcaf8932019-09-09 16:09:35 +0800435 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
436 .WillOnce(Return(std::string(version1)));
Shawn McCarney783406e2024-11-17 21:49:37 -0600437 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
438 .WillOnce(Return(std::string("dummyModel1")));
Lei YUa2c2cd72019-08-09 15:54:10 +0800439 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath0)))
440 .Times(2);
441 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath1)))
442 .Times(2);
Lei YU1517f5f2019-10-14 16:44:42 +0800443 onPsuInventoryChanged(psu0, propAdded);
Lei YU1517f5f2019-10-14 16:44:42 +0800444 onPsuInventoryChanged(psu1, propAdded);
Lei YUa2c2cd72019-08-09 15:54:10 +0800445
446 // on exit, objects are removed
447 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath0)))
448 .Times(2);
449 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(objPath1)))
450 .Times(2);
451 EXPECT_CALL(sdbusMock, sd_bus_emit_object_removed(_, StrEq(dBusPath)))
452 .Times(1);
453}
Lei YU58c26e32019-09-27 17:52:06 +0800454
455TEST_F(TestItemUpdater, scanDirOnNoPSU)
456{
457 constexpr auto psuPath = "/com/example/inventory/psu0";
458 constexpr auto service = "com.example.Software.Psu";
459 constexpr auto version = "version0";
460 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600461 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YU58c26e32019-09-27 17:52:06 +0800462 .WillOnce(Return(std::vector<std::string>({psuPath})));
463 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
464 .WillOnce(Return(service));
Lei YU58c26e32019-09-27 17:52:06 +0800465 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
466 _, StrEq(PRESENT)))
467 .WillOnce(Return(any(PropertyType(false)))); // not present
468
469 // The item updater itself
470 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
471 .Times(1);
472
473 // No activation/version objects are created
474 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
475 .Times(0);
476 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
477
478 // The valid image in test/psu-images-one-valid-one-invalid/model-1/
479 auto objPathValid = getObjPath("psu-test.v0.4");
480 auto objPathInvalid = getObjPath("psu-test.v0.5");
Shawn McCarney487e2e12024-11-25 17:19:46 -0600481 // No activation or version objects will be added on scan dir
Lei YU58c26e32019-09-27 17:52:06 +0800482 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPathValid)))
Faisal Awada760053d2024-05-16 13:31:32 -0500483 .Times(0);
Lei YU58c26e32019-09-27 17:52:06 +0800484 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPathInvalid)))
485 .Times(0);
486 scanDirectory("./psu-images-one-valid-one-invalid");
487}
488
489TEST_F(TestItemUpdater, scanDirOnSamePSUVersion)
490{
491 constexpr auto psuPath = "/com/example/inventory/psu0";
492 constexpr auto service = "com.example.Software.Psu";
493 constexpr auto version = "version0";
494 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600495 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YU58c26e32019-09-27 17:52:06 +0800496 .WillOnce(Return(std::vector<std::string>({psuPath})));
497 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
498 .WillOnce(Return(service));
499 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
500 .WillOnce(Return(std::string(version)));
501 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
502 _, StrEq(PRESENT)))
503 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600504 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
Shawn McCarney487e2e12024-11-25 17:19:46 -0600505 .WillOnce(Return(std::string("model-3")));
Lei YU58c26e32019-09-27 17:52:06 +0800506
507 // The item updater itself
508 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(dBusPath)))
509 .Times(1);
510
511 // activation and version object will be added
512 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
513 .Times(2);
514 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
515
516 // The valid image in test/psu-images-valid-version0/model-3/ has the same
517 // version as the running PSU, so no objects will be created, but only the
518 // path will be set to the version object
519 EXPECT_CALL(sdbusMock, sd_bus_emit_object_added(_, StrEq(objPath)))
520 .Times(0);
521 EXPECT_CALL(sdbusMock, sd_bus_emit_properties_changed_strv(
522 _, StrEq(objPath),
523 StrEq("xyz.openbmc_project.Common.FilePath"),
524 Pointee(StrEq("Path"))))
Shawn McCarney487e2e12024-11-25 17:19:46 -0600525 .Times(1);
Lei YU58c26e32019-09-27 17:52:06 +0800526 scanDirectory("./psu-images-valid-version0");
527}
Lei YUffb36532019-10-15 13:55:24 +0800528
529TEST_F(TestItemUpdater, OnUpdateDoneOnTwoPSUsWithSameVersion)
530{
531 // Simulate there are two PSUs with same version, and updated to a new
532 // version
533 constexpr auto psu0 = "/com/example/inventory/psu0";
534 constexpr auto psu1 = "/com/example/inventory/psu1";
535 constexpr auto service = "com.example.Software.Psu";
536 auto version0 = std::string("version0");
537 auto version1 = std::string("version0");
538 auto objPath0 = getObjPath(version0);
539 auto objPath1 = getObjPath(version1);
540
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600541 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUffb36532019-10-15 13:55:24 +0800542 .WillOnce(Return(std::vector<std::string>({psu0, psu1})));
543 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu0), _))
544 .WillOnce(Return(service));
545 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu1), _))
546 .WillOnce(Return(service));
547 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
548 .WillOnce(Return(std::string(version0)));
549 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu0), _,
550 StrEq(PRESENT)))
551 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600552 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
553 .WillOnce(Return(std::string("dummyModel0")));
Lei YUffb36532019-10-15 13:55:24 +0800554 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
555 .WillOnce(Return(std::string(version1)));
556 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu1), _,
557 StrEq(PRESENT)))
558 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600559 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
560 .WillOnce(Return(std::string("dummyModel1")));
Lei YUffb36532019-10-15 13:55:24 +0800561
562 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
563
Lei YU1517f5f2019-10-14 16:44:42 +0800564 std::string newVersionId = "NewVersionId";
565 AssociationList associations;
566 auto dummyActivation = std::make_unique<Activation>(
567 mockedBus, dBusPath, newVersionId, "", Activation::Status::Active,
568 associations, "", itemUpdater.get(), itemUpdater.get());
569
Lei YUffb36532019-10-15 13:55:24 +0800570 // Now there is one activation and it has two associations
571 auto& activations = GetActivations();
Lei YU1517f5f2019-10-14 16:44:42 +0800572 activations.emplace(newVersionId, std::move(dummyActivation));
Lei YUffb36532019-10-15 13:55:24 +0800573 auto& activation = activations.find(version0)->second;
574 auto assocs = activation->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800575 EXPECT_EQ(2U, assocs.size());
Lei YUffb36532019-10-15 13:55:24 +0800576 EXPECT_EQ(psu0, std::get<2>(assocs[0]));
577 EXPECT_EQ(psu1, std::get<2>(assocs[1]));
578
579 EXPECT_CALL(mockedUtils, isAssociated(StrEq(psu0), _))
580 .WillOnce(Return(true));
Lei YU1517f5f2019-10-14 16:44:42 +0800581 itemUpdater->onUpdateDone(newVersionId, psu0);
Lei YUffb36532019-10-15 13:55:24 +0800582
Manojkiran Eda33cf9f02024-06-17 14:40:44 +0530583 // Now the activation should have one association
Lei YUffb36532019-10-15 13:55:24 +0800584 assocs = activation->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800585 EXPECT_EQ(1U, assocs.size());
Lei YUffb36532019-10-15 13:55:24 +0800586 EXPECT_EQ(psu1, std::get<2>(assocs[0]));
587
588 EXPECT_CALL(mockedUtils, isAssociated(StrEq(psu1), _))
589 .WillOnce(Return(true));
Lei YU1517f5f2019-10-14 16:44:42 +0800590 itemUpdater->onUpdateDone(newVersionId, psu1);
Lei YUffb36532019-10-15 13:55:24 +0800591
Lei YU1517f5f2019-10-14 16:44:42 +0800592 // Now the activation shall be erased and only the dummy one is left
George Liu22c2fbd2024-08-23 15:23:21 +0800593 EXPECT_EQ(1U, activations.size());
Lei YU1517f5f2019-10-14 16:44:42 +0800594 EXPECT_NE(activations.find(newVersionId), activations.end());
Lei YUffb36532019-10-15 13:55:24 +0800595}
596
597TEST_F(TestItemUpdater, OnUpdateDoneOnTwoPSUsWithDifferentVersion)
598{
599 // Simulate there are two PSUs with different version, and updated to a new
600 // version
601 constexpr auto psu0 = "/com/example/inventory/psu0";
602 constexpr auto psu1 = "/com/example/inventory/psu1";
603 constexpr auto service = "com.example.Software.Psu";
604 auto version0 = std::string("version0");
605 auto version1 = std::string("version1");
606 auto objPath0 = getObjPath(version0);
607 auto objPath1 = getObjPath(version1);
608
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600609 EXPECT_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YUffb36532019-10-15 13:55:24 +0800610 .WillOnce(Return(std::vector<std::string>({psu0, psu1})));
611 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu0), _))
612 .WillOnce(Return(service));
613 EXPECT_CALL(mockedUtils, getService(_, StrEq(psu1), _))
614 .WillOnce(Return(service));
615 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu0)))
616 .WillOnce(Return(std::string(version0)));
617 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu0), _,
618 StrEq(PRESENT)))
619 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600620 EXPECT_CALL(mockedUtils, getModel(StrEq(psu0)))
621 .WillOnce(Return(std::string("dummyModel0")));
Lei YUffb36532019-10-15 13:55:24 +0800622 EXPECT_CALL(mockedUtils, getVersion(StrEq(psu1)))
623 .WillOnce(Return(std::string(version1)));
624 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psu1), _,
625 StrEq(PRESENT)))
626 .WillOnce(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600627 EXPECT_CALL(mockedUtils, getModel(StrEq(psu1)))
628 .WillOnce(Return(std::string("dummyModel1")));
Lei YUffb36532019-10-15 13:55:24 +0800629
630 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
631
Lei YU1517f5f2019-10-14 16:44:42 +0800632 std::string newVersionId = "NewVersionId";
633 AssociationList associations;
634 auto dummyActivation = std::make_unique<Activation>(
635 mockedBus, dBusPath, newVersionId, "", Activation::Status::Active,
636 associations, "", itemUpdater.get(), itemUpdater.get());
637
638 auto& activations = GetActivations();
639 activations.emplace(newVersionId, std::move(dummyActivation));
Lei YUffb36532019-10-15 13:55:24 +0800640
641 EXPECT_CALL(mockedUtils, isAssociated(StrEq(psu0), _))
642 .WillOnce(Return(true));
Lei YU1517f5f2019-10-14 16:44:42 +0800643
644 // After psu0 is done, two activations should be left
645 itemUpdater->onUpdateDone(newVersionId, psu0);
George Liu22c2fbd2024-08-23 15:23:21 +0800646 EXPECT_EQ(2U, activations.size());
Lei YUffb36532019-10-15 13:55:24 +0800647 const auto& activation1 = activations.find(version1)->second;
648 const auto& assocs1 = activation1->associations();
George Liu22c2fbd2024-08-23 15:23:21 +0800649 EXPECT_EQ(1U, assocs1.size());
Lei YUffb36532019-10-15 13:55:24 +0800650 EXPECT_EQ(psu1, std::get<2>(assocs1[0]));
651
652 EXPECT_CALL(mockedUtils, isAssociated(StrEq(psu1), _))
653 .WillOnce(Return(true));
Lei YU1517f5f2019-10-14 16:44:42 +0800654 // After psu1 is done, only the dummy activation should be left
655 itemUpdater->onUpdateDone(newVersionId, psu1);
George Liu22c2fbd2024-08-23 15:23:21 +0800656 EXPECT_EQ(1U, activations.size());
Lei YU1517f5f2019-10-14 16:44:42 +0800657 EXPECT_NE(activations.find(newVersionId), activations.end());
658}
659
660TEST_F(TestItemUpdater, OnOnePSURemovedAndAddedWithOldVersion)
661{
662 constexpr auto psuPath = "/com/example/inventory/psu0";
663 constexpr auto service = "com.example.Software.Psu";
664 constexpr auto version = "version0";
665 std::string versionId =
666 version; // In testing versionId is the same as version
667 std::string objPath = getObjPath(version);
Shawn McCarneyd57bd2f2024-12-02 18:40:28 -0600668 ON_CALL(mockedUtils, getPSUInventoryPaths(_))
Lei YU1517f5f2019-10-14 16:44:42 +0800669 .WillByDefault(Return(std::vector<std::string>({psuPath})));
Faisal Awada760053d2024-05-16 13:31:32 -0500670 EXPECT_CALL(mockedUtils, getService(_, StrEq(psuPath), _))
671 .WillOnce(Return(service))
672 .WillOnce(Return(service));
Lei YU1517f5f2019-10-14 16:44:42 +0800673 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
674 .WillOnce(Return(std::string(version)));
675 ON_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath), _,
676 StrEq(PRESENT)))
677 .WillByDefault(Return(any(PropertyType(true)))); // present
Shawn McCarney783406e2024-11-17 21:49:37 -0600678 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
679 .WillOnce(Return(std::string("dummyModel")));
680
Lei YU1517f5f2019-10-14 16:44:42 +0800681 itemUpdater = std::make_unique<ItemUpdater>(mockedBus, dBusPath);
682
683 // Add an association to simulate that it has image in BMC filesystem
684 auto& activation = GetActivations().find(versionId)->second;
685 auto assocs = activation->associations();
686 assocs.emplace_back(ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION,
687 "SomePath");
688 activation->associations(assocs);
689 activation->path("SomeFilePath");
690
691 onPsuInventoryChanged(psuPath, propRemoved);
692
693 // On PSU inserted, it checks and finds a newer version
694 auto oldVersion = "old-version";
695 EXPECT_CALL(mockedUtils, getVersion(StrEq(psuPath)))
696 .WillOnce(Return(std::string(oldVersion)));
697 EXPECT_CALL(mockedUtils, getPropertyImpl(_, StrEq(service), StrEq(psuPath),
698 _, StrEq(MANUFACTURER)))
699 .WillOnce(
700 Return(any(PropertyType(std::string(""))))); // Checking compatible
Shawn McCarney783406e2024-11-17 21:49:37 -0600701 EXPECT_CALL(mockedUtils, getModel(StrEq(psuPath)))
702 .WillOnce(Return(std::string("")))
703 .WillOnce(Return(std::string("")));
Lei YU1517f5f2019-10-14 16:44:42 +0800704 std::set<std::string> expectedVersions = {version, oldVersion};
705 EXPECT_CALL(mockedUtils, getLatestVersion(ContainerEq(expectedVersions)))
706 .WillOnce(Return(version));
707 ON_CALL(mockedUtils, isAssociated(StrEq(psuPath), _))
708 .WillByDefault(Return(false));
709 EXPECT_CALL(sdbusMock, sd_bus_message_new_method_call(_, _, _, _, _,
710 StrEq("StartUnit")))
Lei YU8afeee52019-10-21 15:25:35 +0800711 .Times(3); // There are 3 systemd units are started, enable bmc reboot
712 // guard, start activation, and disable bmc reboot guard
Lei YU1517f5f2019-10-14 16:44:42 +0800713 onPsuInventoryChanged(psuPath, propAdded);
Lei YUffb36532019-10-15 13:55:24 +0800714}