2019-04-03 15:40:03 +00:00
|
|
|
"""Slack platform for notify component."""
|
2020-03-31 03:32:29 +00:00
|
|
|
import asyncio
|
2015-07-31 20:45:41 +00:00
|
|
|
import logging
|
2020-03-31 03:32:29 +00:00
|
|
|
import os
|
|
|
|
from urllib.parse import urlparse
|
2017-08-21 08:23:29 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
from slack import WebClient
|
|
|
|
from slack.errors import SlackApiError
|
2016-09-03 15:01:05 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-03-28 03:36:13 +00:00
|
|
|
from homeassistant.components.notify import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_DATA,
|
|
|
|
ATTR_TARGET,
|
|
|
|
ATTR_TITLE,
|
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
BaseNotificationService,
|
|
|
|
)
|
2019-10-17 04:36:19 +00:00
|
|
|
from homeassistant.const import CONF_API_KEY, CONF_ICON, CONF_USERNAME
|
2020-03-31 03:32:29 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.helpers import aiohttp_client, config_validation as cv
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ATTACHMENTS = "attachments"
|
2020-03-31 03:32:29 +00:00
|
|
|
ATTR_BLOCKS = "blocks"
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_FILE = "file"
|
2020-03-31 03:32:29 +00:00
|
|
|
|
|
|
|
CONF_DEFAULT_CHANNEL = "default_channel"
|
|
|
|
|
|
|
|
DEFAULT_TIMEOUT_SECONDS = 15
|
2016-09-03 15:01:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
2020-03-31 03:32:29 +00:00
|
|
|
vol.Required(CONF_DEFAULT_CHANNEL): cv.string,
|
2019-07-31 19:25:30 +00:00
|
|
|
vol.Optional(CONF_ICON): cv.string,
|
|
|
|
vol.Optional(CONF_USERNAME): cv.string,
|
|
|
|
}
|
|
|
|
)
|
2016-09-03 15:01:05 +00:00
|
|
|
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
async def async_get_service(hass, config, discovery_info=None):
|
|
|
|
"""Set up the Slack notification service."""
|
|
|
|
session = aiohttp_client.async_get_clientsession(hass)
|
|
|
|
client = WebClient(token=config[CONF_API_KEY], run_async=True, session=session)
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
try:
|
2020-03-31 03:32:29 +00:00
|
|
|
await client.auth_test()
|
|
|
|
except SlackApiError as err:
|
|
|
|
_LOGGER.error("Error while setting up integration: %s", err)
|
|
|
|
return
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
return SlackNotificationService(
|
|
|
|
hass,
|
|
|
|
client,
|
|
|
|
config[CONF_DEFAULT_CHANNEL],
|
|
|
|
username=config.get(CONF_USERNAME),
|
|
|
|
icon=config.get(CONF_ICON),
|
|
|
|
)
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
@callback
|
|
|
|
def _async_sanitize_channel_names(channel_list):
|
|
|
|
"""Remove any # symbols from a channel list."""
|
|
|
|
return [channel.lstrip("#") for channel in channel_list]
|
|
|
|
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
class SlackNotificationService(BaseNotificationService):
|
|
|
|
"""Define the Slack notification logic."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
def __init__(self, hass, client, default_channel, username, icon):
|
|
|
|
"""Initialize."""
|
|
|
|
self._client = client
|
2015-07-31 20:45:41 +00:00
|
|
|
self._default_channel = default_channel
|
2020-03-31 03:32:29 +00:00
|
|
|
self._hass = hass
|
2016-09-12 07:49:26 +00:00
|
|
|
self._icon = icon
|
2020-04-13 21:32:23 +00:00
|
|
|
self._username = username
|
2016-09-12 07:49:26 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
async def _async_send_local_file_message(self, path, targets, message, title):
|
|
|
|
"""Upload a local file (with message) to Slack."""
|
|
|
|
if not self._hass.config.is_allowed_path(path):
|
|
|
|
_LOGGER.error("Path does not exist or is not allowed: %s", path)
|
|
|
|
return
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
parsed_url = urlparse(path)
|
|
|
|
filename = os.path.basename(parsed_url.path)
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
try:
|
|
|
|
await self._client.files_upload(
|
|
|
|
channels=",".join(targets),
|
|
|
|
file=path,
|
|
|
|
filename=filename,
|
|
|
|
initial_comment=message,
|
|
|
|
title=title or filename,
|
|
|
|
)
|
|
|
|
except SlackApiError as err:
|
|
|
|
_LOGGER.error("Error while uploading file-based message: %s", err)
|
|
|
|
|
|
|
|
async def _async_send_text_only_message(
|
|
|
|
self, targets, message, title, attachments, blocks
|
|
|
|
):
|
|
|
|
"""Send a text-only message."""
|
|
|
|
tasks = {
|
|
|
|
target: self._client.chat_postMessage(
|
|
|
|
channel=target,
|
|
|
|
text=message,
|
|
|
|
attachments=attachments,
|
|
|
|
blocks=blocks,
|
|
|
|
icon_emoji=self._icon,
|
|
|
|
link_names=True,
|
2020-04-13 21:32:23 +00:00
|
|
|
username=self._username,
|
2020-03-31 03:32:29 +00:00
|
|
|
)
|
|
|
|
for target in targets
|
|
|
|
}
|
|
|
|
|
|
|
|
results = await asyncio.gather(*tasks.values(), return_exceptions=True)
|
|
|
|
for target, result in zip(tasks, results):
|
|
|
|
if isinstance(result, SlackApiError):
|
|
|
|
_LOGGER.error(
|
|
|
|
"There was a Slack API error while sending to %s: %s",
|
|
|
|
target,
|
|
|
|
result,
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_send_message(self, message, **kwargs):
|
|
|
|
"""Send a message to Slack."""
|
|
|
|
data = kwargs[ATTR_DATA] or {}
|
2017-07-07 06:14:24 +00:00
|
|
|
title = kwargs.get(ATTR_TITLE)
|
2020-03-31 03:32:29 +00:00
|
|
|
targets = _async_sanitize_channel_names(
|
|
|
|
kwargs.get(ATTR_TARGET, [self._default_channel])
|
|
|
|
)
|
2016-08-21 18:54:28 +00:00
|
|
|
|
2020-03-31 03:32:29 +00:00
|
|
|
if ATTR_FILE in data:
|
|
|
|
return await self._async_send_local_file_message(
|
|
|
|
data[ATTR_FILE], targets, message, title
|
|
|
|
)
|
|
|
|
|
|
|
|
attachments = data.get(ATTR_ATTACHMENTS, {})
|
|
|
|
if attachments:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Attachments are deprecated and part of Slack's legacy API; support "
|
|
|
|
"for them will be dropped in 0.114.0. In most cases, Blocks should be "
|
|
|
|
"used instead: https://www.home-assistant.io/integrations/slack/"
|
|
|
|
)
|
|
|
|
blocks = data.get(ATTR_BLOCKS, {})
|
|
|
|
|
|
|
|
return await self._async_send_text_only_message(
|
|
|
|
targets, message, title, attachments, blocks
|
|
|
|
)
|