mqtt.py 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. import os
  2. import paho.mqtt.client as mqtt
  3. import json
  4. from dotenv import load_dotenv
  5. # Load the .env file
  6. load_dotenv()
  7. # Define the MQTT server details
  8. broker = os.environ.get("broker")
  9. port = int(os.environ.get("port"))
  10. # MQTT username and password
  11. username = os.environ.get("username")
  12. password = os.environ.get("password")
  13. def create_client() -> mqtt.Client:
  14. """Create an MQTT client and connect it to the broker
  15. Returns:
  16. mqtt.Client: Connected MQTT client instance
  17. """
  18. # Create a new MQTT client instance
  19. client = mqtt.Client()
  20. # Set username and password
  21. client.username_pw_set(username, password)
  22. # Connect to the MQTT broker
  23. client.connect(broker, port, 60)
  24. return client
  25. def create_config(client: mqtt.Client) -> None:
  26. """Create Home Assistant discovery topics
  27. Args:
  28. client (mqtt.Client): MQTT Client
  29. """
  30. # Device-specific information for multiple sensors
  31. node_id = "floppy_player" # Unique device ID
  32. # Define discovery and state topics for each sensor
  33. discovery_topic_disc = f"homeassistant/sensor/floppy_player/current_disc/config"
  34. discovery_topic_disc_type = (
  35. f"homeassistant/sensor/floppy_player/current_disc_type/config"
  36. )
  37. discovery_topic_disc_id = (
  38. f"homeassistant/sensor/floppy_player/current_disc_id/config"
  39. )
  40. current_disc_state_topic_disc = (
  41. f"homeassistant/sensor/floppy_player/current_disc/state"
  42. )
  43. current_disc_type_state_topic_disc = (
  44. f"homeassistant/sensor/floppy_player/current_disc_type/state"
  45. )
  46. current_disc_id_state_topic_disc = (
  47. f"homeassistant/sensor/floppy_player/current_disc_id/state"
  48. )
  49. discovery_topic_status = f"homeassistant/sensor/floppy_player/status/config"
  50. state_topic_status = f"homeassistant/sensor/floppy_player/status/state"
  51. # Sensor 1: current_disc (a text-based sensor)
  52. current_disc_config = {
  53. "name": "Current Disc",
  54. "state_topic": current_disc_state_topic_disc,
  55. "value_template": "{{ value }}", # Textual value
  56. "unique_id": f"{node_id}_current_disc",
  57. "device": {
  58. "identifiers": [node_id],
  59. "name": "Floppy Player",
  60. "model": "v1",
  61. "manufacturer": "Karl",
  62. },
  63. }
  64. current_disc_type_state_config = {
  65. "name": "Current Disc Type",
  66. "state_topic": current_disc_type_state_topic_disc,
  67. "value_template": "{{ value }}", # Textual value
  68. "unique_id": f"{node_id}_current_disc_type",
  69. "device": {
  70. "identifiers": [node_id],
  71. "name": "Floppy Player",
  72. "model": "v1",
  73. "manufacturer": "Karl",
  74. },
  75. }
  76. current_disc_id_state_config = {
  77. "name": "Current Disc Id",
  78. "state_topic": current_disc_id_state_topic_disc,
  79. "value_template": "{{ value }}", # Textual value
  80. "unique_id": f"{node_id}_current_disc_id",
  81. "device": {
  82. "identifiers": [node_id],
  83. "name": "Floppy Player",
  84. "model": "v1",
  85. "manufacturer": "Karl",
  86. },
  87. }
  88. # Sensor 2: status (another text-based sensor)
  89. status_config = {
  90. "name": "Device Status",
  91. "state_topic": state_topic_status,
  92. "value_template": "{{ value }}", # Textual value
  93. "unique_id": f"{node_id}_status",
  94. "device": {
  95. "identifiers": [node_id],
  96. "name": "Floppy Player",
  97. "model": "v1",
  98. "manufacturer": "Karl",
  99. },
  100. }
  101. client.publish(discovery_topic_disc, json.dumps(current_disc_config), retain=True)
  102. client.publish(
  103. discovery_topic_disc_type,
  104. json.dumps(current_disc_type_state_config),
  105. retain=True,
  106. )
  107. client.publish(
  108. discovery_topic_disc_id, json.dumps(current_disc_id_state_config), retain=True
  109. )
  110. client.publish(discovery_topic_status, json.dumps(status_config), retain=True)
  111. def check_current_disc(client: mqtt.Client) -> dict:
  112. """Check the current loaded disc, disc type, and disc ID by subscribing to the retained messages on the state topics.
  113. Args:
  114. client (mqtt.Client): MQTT Client
  115. Returns:
  116. dict: A dictionary containing the current disc, disc type, and disc ID.
  117. """
  118. def on_message(client, userdata, message):
  119. """Callback function to handle received MQTT messages."""
  120. topic = message.topic
  121. userdata[topic] = message.payload.decode()
  122. # Create a dictionary to store the messages
  123. userdata = {
  124. "homeassistant/sensor/floppy_player/current_disc/state": None,
  125. "homeassistant/sensor/floppy_player/current_disc_type/state": None,
  126. "homeassistant/sensor/floppy_player/current_disc_id/state": None,
  127. }
  128. # Set the user data and the on_message callback
  129. client.user_data_set(userdata)
  130. client.on_message = on_message
  131. # Subscribe to the topics
  132. current_disc_state_topic_disc = (
  133. "homeassistant/sensor/floppy_player/current_disc/state"
  134. )
  135. current_disc_type_state_topic_disc = (
  136. "homeassistant/sensor/floppy_player/current_disc_type/state"
  137. )
  138. current_disc_id_state_topic_disc = (
  139. "homeassistant/sensor/floppy_player/current_disc_id/state"
  140. )
  141. topics = [
  142. current_disc_state_topic_disc,
  143. current_disc_type_state_topic_disc,
  144. current_disc_id_state_topic_disc,
  145. ]
  146. for topic in topics:
  147. client.subscribe(topic)
  148. # Run the loop manually until we receive all messages or timeout
  149. timeout = 5 # Timeout after 5 seconds
  150. while any(value is None for value in userdata.values()) and timeout > 0:
  151. client.loop(timeout=0.1) # Process network events for a short time
  152. timeout -= 0.1
  153. # Check if we received all messages
  154. if any(value is None for value in userdata.values()):
  155. raise TimeoutError("Some retained messages were not found.")
  156. # Return the relevant messages
  157. return {
  158. "name": userdata[current_disc_state_topic_disc],
  159. "type": userdata[current_disc_type_state_topic_disc],
  160. "id": userdata[current_disc_id_state_topic_disc],
  161. }
  162. def check_current_status(client: mqtt.Client) -> str:
  163. """Check the current player status by subscribing to the retained message on the state topic.
  164. Args:
  165. client (mqtt.Client): MQTT Client
  166. Returns:
  167. str: Current object at current_disc/state
  168. """
  169. def on_message(client, userdata, message):
  170. """Callback function to handle received MQTT messages."""
  171. userdata['message'] = message.payload.decode()
  172. # Create a dictionary to store the message
  173. userdata = {'message': None}
  174. # Set the user data and the on_message callback
  175. client.user_data_set(userdata)
  176. client.on_message = on_message
  177. # Subscribe to the topic
  178. client.subscribe("homeassistant/sensor/floppy_player/status/state")
  179. # Run the loop manually until we receive a message
  180. timeout = 5 # Timeout after 5 seconds
  181. while userdata['message'] is None and timeout > 0:
  182. client.loop(timeout=0.1) # Process network events for a short time
  183. timeout -= 0.1
  184. # Check if we received the message
  185. if userdata['message'] is None:
  186. raise TimeoutError("No retained message was found.")
  187. return userdata['message']
  188. def update_disc(client: mqtt.Client, disc_message: dict) -> None:
  189. """Update current disc.
  190. Args:
  191. client (mqtt.Client): MQTT Client
  192. disc_message (dict): Current disc information
  193. """
  194. client.publish(
  195. "homeassistant/sensor/floppy_player/current_disc/state",
  196. disc_message[1],
  197. retain=True,
  198. )
  199. client.publish(
  200. "homeassistant/sensor/floppy_player/current_disc_type/state",
  201. disc_message[2],
  202. retain=True,
  203. )
  204. client.publish(
  205. "homeassistant/sensor/floppy_player/current_disc_id/state",
  206. disc_message[3],
  207. retain=True,
  208. )
  209. print(f"Published current disc: {disc_message}")
  210. def control_player(client: mqtt.Client, state: str) -> None:
  211. """Control the player
  212. Args:
  213. client (mqtt.Client): MQTT Client
  214. state (str): Player State
  215. """
  216. client.publish(
  217. "homeassistant/sensor/floppy_player/status/state", state, retain=True
  218. )
  219. print(f"Published status: {state}")