blob: 93125980cc663510fdc4f3709ce81d7d4298f2be [file] [log] [blame]
Andrew Jefferya607a0d2018-05-23 16:15:26 +09301#include "physical.hpp"
2
Andrew Jeffery280afaf2018-05-24 16:15:46 +09303#include <sys/param.h>
4
Andrew Jefferya607a0d2018-05-23 16:15:26 +09305#include <sdbusplus/bus.hpp>
6
Andrew Jeffery280afaf2018-05-24 16:15:46 +09307#include <gmock/gmock.h>
Andrew Jefferya607a0d2018-05-23 16:15:26 +09308#include <gtest/gtest.h>
9
10constexpr auto LED_OBJ = "/foo/bar/led";
Andrew Jefferya607a0d2018-05-23 16:15:26 +093011
12using Action = sdbusplus::xyz::openbmc_project::Led::server::Physical::Action;
Andrew Jeffery42e02d32018-05-24 13:34:05 +093013namespace fs = std::experimental::filesystem;
Andrew Jefferya607a0d2018-05-23 16:15:26 +093014
Andrew Jeffery280afaf2018-05-24 16:15:46 +093015fs::path create_sandbox()
16{
17 /* If your tests need to touch the filesystem, always use mkdtemp() or
18 * mkstemp() for creating directories and files. Tests can be run in
19 * parallel with `make -j`, and if use the same path in multiple tests they
20 * will stomp on eachother and likely fail.
21 */
22 static constexpr auto tmplt = "/tmp/MockLed.XXXXXX";
23 char buffer[MAXPATHLEN] = {0};
24
25 strncpy(buffer, tmplt, sizeof(buffer) - 1);
26 auto dir = mkdtemp(buffer);
27 if (!dir)
28 {
29 throw std::system_error(errno, std::system_category());
30 }
31
32 /* We want to limit behaviours to mocks, and if methods aren't mocked they
33 * may fall back to their base class implementation. Stop read/write to
34 * directory to prevent streams from creating files.
35 */
36 if (chmod(dir, S_IXUSR | S_IXGRP) == -1)
37 {
38 throw std::system_error(errno, std::system_category());
39 }
40
41 return fs::path(dir);
42}
43
44class MockLed : public phosphor::led::SysfsLed
45{
46 public:
47 /* Use a no-args ctor here to avoid headaches with {Nice,Strict}Mock */
48 MockLed() : SysfsLed(create_sandbox())
49 {
50 }
51
52 virtual ~MockLed()
53 {
54 chmod(root.c_str(), S_IRUSR | S_IWUSR | S_IXUSR);
55 fs::remove_all(root);
56 }
57
58 MOCK_METHOD0(getBrightness, unsigned long());
59 MOCK_METHOD1(setBrightness, void(unsigned long value));
60 MOCK_METHOD0(getMaxBrightness, unsigned long());
61 MOCK_METHOD0(getTrigger, std::string());
62 MOCK_METHOD1(setTrigger, void(const std::string& trigger));
63 MOCK_METHOD0(getDelayOn, unsigned long());
64 MOCK_METHOD1(setDelayOn, void(unsigned long ms));
65 MOCK_METHOD0(getDelayOff, unsigned long());
66 MOCK_METHOD1(setDelayOff, void(unsigned long ms));
67};
68
69using ::testing::NiceMock;
70using ::testing::Return;
71using ::testing::Throw;
72
Andrew Jeffery56692702018-05-25 11:41:43 +093073TEST(Physical, ctor_none_trigger)
Andrew Jefferya607a0d2018-05-23 16:15:26 +093074{
75 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
Andrew Jeffery280afaf2018-05-24 16:15:46 +093076 /* NiceMock ignores calls to methods with no expectations defined */
77 NiceMock<MockLed> led;
78 ON_CALL(led, getTrigger()).WillByDefault(Return("none"));
Andrew Jeffery42e02d32018-05-24 13:34:05 +093079 phosphor::led::Physical phy(bus, LED_OBJ, led);
Andrew Jefferya607a0d2018-05-23 16:15:26 +093080}
81
Andrew Jeffery56692702018-05-25 11:41:43 +093082TEST(Physical, ctor_timer_trigger)
83{
84 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
85 NiceMock<MockLed> led;
86 EXPECT_CALL(led, getTrigger()).WillOnce(Return("timer"));
87 EXPECT_CALL(led, getDelayOn()).WillOnce(Return(500));
88 EXPECT_CALL(led, getDelayOff()).WillOnce(Return(500));
89 phosphor::led::Physical phy(bus, LED_OBJ, led);
90}
91
Andrew Jefferya607a0d2018-05-23 16:15:26 +093092TEST(Physical, off)
93{
94 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
Andrew Jeffery280afaf2018-05-24 16:15:46 +093095 NiceMock<MockLed> led;
96 ON_CALL(led, getTrigger()).WillByDefault(Return("none"));
Andrew Jeffery42e02d32018-05-24 13:34:05 +093097 phosphor::led::Physical phy(bus, LED_OBJ, led);
98 phy.state(Action::Off);
Andrew Jefferya607a0d2018-05-23 16:15:26 +093099}
100
101TEST(Physical, on)
102{
103 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
Andrew Jeffery280afaf2018-05-24 16:15:46 +0930104 NiceMock<MockLed> led;
105 ON_CALL(led, getTrigger()).WillByDefault(Return("none"));
Andrew Jeffery42e02d32018-05-24 13:34:05 +0930106 phosphor::led::Physical phy(bus, LED_OBJ, led);
107 phy.state(Action::On);
Andrew Jefferya607a0d2018-05-23 16:15:26 +0930108}
109
110TEST(Physical, blink)
111{
112 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
Andrew Jeffery280afaf2018-05-24 16:15:46 +0930113 NiceMock<MockLed> led;
114 ON_CALL(led, getTrigger()).WillByDefault(Return("none"));
Andrew Jeffery42e02d32018-05-24 13:34:05 +0930115 phosphor::led::Physical phy(bus, LED_OBJ, led);
116 phy.state(Action::Blink);
Andrew Jefferya607a0d2018-05-23 16:15:26 +0930117}
Andrew Jeffery04275e02018-05-25 13:04:39 +0930118
119TEST(Physical, ctor_none_trigger_asserted_brightness)
120{
121 sdbusplus::bus::bus bus = sdbusplus::bus::new_default();
122 NiceMock<MockLed> led;
123 EXPECT_CALL(led, getTrigger()).WillRepeatedly(Return("none"));
124 constexpr auto val = phosphor::led::ASSERT;
125 EXPECT_CALL(led, getBrightness()).WillRepeatedly(Return(val));
126 phosphor::led::Physical phy(bus, LED_OBJ, led);
127}