2016-04-13 11:50:28 +00:00
|
|
|
"""
|
|
|
|
Support for a local MQTT broker.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/mqtt/#use-the-embedded-broker
|
|
|
|
"""
|
2016-03-08 00:45:37 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
import tempfile
|
|
|
|
import threading
|
|
|
|
|
|
|
|
from homeassistant.components.mqtt import PROTOCOL_311
|
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STOP
|
|
|
|
|
2016-05-07 17:59:23 +00:00
|
|
|
REQUIREMENTS = ['hbmqtt==0.7.1']
|
2016-03-08 00:45:37 +00:00
|
|
|
DEPENDENCIES = ['http']
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def broker_coro(loop, config):
|
|
|
|
"""Start broker coroutine."""
|
|
|
|
from hbmqtt.broker import Broker
|
|
|
|
broker = Broker(config, loop)
|
|
|
|
yield from broker.start()
|
|
|
|
return broker
|
|
|
|
|
|
|
|
|
|
|
|
def loop_run(loop, broker, shutdown_complete):
|
|
|
|
"""Run broker and clean up when done."""
|
|
|
|
loop.run_forever()
|
|
|
|
# run_forever ends when stop is called because we're shutting down
|
|
|
|
loop.run_until_complete(broker.shutdown())
|
|
|
|
loop.close()
|
|
|
|
shutdown_complete.set()
|
|
|
|
|
|
|
|
|
|
|
|
def start(hass, server_config):
|
|
|
|
"""Initialize MQTT Server."""
|
|
|
|
from hbmqtt.broker import BrokerException
|
|
|
|
|
|
|
|
loop = asyncio.new_event_loop()
|
|
|
|
|
|
|
|
try:
|
|
|
|
passwd = tempfile.NamedTemporaryFile()
|
|
|
|
|
|
|
|
if server_config is None:
|
|
|
|
server_config, client_config = generate_config(hass, passwd)
|
|
|
|
else:
|
|
|
|
client_config = None
|
|
|
|
|
|
|
|
start_server = asyncio.gather(broker_coro(loop, server_config),
|
|
|
|
loop=loop)
|
|
|
|
loop.run_until_complete(start_server)
|
|
|
|
# Result raises exception if one was raised during startup
|
|
|
|
broker = start_server.result()[0]
|
|
|
|
except BrokerException:
|
|
|
|
logging.getLogger(__name__).exception('Error initializing MQTT server')
|
|
|
|
loop.close()
|
|
|
|
return False, None
|
|
|
|
finally:
|
|
|
|
passwd.close()
|
|
|
|
|
|
|
|
shutdown_complete = threading.Event()
|
|
|
|
|
|
|
|
def shutdown(event):
|
|
|
|
"""Gracefully shutdown MQTT broker."""
|
|
|
|
loop.call_soon_threadsafe(loop.stop)
|
|
|
|
shutdown_complete.wait()
|
|
|
|
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, shutdown)
|
|
|
|
|
|
|
|
threading.Thread(target=loop_run, args=(loop, broker, shutdown_complete),
|
|
|
|
name="MQTT-server").start()
|
|
|
|
|
|
|
|
return True, client_config
|
|
|
|
|
|
|
|
|
|
|
|
def generate_config(hass, passwd):
|
|
|
|
"""Generate a configuration based on current Home Assistant instance."""
|
|
|
|
config = {
|
|
|
|
'listeners': {
|
|
|
|
'default': {
|
|
|
|
'max-connections': 50000,
|
|
|
|
'bind': '0.0.0.0:1883',
|
|
|
|
'type': 'tcp',
|
|
|
|
},
|
|
|
|
'ws-1': {
|
|
|
|
'bind': '0.0.0.0:8080',
|
|
|
|
'type': 'ws',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'auth': {
|
|
|
|
'allow-anonymous': hass.config.api.api_password is None
|
|
|
|
},
|
|
|
|
'plugins': ['auth_anonymous'],
|
|
|
|
}
|
|
|
|
|
|
|
|
if hass.config.api.api_password:
|
|
|
|
username = 'homeassistant'
|
|
|
|
password = hass.config.api.api_password
|
|
|
|
|
|
|
|
# Encrypt with what hbmqtt uses to verify
|
|
|
|
from passlib.apps import custom_app_context
|
|
|
|
|
|
|
|
passwd.write(
|
|
|
|
'homeassistant:{}\n'.format(
|
|
|
|
custom_app_context.encrypt(
|
|
|
|
hass.config.api.api_password)).encode('utf-8'))
|
|
|
|
passwd.flush()
|
|
|
|
|
|
|
|
config['auth']['password-file'] = passwd.name
|
|
|
|
config['plugins'].append('auth_file')
|
|
|
|
else:
|
|
|
|
username = None
|
|
|
|
password = None
|
|
|
|
|
|
|
|
client_config = ('localhost', 1883, username, password, None, PROTOCOL_311)
|
|
|
|
|
|
|
|
return config, client_config
|