mirror of
https://github.com/home-assistant/core.git
synced 2025-07-25 22:27:07 +00:00

* Add mysensors notify platform * Make add_devices optional in platform callback function. * Use new argument structure for all existing mysensors platforms. * Add notify platform. * Update mysensors gateway. * Refactor notify setup * Enable discovery of notify platforms. * Update and add tests for notify component and some platforms. * Continue setup of notify platforms if a platform fails setup. * Remove notify tests that check platform config. These tests are not needed when config validation is used. * Add config validation to APNS notify platform. * Use discovery to set up mysensors notify platform. * Add discovery_info to get_service and update tests * Add discovery_info as keyword argument to the get_service function signature and update all notify platforms. * Update existing notify tests to check config validation using test helper. * Add removed tests back in that checked config in apns, command_line and file platforms, but use config validation test helper to verify config. * Add a test for notify file to increase coverage. * Fix some PEP issues. * Fix comments and use more constants * Move apns notify service under notify domain
165 lines
5.0 KiB
Python
165 lines
5.0 KiB
Python
"""
|
|
Provides functionality to notify people.
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
https://home-assistant.io/components/notify/
|
|
"""
|
|
import logging
|
|
import os
|
|
from functools import partial
|
|
|
|
import voluptuous as vol
|
|
|
|
import homeassistant.bootstrap as bootstrap
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.config import load_yaml_config_file
|
|
from homeassistant.const import CONF_NAME, CONF_PLATFORM
|
|
from homeassistant.helpers import config_per_platform, discovery
|
|
from homeassistant.util import slugify
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
# Platform specific data
|
|
ATTR_DATA = 'data'
|
|
|
|
# Text to notify user of
|
|
ATTR_MESSAGE = 'message'
|
|
|
|
# Target of the notification (user, device, etc)
|
|
ATTR_TARGET = 'target'
|
|
|
|
# Title of notification
|
|
ATTR_TITLE = 'title'
|
|
ATTR_TITLE_DEFAULT = "Home Assistant"
|
|
|
|
DOMAIN = 'notify'
|
|
|
|
SERVICE_NOTIFY = 'notify'
|
|
|
|
PLATFORM_SCHEMA = vol.Schema({
|
|
vol.Required(CONF_PLATFORM): cv.string,
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
NOTIFY_SERVICE_SCHEMA = vol.Schema({
|
|
vol.Required(ATTR_MESSAGE): cv.template,
|
|
vol.Optional(ATTR_TITLE): cv.template,
|
|
vol.Optional(ATTR_TARGET): vol.All(cv.ensure_list, [cv.string]),
|
|
vol.Optional(ATTR_DATA): dict,
|
|
})
|
|
|
|
|
|
def send_message(hass, message, title=None, data=None):
|
|
"""Send a notification message."""
|
|
info = {
|
|
ATTR_MESSAGE: message
|
|
}
|
|
|
|
if title is not None:
|
|
info[ATTR_TITLE] = title
|
|
|
|
if data is not None:
|
|
info[ATTR_DATA] = data
|
|
|
|
hass.services.call(DOMAIN, SERVICE_NOTIFY, info)
|
|
|
|
|
|
def setup(hass, config):
|
|
"""Setup the notify services."""
|
|
descriptions = load_yaml_config_file(
|
|
os.path.join(os.path.dirname(__file__), 'services.yaml'))
|
|
|
|
targets = {}
|
|
|
|
def setup_notify_platform(platform, p_config=None, discovery_info=None):
|
|
"""Set up a notify platform."""
|
|
if p_config is None:
|
|
p_config = {}
|
|
if discovery_info is None:
|
|
discovery_info = {}
|
|
|
|
notify_implementation = bootstrap.prepare_setup_platform(
|
|
hass, config, DOMAIN, platform)
|
|
|
|
if notify_implementation is None:
|
|
_LOGGER.error("Unknown notification service specified")
|
|
return False
|
|
|
|
notify_service = notify_implementation.get_service(
|
|
hass, p_config, discovery_info)
|
|
|
|
if notify_service is None:
|
|
_LOGGER.error("Failed to initialize notification service %s",
|
|
platform)
|
|
return False
|
|
|
|
def notify_message(notify_service, call):
|
|
"""Handle sending notification message service calls."""
|
|
kwargs = {}
|
|
message = call.data[ATTR_MESSAGE]
|
|
title = call.data.get(ATTR_TITLE)
|
|
|
|
if title:
|
|
title.hass = hass
|
|
kwargs[ATTR_TITLE] = title.render()
|
|
|
|
if targets.get(call.service) is not None:
|
|
kwargs[ATTR_TARGET] = [targets[call.service]]
|
|
elif call.data.get(ATTR_TARGET) is not None:
|
|
kwargs[ATTR_TARGET] = call.data.get(ATTR_TARGET)
|
|
|
|
message.hass = hass
|
|
kwargs[ATTR_MESSAGE] = message.render()
|
|
kwargs[ATTR_DATA] = call.data.get(ATTR_DATA)
|
|
|
|
notify_service.send_message(**kwargs)
|
|
|
|
service_call_handler = partial(notify_message, notify_service)
|
|
|
|
if hasattr(notify_service, 'targets'):
|
|
platform_name = (
|
|
p_config.get(CONF_NAME) or discovery_info.get(CONF_NAME) or
|
|
platform)
|
|
for name, target in notify_service.targets.items():
|
|
target_name = slugify('{}_{}'.format(platform_name, name))
|
|
targets[target_name] = target
|
|
hass.services.register(DOMAIN, target_name,
|
|
service_call_handler,
|
|
descriptions.get(SERVICE_NOTIFY),
|
|
schema=NOTIFY_SERVICE_SCHEMA)
|
|
|
|
platform_name = (
|
|
p_config.get(CONF_NAME) or discovery_info.get(CONF_NAME) or
|
|
SERVICE_NOTIFY)
|
|
platform_name_slug = slugify(platform_name)
|
|
|
|
hass.services.register(
|
|
DOMAIN, platform_name_slug, service_call_handler,
|
|
descriptions.get(SERVICE_NOTIFY), schema=NOTIFY_SERVICE_SCHEMA)
|
|
|
|
return True
|
|
|
|
for platform, p_config in config_per_platform(config, DOMAIN):
|
|
if not setup_notify_platform(platform, p_config):
|
|
_LOGGER.error("Failed to set up platform %s", platform)
|
|
continue
|
|
|
|
def platform_discovered(platform, info):
|
|
"""Callback to load a platform."""
|
|
setup_notify_platform(platform, discovery_info=info)
|
|
|
|
discovery.listen_platform(hass, DOMAIN, platform_discovered)
|
|
|
|
return True
|
|
|
|
|
|
class BaseNotificationService(object):
|
|
"""An abstract class for notification services."""
|
|
|
|
def send_message(self, message, **kwargs):
|
|
"""Send a message.
|
|
|
|
kwargs can contain ATTR_TITLE to specify a title.
|
|
"""
|
|
raise NotImplementedError
|