test_dbus.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. # systemctl-mqtt - MQTT client triggering & reporting shutdown on systemd-based systems
  2. #
  3. # Copyright (C) 2020 Fabian Peter Hammerle <fabian@hammerle.me>
  4. #
  5. # This program is free software: you can redistribute it and/or modify
  6. # it under the terms of the GNU General Public License as published by
  7. # the Free Software Foundation, either version 3 of the License, or
  8. # any later version.
  9. #
  10. # This program is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. # GNU General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU General Public License
  16. # along with this program. If not, see <https://www.gnu.org/licenses/>.
  17. import asyncio
  18. import datetime
  19. import logging
  20. import typing
  21. import unittest.mock
  22. import jeepney
  23. import jeepney.low_level
  24. import jeepney.wrappers
  25. import pytest
  26. import systemctl_mqtt._dbus.login_manager
  27. # pylint: disable=protected-access
  28. def test_get_login_manager_proxy():
  29. login_manager = systemctl_mqtt._dbus.login_manager.get_login_manager_proxy()
  30. assert isinstance(login_manager, jeepney.io.blocking.Proxy)
  31. assert login_manager._msggen.interface == "org.freedesktop.login1.Manager"
  32. # https://freedesktop.org/wiki/Software/systemd/logind/
  33. assert login_manager.CanPowerOff() in {("yes",), ("challenge",)}
  34. def test__log_shutdown_inhibitors_some(caplog):
  35. login_manager = unittest.mock.MagicMock()
  36. login_manager.ListInhibitors.return_value = (
  37. [
  38. (
  39. "shutdown:sleep",
  40. "Developer",
  41. "Haven't pushed my commits yet",
  42. "delay",
  43. 1000,
  44. 1234,
  45. ),
  46. ("shutdown", "Editor", "", "Unsafed files open", 0, 42),
  47. ],
  48. )
  49. with caplog.at_level(logging.DEBUG):
  50. systemctl_mqtt._dbus.login_manager._log_shutdown_inhibitors(login_manager)
  51. assert len(caplog.records) == 2
  52. assert caplog.records[0].levelno == logging.DEBUG
  53. assert (
  54. caplog.records[0].message
  55. == "detected shutdown inhibitor Developer (pid=1234, uid=1000, mode=delay): "
  56. + "Haven't pushed my commits yet"
  57. )
  58. def test__log_shutdown_inhibitors_none(caplog):
  59. login_manager = unittest.mock.MagicMock()
  60. login_manager.ListInhibitors.return_value = ([],)
  61. with caplog.at_level(logging.DEBUG):
  62. systemctl_mqtt._dbus.login_manager._log_shutdown_inhibitors(login_manager)
  63. assert len(caplog.records) == 1
  64. assert caplog.records[0].levelno == logging.DEBUG
  65. assert caplog.records[0].message == "no shutdown inhibitor locks found"
  66. def test__log_shutdown_inhibitors_fail(caplog):
  67. login_manager = unittest.mock.MagicMock()
  68. login_manager.ListInhibitors.side_effect = DBusErrorResponseMock("error", "mocked")
  69. with caplog.at_level(logging.DEBUG):
  70. systemctl_mqtt._dbus.login_manager._log_shutdown_inhibitors(login_manager)
  71. assert len(caplog.records) == 1
  72. assert caplog.records[0].levelno == logging.WARNING
  73. assert (
  74. caplog.records[0].message
  75. == "failed to fetch shutdown inhibitors: [error] mocked"
  76. )
  77. @pytest.mark.parametrize("action", ["poweroff", "reboot"])
  78. @pytest.mark.parametrize("delay", [datetime.timedelta(0), datetime.timedelta(hours=1)])
  79. def test__schedule_shutdown(action, delay):
  80. login_manager_mock = unittest.mock.MagicMock()
  81. with unittest.mock.patch(
  82. "systemctl_mqtt._dbus.login_manager.get_login_manager_proxy",
  83. return_value=login_manager_mock,
  84. ):
  85. login_manager_mock.ListInhibitors.return_value = ([],)
  86. systemctl_mqtt._dbus.login_manager.schedule_shutdown(action=action, delay=delay)
  87. login_manager_mock.ScheduleShutdown.assert_called_once()
  88. schedule_args, schedule_kwargs = login_manager_mock.ScheduleShutdown.call_args
  89. assert not schedule_args
  90. assert schedule_kwargs.pop("action") == action
  91. actual_delay = schedule_kwargs.pop("time") - datetime.datetime.now()
  92. assert actual_delay.total_seconds() == pytest.approx(delay.total_seconds(), abs=0.1)
  93. assert not schedule_kwargs
  94. class DBusErrorResponseMock(jeepney.wrappers.DBusErrorResponse):
  95. # pylint: disable=missing-class-docstring,super-init-not-called
  96. def __init__(self, name: str, data: typing.Any):
  97. self.name = name
  98. self.data = data
  99. @pytest.mark.parametrize("action", ["poweroff"])
  100. @pytest.mark.parametrize(
  101. ("error_name", "error_message", "log_message"),
  102. [
  103. (
  104. "test error",
  105. "test message",
  106. "[test error] ('test message',)",
  107. ),
  108. (
  109. "org.freedesktop.DBus.Error.InteractiveAuthorizationRequired",
  110. "Interactive authentication required.",
  111. "unauthorized; missing polkit authorization rules?",
  112. ),
  113. ],
  114. )
  115. def test__schedule_shutdown_fail(
  116. caplog, action, error_name, error_message, log_message
  117. ):
  118. login_manager_mock = unittest.mock.MagicMock()
  119. login_manager_mock.ScheduleShutdown.side_effect = DBusErrorResponseMock(
  120. name=error_name,
  121. data=(error_message,),
  122. )
  123. login_manager_mock.ListInhibitors.return_value = ([],)
  124. with unittest.mock.patch(
  125. "systemctl_mqtt._dbus.login_manager.get_login_manager_proxy",
  126. return_value=login_manager_mock,
  127. ), caplog.at_level(logging.DEBUG):
  128. systemctl_mqtt._dbus.login_manager.schedule_shutdown(
  129. action=action, delay=datetime.timedelta(seconds=21)
  130. )
  131. login_manager_mock.ScheduleShutdown.assert_called_once()
  132. assert len(caplog.records) == 3
  133. assert caplog.records[0].levelno == logging.INFO
  134. assert caplog.records[0].message.startswith(f"scheduling {action} for ")
  135. assert caplog.records[1].levelno == logging.ERROR
  136. assert caplog.records[1].message == f"failed to schedule {action}: {log_message}"
  137. assert "inhibitor" in caplog.records[2].message
  138. def test_suspend(caplog):
  139. login_manager_mock = unittest.mock.MagicMock()
  140. with unittest.mock.patch(
  141. "systemctl_mqtt._dbus.login_manager.get_login_manager_proxy",
  142. return_value=login_manager_mock,
  143. ), caplog.at_level(logging.INFO):
  144. systemctl_mqtt._dbus.login_manager.suspend()
  145. login_manager_mock.Suspend.assert_called_once_with(interactive=False)
  146. assert len(caplog.records) == 1
  147. assert caplog.records[0].levelno == logging.INFO
  148. assert caplog.records[0].message == "suspending system"
  149. def test_lock_all_sessions(caplog):
  150. login_manager_mock = unittest.mock.MagicMock()
  151. with unittest.mock.patch(
  152. "systemctl_mqtt._dbus.login_manager.get_login_manager_proxy",
  153. return_value=login_manager_mock,
  154. ), caplog.at_level(logging.INFO):
  155. systemctl_mqtt._dbus.login_manager.lock_all_sessions()
  156. login_manager_mock.LockSessions.assert_called_once_with()
  157. assert len(caplog.records) == 1
  158. assert caplog.records[0].levelno == logging.INFO
  159. assert caplog.records[0].message == "instruct all sessions to activate screen locks"
  160. async def _get_unit_path_mock( # pylint: disable=unused-argument
  161. *, service_manager: jeepney.io.asyncio.Proxy, unit_name: str
  162. ) -> str:
  163. return "/org/freedesktop/systemd1/unit/" + unit_name
  164. @pytest.mark.asyncio
  165. @pytest.mark.parametrize(
  166. "monitored_system_unit_names", [[], ["foo.service", "bar.service"]]
  167. )
  168. async def test__dbus_signal_loop(monitored_system_unit_names: typing.List[str]) -> None:
  169. # pylint: disable=too-many-locals,too-many-arguments
  170. state_mock = unittest.mock.AsyncMock()
  171. with unittest.mock.patch(
  172. "jeepney.io.asyncio.open_dbus_router",
  173. ) as open_dbus_router_mock, unittest.mock.patch(
  174. "systemctl_mqtt._get_unit_path", _get_unit_path_mock
  175. ), unittest.mock.patch(
  176. "systemctl_mqtt._dbus_signal_loop_unit"
  177. ) as dbus_signal_loop_unit_mock:
  178. async with open_dbus_router_mock() as dbus_router_mock:
  179. pass
  180. add_match_reply = unittest.mock.Mock()
  181. add_match_reply.body = ()
  182. dbus_router_mock.send_and_get_reply.return_value = add_match_reply
  183. msg_queue: asyncio.Queue[jeepney.low_level.Message] = asyncio.Queue()
  184. await msg_queue.put(jeepney.low_level.Message(header=None, body=(False,)))
  185. await msg_queue.put(jeepney.low_level.Message(header=None, body=(True,)))
  186. await msg_queue.put(jeepney.low_level.Message(header=None, body=(False,)))
  187. dbus_router_mock.filter = unittest.mock.MagicMock()
  188. dbus_router_mock.filter.return_value.__enter__.return_value = msg_queue
  189. state_mock.monitored_system_unit_names = monitored_system_unit_names
  190. # asyncio.TaskGroup added in python3.11
  191. loop_task = asyncio.create_task(
  192. systemctl_mqtt._dbus_signal_loop(
  193. state=state_mock, mqtt_client=unittest.mock.MagicMock()
  194. )
  195. )
  196. async def _abort_after_msg_queue():
  197. await msg_queue.join()
  198. loop_task.cancel()
  199. with pytest.raises(asyncio.exceptions.CancelledError):
  200. await asyncio.gather(*(loop_task, _abort_after_msg_queue()))
  201. assert unittest.mock.call(bus="SYSTEM") in open_dbus_router_mock.call_args_list
  202. dbus_router_mock.filter.assert_called_once()
  203. (filter_match_rule,) = dbus_router_mock.filter.call_args[0]
  204. assert (
  205. filter_match_rule.header_fields["interface"] == "org.freedesktop.login1.Manager"
  206. )
  207. assert filter_match_rule.header_fields["member"] == "PrepareForShutdown"
  208. add_match_msg = dbus_router_mock.send_and_get_reply.call_args[0][0]
  209. assert (
  210. add_match_msg.header.fields[jeepney.low_level.HeaderFields.member] == "AddMatch"
  211. )
  212. assert add_match_msg.body == (
  213. "interface='org.freedesktop.login1.Manager',member='PrepareForShutdown'"
  214. ",path='/org/freedesktop/login1',type='signal'",
  215. )
  216. assert [
  217. c[1]["active"] for c in state_mock.preparing_for_shutdown_handler.call_args_list
  218. ] == [False, True, False]
  219. assert not any(args for args, _ in dbus_signal_loop_unit_mock.await_args_list)
  220. dbus_signal_loop_unit_kwargs = [
  221. kwargs for _, kwargs in dbus_signal_loop_unit_mock.await_args_list
  222. ]
  223. assert [(a["unit_name"], a["unit_path"]) for a in dbus_signal_loop_unit_kwargs] == [
  224. (n, f"/org/freedesktop/systemd1/unit/{n}") for n in monitored_system_unit_names
  225. ]
  226. def _mock_get_active_state_reply(state: str) -> unittest.mock.MagicMock:
  227. reply_mock = unittest.mock.MagicMock()
  228. reply_mock.body = (("s", state),)
  229. return reply_mock
  230. @pytest.mark.asyncio
  231. async def test__dbus_signal_loop_unit() -> None:
  232. state = systemctl_mqtt._State(
  233. mqtt_topic_prefix="prefix",
  234. homeassistant_discovery_prefix="unused",
  235. homeassistant_discovery_object_id="unused",
  236. poweroff_delay=datetime.timedelta(),
  237. monitored_system_unit_names=[],
  238. )
  239. mqtt_client_mock = unittest.mock.AsyncMock()
  240. dbus_router_mock = unittest.mock.AsyncMock()
  241. bus_proxy_mock = unittest.mock.AsyncMock()
  242. bus_proxy_mock.AddMatch.return_value = ()
  243. get_active_state_reply_mock = unittest.mock.MagicMock()
  244. get_active_state_reply_mock.body = (("s", "active"),)
  245. states = [
  246. "active",
  247. "deactivating",
  248. "inactive",
  249. "inactive",
  250. "activating",
  251. "active",
  252. "active",
  253. "active",
  254. "inactive",
  255. ]
  256. dbus_router_mock.send_and_get_reply.side_effect = [
  257. _mock_get_active_state_reply(s) for s in states
  258. ]
  259. msg_queue: asyncio.Queue[jeepney.low_level.Message] = asyncio.Queue()
  260. for _ in range(len(states) - 1):
  261. await msg_queue.put(jeepney.low_level.Message(header=None, body=()))
  262. dbus_router_mock.filter = unittest.mock.MagicMock()
  263. dbus_router_mock.filter.return_value.__enter__.return_value = msg_queue
  264. loop_task = asyncio.create_task(
  265. systemctl_mqtt._dbus_signal_loop_unit(
  266. state=state,
  267. mqtt_client=mqtt_client_mock,
  268. dbus_router=dbus_router_mock,
  269. bus_proxy=bus_proxy_mock,
  270. unit_name="foo.service",
  271. unit_path="/org/freedesktop/systemd1/unit/whatever.service",
  272. )
  273. )
  274. async def _abort_after_msg_queue():
  275. await msg_queue.join()
  276. loop_task.cancel()
  277. with pytest.raises(asyncio.exceptions.CancelledError):
  278. await asyncio.gather(*(loop_task, _abort_after_msg_queue()))
  279. bus_proxy_mock.AddMatch.assert_awaited_once()
  280. ((match_rule,), add_match_kwargs) = bus_proxy_mock.AddMatch.await_args
  281. assert match_rule.header_fields["interface"] == "org.freedesktop.DBus.Properties"
  282. assert match_rule.header_fields["member"] == "PropertiesChanged"
  283. assert not add_match_kwargs
  284. assert mqtt_client_mock.publish.await_args_list == [
  285. unittest.mock.call(
  286. topic="prefix/unit/system/foo.service/active-state", payload=s
  287. )
  288. for s in [ # consecutive duplicates filtered
  289. "active",
  290. "deactivating",
  291. "inactive",
  292. "activating",
  293. "active",
  294. "inactive",
  295. ]
  296. ]