__init__.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  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 abc
  18. import argparse
  19. import asyncio
  20. import datetime
  21. import functools
  22. import importlib.metadata
  23. import json
  24. import logging
  25. import os
  26. import pathlib
  27. import socket
  28. import ssl
  29. import threading
  30. import typing
  31. import aiomqtt
  32. import jeepney
  33. import jeepney.bus_messages
  34. import jeepney.io.asyncio
  35. import systemctl_mqtt._dbus
  36. import systemctl_mqtt._homeassistant
  37. import systemctl_mqtt._mqtt
  38. _MQTT_DEFAULT_PORT = 1883
  39. _MQTT_DEFAULT_TLS_PORT = 8883
  40. # > payload_not_available string (Optional, default: offline)
  41. # https://web.archive.org/web/20250101075341/https://www.home-assistant.io/integrations/sensor.mqtt/#payload_not_available
  42. _MQTT_PAYLOAD_NOT_AVAILABLE = "offline"
  43. _MQTT_PAYLOAD_AVAILABLE = "online"
  44. _ARGUMENT_LOG_LEVEL_MAPPING = {
  45. a: getattr(logging, a.upper())
  46. for a in ("debug", "info", "warning", "error", "critical")
  47. }
  48. _LOGGER = logging.getLogger(__name__)
  49. class _State:
  50. def __init__(
  51. self,
  52. *,
  53. mqtt_topic_prefix: str,
  54. homeassistant_discovery_prefix: str,
  55. homeassistant_discovery_object_id: str,
  56. poweroff_delay: datetime.timedelta,
  57. ) -> None:
  58. self._mqtt_topic_prefix = mqtt_topic_prefix
  59. self._homeassistant_discovery_prefix = homeassistant_discovery_prefix
  60. self._homeassistant_discovery_object_id = homeassistant_discovery_object_id
  61. self._login_manager = systemctl_mqtt._dbus.get_login_manager_proxy()
  62. self._shutdown_lock: typing.Optional[jeepney.fds.FileDescriptor] = None
  63. self._shutdown_lock_mutex = threading.Lock()
  64. self.poweroff_delay = poweroff_delay
  65. @property
  66. def mqtt_topic_prefix(self) -> str:
  67. return self._mqtt_topic_prefix
  68. @property
  69. def mqtt_availability_topic(self) -> str:
  70. # > mqtt.ATTR_TOPIC: "homeassistant/status",
  71. # https://github.com/home-assistant/core/blob/2024.12.5/tests/components/mqtt/conftest.py#L23
  72. # > _MQTT_AVAILABILITY_TOPIC = "switchbot-mqtt/status"
  73. # https://github.com/fphammerle/switchbot-mqtt/blob/v3.3.1/switchbot_mqtt/__init__.py#L30
  74. return self._mqtt_topic_prefix + "/status"
  75. @property
  76. def shutdown_lock_acquired(self) -> bool:
  77. return self._shutdown_lock is not None
  78. def acquire_shutdown_lock(self) -> None:
  79. with self._shutdown_lock_mutex:
  80. assert self._shutdown_lock is None
  81. # https://www.freedesktop.org/wiki/Software/systemd/inhibit/
  82. (self._shutdown_lock,) = self._login_manager.Inhibit(
  83. what="shutdown",
  84. who="systemctl-mqtt",
  85. why="Report shutdown via MQTT",
  86. mode="delay",
  87. )
  88. assert isinstance(
  89. self._shutdown_lock, jeepney.fds.FileDescriptor
  90. ), self._shutdown_lock
  91. _LOGGER.debug("acquired shutdown inhibitor lock")
  92. def release_shutdown_lock(self) -> None:
  93. with self._shutdown_lock_mutex:
  94. if self._shutdown_lock:
  95. self._shutdown_lock.close()
  96. _LOGGER.debug("released shutdown inhibitor lock")
  97. self._shutdown_lock = None
  98. @property
  99. def _preparing_for_shutdown_topic(self) -> str:
  100. return self.mqtt_topic_prefix + "/preparing-for-shutdown"
  101. async def _publish_preparing_for_shutdown(
  102. self, *, mqtt_client: aiomqtt.Client, active: bool
  103. ) -> None:
  104. topic = self._preparing_for_shutdown_topic
  105. # pylint: disable=protected-access
  106. payload = systemctl_mqtt._mqtt.encode_bool(active)
  107. _LOGGER.info("publishing %r on %s", payload, topic)
  108. await mqtt_client.publish(topic=topic, payload=payload, retain=False)
  109. async def preparing_for_shutdown_handler(
  110. self, active: bool, mqtt_client: aiomqtt.Client
  111. ) -> None:
  112. active = bool(active)
  113. await self._publish_preparing_for_shutdown(
  114. mqtt_client=mqtt_client, active=active
  115. )
  116. if active:
  117. self.release_shutdown_lock()
  118. else:
  119. self.acquire_shutdown_lock()
  120. async def publish_preparing_for_shutdown(self, mqtt_client: aiomqtt.Client) -> None:
  121. try:
  122. ((return_type, active),) = self._login_manager.Get("PreparingForShutdown")
  123. except jeepney.wrappers.DBusErrorResponse as exc:
  124. _LOGGER.error(
  125. "failed to read logind's PreparingForShutdown property: %s", exc
  126. )
  127. return
  128. assert return_type == "b", return_type
  129. assert isinstance(active, bool), active
  130. await self._publish_preparing_for_shutdown(
  131. mqtt_client=mqtt_client, active=active
  132. )
  133. async def publish_homeassistant_device_config(
  134. self, mqtt_client: aiomqtt.Client
  135. ) -> None:
  136. # <discovery_prefix>/<component>/[<node_id>/]<object_id>/config
  137. # https://www.home-assistant.io/integrations/mqtt/#mqtt-discovery
  138. discovery_topic = "/".join(
  139. (
  140. self._homeassistant_discovery_prefix,
  141. "device",
  142. self._homeassistant_discovery_object_id,
  143. "config",
  144. )
  145. )
  146. hostname = (
  147. # pylint: disable=protected-access; function in internal module
  148. systemctl_mqtt._utils.get_hostname()
  149. )
  150. package_metadata = importlib.metadata.metadata(__name__)
  151. unique_id_prefix = "systemctl-mqtt-" + hostname
  152. config = {
  153. "device": {"identifiers": [hostname], "name": hostname},
  154. "origin": {
  155. "name": package_metadata["Name"],
  156. "sw_version": package_metadata["Version"],
  157. "support_url": package_metadata["Home-page"],
  158. },
  159. "availability": {"topic": self.mqtt_availability_topic},
  160. "components": {
  161. "logind/preparing-for-shutdown": {
  162. "unique_id": unique_id_prefix + "-logind-preparing-for-shutdown",
  163. "object_id": f"{hostname}_logind_preparing_for_shutdown", # entity id
  164. "name": "preparing for shutdown", # home assistant prepends device name
  165. "platform": "binary_sensor",
  166. "state_topic": self._preparing_for_shutdown_topic,
  167. # pylint: disable=protected-access
  168. "payload_on": systemctl_mqtt._mqtt.encode_bool(True),
  169. "payload_off": systemctl_mqtt._mqtt.encode_bool(False),
  170. },
  171. },
  172. }
  173. for mqtt_topic_suffix in _MQTT_TOPIC_SUFFIX_ACTION_MAPPING.keys():
  174. # false positive warning by mypy:
  175. # > Unsupported target for indexed assignment
  176. config["components"]["logind/" + mqtt_topic_suffix] = { # type: ignore
  177. "unique_id": unique_id_prefix + "-logind-" + mqtt_topic_suffix,
  178. "object_id": hostname
  179. + "_logind_"
  180. + mqtt_topic_suffix.replace("-", "_"), # entity id
  181. "name": mqtt_topic_suffix.replace("-", " "),
  182. "platform": "button",
  183. "command_topic": self.mqtt_topic_prefix + "/" + mqtt_topic_suffix,
  184. }
  185. _LOGGER.debug("publishing home assistant config on %s", discovery_topic)
  186. await mqtt_client.publish(
  187. topic=discovery_topic, payload=json.dumps(config), retain=False
  188. )
  189. class _MQTTAction(metaclass=abc.ABCMeta):
  190. @abc.abstractmethod
  191. def trigger(self, state: _State) -> None:
  192. pass # pragma: no cover
  193. def __str__(self) -> str:
  194. return type(self).__name__
  195. class _MQTTActionSchedulePoweroff(_MQTTAction):
  196. # pylint: disable=too-few-public-methods
  197. def trigger(self, state: _State) -> None:
  198. # pylint: disable=protected-access
  199. systemctl_mqtt._dbus.schedule_shutdown(
  200. action="poweroff", delay=state.poweroff_delay
  201. )
  202. class _MQTTActionLockAllSessions(_MQTTAction):
  203. # pylint: disable=too-few-public-methods
  204. def trigger(self, state: _State) -> None:
  205. # pylint: disable=protected-access
  206. systemctl_mqtt._dbus.lock_all_sessions()
  207. class _MQTTActionSuspend(_MQTTAction):
  208. # pylint: disable=too-few-public-methods
  209. def trigger(self, state: _State) -> None:
  210. # pylint: disable=protected-access
  211. systemctl_mqtt._dbus.suspend()
  212. _MQTT_TOPIC_SUFFIX_ACTION_MAPPING = {
  213. "poweroff": _MQTTActionSchedulePoweroff(),
  214. "lock-all-sessions": _MQTTActionLockAllSessions(),
  215. "suspend": _MQTTActionSuspend(),
  216. }
  217. async def _mqtt_message_loop(*, state: _State, mqtt_client: aiomqtt.Client) -> None:
  218. action_by_topic: typing.Dict[str, _MQTTAction] = {}
  219. for topic_suffix, action in _MQTT_TOPIC_SUFFIX_ACTION_MAPPING.items():
  220. topic = state.mqtt_topic_prefix + "/" + topic_suffix
  221. _LOGGER.info("subscribing to %s", topic)
  222. await mqtt_client.subscribe(topic)
  223. action_by_topic[topic] = action
  224. async for message in mqtt_client.messages:
  225. if message.retain:
  226. _LOGGER.info("ignoring retained message on topic %r", message.topic.value)
  227. else:
  228. _LOGGER.debug(
  229. "received message on topic %r: %r", message.topic.value, message.payload
  230. )
  231. action_by_topic[message.topic.value].trigger(state=state)
  232. async def _dbus_signal_loop(*, state: _State, mqtt_client: aiomqtt.Client) -> None:
  233. async with jeepney.io.asyncio.open_dbus_router(bus="SYSTEM") as router:
  234. # router: jeepney.io.asyncio.DBusRouter
  235. bus_proxy = jeepney.io.asyncio.Proxy(
  236. msggen=jeepney.bus_messages.message_bus, router=router
  237. )
  238. preparing_for_shutdown_match_rule = (
  239. # pylint: disable=protected-access
  240. systemctl_mqtt._dbus.get_login_manager_signal_match_rule(
  241. "PrepareForShutdown"
  242. )
  243. )
  244. assert await bus_proxy.AddMatch(preparing_for_shutdown_match_rule) == ()
  245. with router.filter(preparing_for_shutdown_match_rule) as queue:
  246. while True:
  247. message: jeepney.low_level.Message = await queue.get()
  248. (preparing_for_shutdown,) = message.body
  249. await state.preparing_for_shutdown_handler(
  250. active=preparing_for_shutdown, mqtt_client=mqtt_client
  251. )
  252. queue.task_done()
  253. async def _run( # pylint: disable=too-many-arguments
  254. *,
  255. mqtt_host: str,
  256. mqtt_port: int,
  257. mqtt_username: typing.Optional[str],
  258. mqtt_password: typing.Optional[str],
  259. mqtt_topic_prefix: str,
  260. homeassistant_discovery_prefix: str,
  261. homeassistant_discovery_object_id: str,
  262. poweroff_delay: datetime.timedelta,
  263. mqtt_disable_tls: bool = False,
  264. ) -> None:
  265. state = _State(
  266. mqtt_topic_prefix=mqtt_topic_prefix,
  267. homeassistant_discovery_prefix=homeassistant_discovery_prefix,
  268. homeassistant_discovery_object_id=homeassistant_discovery_object_id,
  269. poweroff_delay=poweroff_delay,
  270. )
  271. _LOGGER.info(
  272. "connecting to MQTT broker %s:%d (TLS %s)",
  273. mqtt_host,
  274. mqtt_port,
  275. "disabled" if mqtt_disable_tls else "enabled",
  276. )
  277. if mqtt_password and not mqtt_username:
  278. raise ValueError("Missing MQTT username")
  279. async with aiomqtt.Client( # raises aiomqtt.MqttError
  280. hostname=mqtt_host,
  281. port=mqtt_port,
  282. # > The settings [...] usually represent a higher security level than
  283. # > when calling the SSLContext constructor directly.
  284. # https://web.archive.org/web/20230714183106/https://docs.python.org/3/library/ssl.html
  285. tls_context=None if mqtt_disable_tls else ssl.create_default_context(),
  286. username=None if mqtt_username is None else mqtt_username,
  287. password=None if mqtt_password is None else mqtt_password,
  288. will=aiomqtt.Will( # e.g. on SIGTERM & SIGKILL
  289. topic=state.mqtt_availability_topic,
  290. payload=_MQTT_PAYLOAD_NOT_AVAILABLE,
  291. retain=True,
  292. ),
  293. ) as mqtt_client:
  294. _LOGGER.debug("connected to MQTT broker %s:%d", mqtt_host, mqtt_port)
  295. if not state.shutdown_lock_acquired:
  296. state.acquire_shutdown_lock()
  297. await state.publish_homeassistant_device_config(mqtt_client=mqtt_client)
  298. await state.publish_preparing_for_shutdown(mqtt_client=mqtt_client)
  299. try:
  300. await mqtt_client.publish(
  301. topic=state.mqtt_availability_topic,
  302. payload=_MQTT_PAYLOAD_AVAILABLE,
  303. retain=True,
  304. )
  305. # asynpio.TaskGroup added in python3.11
  306. await asyncio.gather(
  307. _mqtt_message_loop(state=state, mqtt_client=mqtt_client),
  308. _dbus_signal_loop(state=state, mqtt_client=mqtt_client),
  309. return_exceptions=False,
  310. )
  311. finally: # e.g. on SIGINT
  312. # https://web.archive.org/web/20250101080719/https://github.com/empicano/aiomqtt/issues/28
  313. await mqtt_client.publish(
  314. topic=state.mqtt_availability_topic,
  315. payload=_MQTT_PAYLOAD_NOT_AVAILABLE,
  316. retain=True,
  317. )
  318. def _main() -> None:
  319. logging.basicConfig(
  320. level=logging.INFO,
  321. format="%(asctime)s:%(levelname)s:%(message)s",
  322. datefmt="%Y-%m-%dT%H:%M:%S%z",
  323. )
  324. argparser = argparse.ArgumentParser(
  325. description="MQTT client triggering & reporting shutdown on systemd-based systems",
  326. )
  327. argparser.add_argument(
  328. "--log-level",
  329. choices=_ARGUMENT_LOG_LEVEL_MAPPING.keys(),
  330. default="info",
  331. help="log level (default: %(default)s)",
  332. )
  333. argparser.add_argument("--mqtt-host", type=str, required=True)
  334. argparser.add_argument(
  335. "--mqtt-port",
  336. type=int,
  337. help=f"default {_MQTT_DEFAULT_TLS_PORT} ({_MQTT_DEFAULT_PORT} with --mqtt-disable-tls)",
  338. )
  339. argparser.add_argument("--mqtt-username", type=str)
  340. argparser.add_argument("--mqtt-disable-tls", action="store_true")
  341. password_argument_group = argparser.add_mutually_exclusive_group()
  342. password_argument_group.add_argument("--mqtt-password", type=str)
  343. password_argument_group.add_argument(
  344. "--mqtt-password-file",
  345. type=pathlib.Path,
  346. metavar="PATH",
  347. dest="mqtt_password_path",
  348. help="stripping trailing newline",
  349. )
  350. argparser.add_argument(
  351. "--mqtt-topic-prefix",
  352. type=str,
  353. # pylint: disable=protected-access
  354. default="systemctl/" + systemctl_mqtt._utils.get_hostname(),
  355. help="default: %(default)s",
  356. )
  357. # https://www.home-assistant.io/docs/mqtt/discovery/#discovery_prefix
  358. argparser.add_argument(
  359. "--homeassistant-discovery-prefix",
  360. type=str,
  361. default="homeassistant",
  362. help="home assistant's prefix for discovery topics" + " (default: %(default)s)",
  363. )
  364. argparser.add_argument(
  365. "--homeassistant-discovery-object-id",
  366. type=str,
  367. # pylint: disable=protected-access
  368. default=systemctl_mqtt._homeassistant.get_default_discovery_object_id(),
  369. help="part of discovery topic (default: %(default)s)",
  370. )
  371. argparser.add_argument(
  372. "--poweroff-delay-seconds", type=float, default=4.0, help="default: %(default)s"
  373. )
  374. args = argparser.parse_args()
  375. logging.root.setLevel(_ARGUMENT_LOG_LEVEL_MAPPING[args.log_level])
  376. if args.mqtt_port:
  377. mqtt_port = args.mqtt_port
  378. elif args.mqtt_disable_tls:
  379. mqtt_port = _MQTT_DEFAULT_PORT
  380. else:
  381. mqtt_port = _MQTT_DEFAULT_TLS_PORT
  382. if args.mqtt_password_path:
  383. # .read_text() replaces \r\n with \n
  384. mqtt_password = args.mqtt_password_path.read_bytes().decode()
  385. if mqtt_password.endswith("\r\n"):
  386. mqtt_password = mqtt_password[:-2]
  387. elif mqtt_password.endswith("\n"):
  388. mqtt_password = mqtt_password[:-1]
  389. else:
  390. mqtt_password = args.mqtt_password
  391. # pylint: disable=protected-access
  392. if not systemctl_mqtt._homeassistant.validate_discovery_object_id(
  393. args.homeassistant_discovery_object_id
  394. ):
  395. raise ValueError(
  396. # pylint: disable=protected-access
  397. "invalid home assistant discovery object id"
  398. f" {args.homeassistant_discovery_object_id!r} (length >= 1"
  399. ", allowed characters:"
  400. f" {systemctl_mqtt._homeassistant.NODE_ID_ALLOWED_CHARS})"
  401. "\nchange --homeassistant-discovery-object-id"
  402. )
  403. asyncio.run(
  404. _run(
  405. mqtt_host=args.mqtt_host,
  406. mqtt_port=mqtt_port,
  407. mqtt_disable_tls=args.mqtt_disable_tls,
  408. mqtt_username=args.mqtt_username,
  409. mqtt_password=mqtt_password,
  410. mqtt_topic_prefix=args.mqtt_topic_prefix,
  411. homeassistant_discovery_prefix=args.homeassistant_discovery_prefix,
  412. homeassistant_discovery_object_id=args.homeassistant_discovery_object_id,
  413. poweroff_delay=datetime.timedelta(seconds=args.poweroff_delay_seconds),
  414. )
  415. )