mqtt.py 7.0 KB

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