mqtt.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  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. state_topic_disc = f"homeassistant/sensor/floppy_player/current_disc/state"
  31. discovery_topic_status = f"homeassistant/sensor/floppy_player/status/config"
  32. state_topic_status = f"homeassistant/sensor/floppy_player/status/state"
  33. # Sensor 1: current_disc (a text-based sensor)
  34. disc_config = {
  35. "name": "Current Disc",
  36. "state_topic": state_topic_disc,
  37. "value_template": "{{ value }}", # Textual value
  38. "unique_id": f"{node_id}_current_disc",
  39. "device": {
  40. "identifiers": [node_id],
  41. "name": "Floppy Player",
  42. "model": "v1",
  43. "manufacturer": "Karl"
  44. }
  45. }
  46. # Sensor 2: status (another text-based sensor)
  47. status_config = {
  48. "name": "Device Status",
  49. "state_topic": state_topic_status,
  50. "value_template": "{{ value }}", # Textual value
  51. "unique_id": f"{node_id}_status",
  52. "device": {
  53. "identifiers": [node_id],
  54. "name": "Floppy Player",
  55. "model": "v1",
  56. "manufacturer": "Karl"
  57. }
  58. }
  59. client.publish(discovery_topic_disc, json.dumps(disc_config), retain=True)
  60. client.publish(discovery_topic_status, json.dumps(status_config), retain=True)
  61. def check_current_disc(client: mqtt.Client) -> str:
  62. """Check the current loaded disc by subscribing to the retained message on the state topic.
  63. Args:
  64. client (mqtt.Client): MQTT Client
  65. Returns:
  66. str: Current object at current_disc/state
  67. """
  68. def on_message(client, userdata, message):
  69. """Callback function to handle received MQTT messages."""
  70. userdata['message'] = message.payload.decode()
  71. # Create a dictionary to store the message
  72. userdata = {'message': None}
  73. # Set the user data and the on_message callback
  74. client.user_data_set(userdata)
  75. client.on_message = on_message
  76. # Subscribe to the topic
  77. client.subscribe("homeassistant/sensor/floppy_player/current_disc/state")
  78. # Run the loop manually until we receive a message
  79. timeout = 5 # Timeout after 5 seconds
  80. while userdata['message'] is None and timeout > 0:
  81. client.loop(timeout=0.1) # Process network events for a short time
  82. timeout -= 0.1
  83. # Check if we received the message
  84. if userdata['message'] is None:
  85. raise TimeoutError("No retained message was found.")
  86. return userdata['message']
  87. def update_disc(client: mqtt.Client, disc_message: dict) -> None:
  88. """Update current disc.
  89. Args:
  90. client (mqtt.Client): MQTT Client
  91. disc_message (dict): Current disc information
  92. """
  93. client.publish("homeassistant/sensor/floppy_player/current_disc/state", json.dumps(disc_message), retain=True)
  94. print(f"Published current disc: {disc_message}")
  95. def control_player(client: mqtt.Client, state: str) -> None:
  96. """Control the player
  97. Args:
  98. client (mqtt.Client): MQTT Client
  99. state (str): Player State
  100. """
  101. client.publish("homeassistant/sensor/floppy_player/status/state", state, retain=True)
  102. print(f"Published status: {state}")