2015-07-31 20:45:41 +00:00
|
|
|
"""
|
|
|
|
Slack platform for notify component.
|
|
|
|
|
2015-10-13 20:45:36 +00:00
|
|
|
For more details about this platform, please refer to the documentation at
|
2015-11-09 17:33:11 +00:00
|
|
|
https://home-assistant.io/components/notify.slack/
|
2015-07-31 20:45:41 +00:00
|
|
|
"""
|
|
|
|
import logging
|
2017-07-07 06:14:24 +00:00
|
|
|
import requests
|
|
|
|
from requests.auth import HTTPDigestAuth
|
|
|
|
from requests.auth import HTTPBasicAuth
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2016-09-03 15:01:05 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components.notify import (
|
2017-07-07 06:14:24 +00:00
|
|
|
ATTR_TARGET, ATTR_TITLE, ATTR_DATA,
|
|
|
|
PLATFORM_SCHEMA, BaseNotificationService)
|
2016-09-12 07:49:26 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_API_KEY, CONF_USERNAME, CONF_ICON)
|
2016-09-03 15:01:05 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2017-05-29 08:26:56 +00:00
|
|
|
REQUIREMENTS = ['slacker==0.9.50']
|
2016-09-03 15:01:05 +00:00
|
|
|
|
2015-07-31 20:45:41 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-09-03 15:01:05 +00:00
|
|
|
CONF_CHANNEL = 'default_channel'
|
2017-07-07 06:14:24 +00:00
|
|
|
CONF_TIMEOUT = 15
|
|
|
|
|
|
|
|
# Top level attributes in 'data'
|
|
|
|
ATTR_ATTACHMENTS = 'attachments'
|
|
|
|
ATTR_FILE = 'file'
|
|
|
|
# Attributes contained in file
|
|
|
|
ATTR_FILE_URL = 'url'
|
|
|
|
ATTR_FILE_PATH = 'path'
|
|
|
|
ATTR_FILE_USERNAME = 'username'
|
|
|
|
ATTR_FILE_PASSWORD = 'password'
|
|
|
|
ATTR_FILE_AUTH = 'auth'
|
|
|
|
# Any other value or absense of 'auth' lead to basic authentication being used
|
|
|
|
ATTR_FILE_AUTH_DIGEST = 'digest'
|
2016-09-03 15:01:05 +00:00
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Required(CONF_CHANNEL): cv.string,
|
2016-09-12 07:49:26 +00:00
|
|
|
vol.Optional(CONF_USERNAME): cv.string,
|
|
|
|
vol.Optional(CONF_ICON): cv.string,
|
2016-09-03 15:01:05 +00:00
|
|
|
})
|
|
|
|
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
# pylint: disable=unused-variable
|
2017-01-15 02:53:14 +00:00
|
|
|
def get_service(hass, config, discovery_info=None):
|
2016-03-08 10:46:32 +00:00
|
|
|
"""Get the Slack notification service."""
|
2015-11-09 06:15:34 +00:00
|
|
|
import slacker
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
return SlackNotificationService(
|
2016-09-03 15:01:05 +00:00
|
|
|
config[CONF_CHANNEL],
|
2016-09-12 07:49:26 +00:00
|
|
|
config[CONF_API_KEY],
|
|
|
|
config.get(CONF_USERNAME, None),
|
2017-07-07 06:14:24 +00:00
|
|
|
config.get(CONF_ICON, None),
|
|
|
|
hass.config.is_allowed_path)
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2015-11-09 06:15:34 +00:00
|
|
|
except slacker.Error:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.exception("Authentication failed")
|
2015-11-09 06:15:34 +00:00
|
|
|
return None
|
2015-07-31 20:45:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
class SlackNotificationService(BaseNotificationService):
|
2016-03-08 10:46:32 +00:00
|
|
|
"""Implement the notification service for Slack."""
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2017-07-07 06:14:24 +00:00
|
|
|
def __init__(self, default_channel,
|
|
|
|
api_token, username,
|
|
|
|
icon, is_allowed_path):
|
2016-03-08 10:46:32 +00:00
|
|
|
"""Initialize the service."""
|
2015-07-31 20:45:41 +00:00
|
|
|
from slacker import Slacker
|
|
|
|
self._default_channel = default_channel
|
|
|
|
self._api_token = api_token
|
2016-09-12 07:49:26 +00:00
|
|
|
self._username = username
|
|
|
|
self._icon = icon
|
|
|
|
if self._username or self._icon:
|
|
|
|
self._as_user = False
|
|
|
|
else:
|
|
|
|
self._as_user = True
|
|
|
|
|
2017-07-07 06:14:24 +00:00
|
|
|
self.is_allowed_path = is_allowed_path
|
2015-07-31 20:45:41 +00:00
|
|
|
self.slack = Slacker(self._api_token)
|
|
|
|
self.slack.auth.test()
|
|
|
|
|
|
|
|
def send_message(self, message="", **kwargs):
|
2016-03-08 10:46:32 +00:00
|
|
|
"""Send a message to a user."""
|
2015-11-09 06:15:34 +00:00
|
|
|
import slacker
|
2015-07-31 20:45:41 +00:00
|
|
|
|
2016-10-12 04:16:11 +00:00
|
|
|
if kwargs.get(ATTR_TARGET) is None:
|
|
|
|
targets = [self._default_channel]
|
|
|
|
else:
|
|
|
|
targets = kwargs.get(ATTR_TARGET)
|
2016-10-02 05:19:17 +00:00
|
|
|
|
2017-07-07 06:14:24 +00:00
|
|
|
data = kwargs.get(ATTR_DATA)
|
|
|
|
attachments = data.get(ATTR_ATTACHMENTS) if data else None
|
|
|
|
file = data.get(ATTR_FILE) if data else None
|
|
|
|
title = kwargs.get(ATTR_TITLE)
|
2016-08-21 18:54:28 +00:00
|
|
|
|
2016-10-02 05:19:17 +00:00
|
|
|
for target in targets:
|
|
|
|
try:
|
2017-07-07 06:14:24 +00:00
|
|
|
if file is not None:
|
|
|
|
# Load from file or url
|
|
|
|
file_as_bytes = self.load_file(
|
|
|
|
url=file.get(ATTR_FILE_URL),
|
|
|
|
local_path=file.get(ATTR_FILE_PATH),
|
|
|
|
username=file.get(ATTR_FILE_USERNAME),
|
|
|
|
password=file.get(ATTR_FILE_PASSWORD),
|
|
|
|
auth=file.get(ATTR_FILE_AUTH))
|
|
|
|
# Choose filename
|
|
|
|
if file.get(ATTR_FILE_URL):
|
|
|
|
filename = file.get(ATTR_FILE_URL)
|
|
|
|
else:
|
|
|
|
filename = file.get(ATTR_FILE_PATH)
|
|
|
|
# Prepare structure for slack API
|
|
|
|
data = {
|
|
|
|
'content': None,
|
|
|
|
'filetype': None,
|
|
|
|
'filename': filename,
|
|
|
|
# if optional title is none use the filename
|
|
|
|
'title': title if title else filename,
|
|
|
|
'initial_comment': message,
|
|
|
|
'channels': target
|
|
|
|
}
|
|
|
|
# Post to slack
|
|
|
|
self.slack.files.post('files.upload',
|
|
|
|
data=data,
|
|
|
|
files={'file': file_as_bytes})
|
|
|
|
else:
|
|
|
|
self.slack.chat.post_message(
|
|
|
|
target, message, as_user=self._as_user,
|
|
|
|
username=self._username, icon_emoji=self._icon,
|
|
|
|
attachments=attachments, link_names=True)
|
2016-10-02 05:19:17 +00:00
|
|
|
except slacker.Error as err:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Could not send notification. Error: %s", err)
|
2017-07-07 06:14:24 +00:00
|
|
|
|
|
|
|
def load_file(self, url=None, local_path=None,
|
|
|
|
username=None, password=None, auth=None):
|
|
|
|
"""Load image/document/etc from a local path or url."""
|
|
|
|
try:
|
|
|
|
if url is not None:
|
|
|
|
# check whether authentication parameters are provided
|
|
|
|
if username is not None and password is not None:
|
|
|
|
# Use digest or basic authentication
|
|
|
|
if ATTR_FILE_AUTH_DIGEST == auth:
|
|
|
|
auth_ = HTTPDigestAuth(username, password)
|
|
|
|
else:
|
|
|
|
auth_ = HTTPBasicAuth(username, password)
|
|
|
|
# load file from url with authentication
|
|
|
|
req = requests.get(url, auth=auth_, timeout=CONF_TIMEOUT)
|
|
|
|
else:
|
|
|
|
# load file from url without authentication
|
|
|
|
req = requests.get(url, timeout=CONF_TIMEOUT)
|
|
|
|
return req.content
|
|
|
|
|
|
|
|
elif local_path is not None:
|
|
|
|
# Check whether path is whitelisted in configuration.yaml
|
|
|
|
if self.is_allowed_path(local_path):
|
|
|
|
# load file from local path on server
|
|
|
|
return open(local_path, "rb")
|
|
|
|
_LOGGER.warning("'%s' is not secure to load data from!",
|
|
|
|
local_path)
|
|
|
|
else:
|
|
|
|
# neither url nor path provided
|
|
|
|
_LOGGER.warning("Neither url nor local path found in params!")
|
|
|
|
|
|
|
|
except OSError as error:
|
|
|
|
_LOGGER.error("Can't load from url or local path: %s", error)
|
|
|
|
|
|
|
|
return None
|