2017-07-01 20:58:12 +00:00
|
|
|
"""
|
|
|
|
Support for Snips on-device ASR and NLU.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/snips/
|
|
|
|
"""
|
|
|
|
import asyncio
|
|
|
|
import json
|
|
|
|
import logging
|
|
|
|
import voluptuous as vol
|
2017-07-22 04:38:53 +00:00
|
|
|
from homeassistant.helpers import intent, config_validation as cv
|
2017-07-01 20:58:12 +00:00
|
|
|
|
|
|
|
DOMAIN = 'snips'
|
|
|
|
DEPENDENCIES = ['mqtt']
|
|
|
|
CONF_INTENTS = 'intents'
|
|
|
|
CONF_ACTION = 'action'
|
|
|
|
|
2017-12-08 17:16:08 +00:00
|
|
|
INTENT_TOPIC = 'hermes/intent/#'
|
2017-07-01 20:58:12 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2017-07-01 20:58:12 +00:00
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
2017-07-22 04:38:53 +00:00
|
|
|
DOMAIN: {}
|
2017-07-01 20:58:12 +00:00
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
INTENT_SCHEMA = vol.Schema({
|
2017-07-03 13:07:59 +00:00
|
|
|
vol.Required('input'): str,
|
2017-07-01 20:58:12 +00:00
|
|
|
vol.Required('intent'): {
|
2017-07-03 13:07:59 +00:00
|
|
|
vol.Required('intentName'): str
|
2017-07-01 20:58:12 +00:00
|
|
|
},
|
|
|
|
vol.Optional('slots'): [{
|
2017-07-03 13:07:59 +00:00
|
|
|
vol.Required('slotName'): str,
|
2017-07-01 20:58:12 +00:00
|
|
|
vol.Required('value'): {
|
|
|
|
vol.Required('kind'): str,
|
2017-12-08 17:16:08 +00:00
|
|
|
vol.Optional('value'): cv.match_all,
|
|
|
|
vol.Optional('rawValue'): cv.match_all
|
2017-07-01 20:58:12 +00:00
|
|
|
}
|
|
|
|
}]
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config):
|
|
|
|
"""Activate Snips component."""
|
|
|
|
@asyncio.coroutine
|
|
|
|
def message_received(topic, payload, qos):
|
|
|
|
"""Handle new messages on MQTT."""
|
2017-07-22 04:38:53 +00:00
|
|
|
_LOGGER.debug("New intent: %s", payload)
|
2017-07-01 20:58:12 +00:00
|
|
|
|
|
|
|
try:
|
2017-07-22 04:38:53 +00:00
|
|
|
request = json.loads(payload)
|
2017-07-01 20:58:12 +00:00
|
|
|
except TypeError:
|
2017-07-22 04:38:53 +00:00
|
|
|
_LOGGER.error('Received invalid JSON: %s', payload)
|
2017-07-01 20:58:12 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
2017-07-22 04:38:53 +00:00
|
|
|
request = INTENT_SCHEMA(request)
|
2017-07-01 20:58:12 +00:00
|
|
|
except vol.Invalid as err:
|
2017-07-22 04:38:53 +00:00
|
|
|
_LOGGER.error('Intent has invalid schema: %s. %s', err, request)
|
2017-07-01 20:58:12 +00:00
|
|
|
return
|
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
intent_type = request['intent']['intentName'].split('__')[-1]
|
2017-12-08 17:16:08 +00:00
|
|
|
slots = {}
|
|
|
|
for slot in request.get('slots', []):
|
|
|
|
if 'value' in slot['value']:
|
|
|
|
slots[slot['slotName']] = {'value': slot['value']['value']}
|
|
|
|
else:
|
|
|
|
slots[slot['slotName']] = {'value': slot['rawValue']}
|
2017-07-01 20:58:12 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
try:
|
|
|
|
yield from intent.async_handle(
|
|
|
|
hass, DOMAIN, intent_type, slots, request['input'])
|
|
|
|
except intent.IntentError:
|
2017-09-04 11:40:08 +00:00
|
|
|
_LOGGER.exception("Error while handling intent: %s.", intent_type)
|
2017-07-01 20:58:12 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
yield from hass.components.mqtt.async_subscribe(
|
|
|
|
INTENT_TOPIC, message_received)
|
2017-07-01 20:58:12 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
return True
|