2016-01-21 01:02:32 +00:00
|
|
|
"""
|
|
|
|
Twitter platform for notify component.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/notify.twitter/
|
|
|
|
"""
|
2017-07-11 02:58:01 +00:00
|
|
|
import json
|
2016-01-21 01:02:32 +00:00
|
|
|
import logging
|
2017-07-11 02:58:01 +00:00
|
|
|
import mimetypes
|
|
|
|
import os
|
2016-02-19 05:27:50 +00:00
|
|
|
|
2016-09-02 04:27:38 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-10-29 16:12:43 +00:00
|
|
|
from homeassistant.components.notify import (
|
2017-07-11 02:58:01 +00:00
|
|
|
ATTR_DATA, PLATFORM_SCHEMA, BaseNotificationService)
|
2017-01-23 21:24:45 +00:00
|
|
|
from homeassistant.const import CONF_ACCESS_TOKEN, CONF_USERNAME
|
2016-01-21 01:02:32 +00:00
|
|
|
|
2017-03-02 04:57:51 +00:00
|
|
|
REQUIREMENTS = ['TwitterAPI==2.4.5']
|
2016-01-21 01:02:32 +00:00
|
|
|
|
2016-10-29 16:12:43 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CONF_CONSUMER_KEY = 'consumer_key'
|
|
|
|
CONF_CONSUMER_SECRET = 'consumer_secret'
|
|
|
|
CONF_ACCESS_TOKEN_SECRET = 'access_token_secret'
|
2016-01-21 01:02:32 +00:00
|
|
|
|
2017-07-11 02:58:01 +00:00
|
|
|
ATTR_MEDIA = 'media'
|
|
|
|
|
2016-09-02 04:27:38 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_ACCESS_TOKEN): cv.string,
|
|
|
|
vol.Required(CONF_ACCESS_TOKEN_SECRET): cv.string,
|
2017-01-23 21:24:45 +00:00
|
|
|
vol.Required(CONF_CONSUMER_KEY): cv.string,
|
|
|
|
vol.Required(CONF_CONSUMER_SECRET): cv.string,
|
|
|
|
vol.Optional(CONF_USERNAME): cv.string,
|
2016-09-02 04:27:38 +00:00
|
|
|
})
|
|
|
|
|
2016-01-21 01:02:32 +00:00
|
|
|
|
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 Twitter notification service."""
|
2016-10-29 16:12:43 +00:00
|
|
|
return TwitterNotificationService(
|
2017-07-11 02:58:01 +00:00
|
|
|
hass,
|
2016-10-29 16:12:43 +00:00
|
|
|
config[CONF_CONSUMER_KEY], config[CONF_CONSUMER_SECRET],
|
2017-01-23 21:24:45 +00:00
|
|
|
config[CONF_ACCESS_TOKEN], config[CONF_ACCESS_TOKEN_SECRET],
|
|
|
|
config.get(CONF_USERNAME)
|
2016-10-29 16:12:43 +00:00
|
|
|
)
|
2016-01-21 01:02:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TwitterNotificationService(BaseNotificationService):
|
2016-10-29 16:12:43 +00:00
|
|
|
"""Implementation of a notification service for the Twitter service."""
|
2016-01-21 01:02:32 +00:00
|
|
|
|
2017-07-11 02:58:01 +00:00
|
|
|
def __init__(self, hass, consumer_key, consumer_secret, access_token_key,
|
2017-01-23 21:24:45 +00:00
|
|
|
access_token_secret, username):
|
2016-03-08 10:46:32 +00:00
|
|
|
"""Initialize the service."""
|
2016-01-21 01:02:32 +00:00
|
|
|
from TwitterAPI import TwitterAPI
|
2017-01-23 21:24:45 +00:00
|
|
|
self.user = username
|
2017-07-11 02:58:01 +00:00
|
|
|
self.hass = hass
|
2016-01-21 01:02:32 +00:00
|
|
|
self.api = TwitterAPI(consumer_key, consumer_secret, access_token_key,
|
|
|
|
access_token_secret)
|
|
|
|
|
|
|
|
def send_message(self, message="", **kwargs):
|
2017-07-11 02:58:01 +00:00
|
|
|
"""Tweet a message, optionally with media."""
|
|
|
|
data = kwargs.get(ATTR_DATA)
|
2017-07-17 17:45:42 +00:00
|
|
|
|
|
|
|
media = None
|
|
|
|
if data:
|
|
|
|
media = data.get(ATTR_MEDIA)
|
|
|
|
if not self.hass.config.is_allowed_path(media):
|
|
|
|
_LOGGER.warning("'%s' is not in a whitelisted area.", media)
|
|
|
|
return
|
2017-07-11 02:58:01 +00:00
|
|
|
|
|
|
|
media_id = self.upload_media(media)
|
|
|
|
|
2017-01-23 21:24:45 +00:00
|
|
|
if self.user:
|
2017-07-11 02:58:01 +00:00
|
|
|
resp = self.api.request('direct_messages/new',
|
|
|
|
{'text': message, 'user': self.user,
|
|
|
|
'media_ids': media_id})
|
2017-01-23 21:24:45 +00:00
|
|
|
else:
|
2017-07-11 02:58:01 +00:00
|
|
|
resp = self.api.request('statuses/update',
|
|
|
|
{'status': message, 'media_ids': media_id})
|
2017-01-23 21:24:45 +00:00
|
|
|
|
2016-01-21 01:02:32 +00:00
|
|
|
if resp.status_code != 200:
|
2017-07-11 02:58:01 +00:00
|
|
|
self.log_error_resp(resp)
|
|
|
|
|
|
|
|
def upload_media(self, media_path=None):
|
|
|
|
"""Upload media."""
|
|
|
|
if not media_path:
|
|
|
|
return None
|
|
|
|
|
|
|
|
(media_type, _) = mimetypes.guess_type(media_path)
|
|
|
|
total_bytes = os.path.getsize(media_path)
|
|
|
|
|
|
|
|
file = open(media_path, 'rb')
|
|
|
|
resp = self.upload_media_init(media_type, total_bytes)
|
|
|
|
|
|
|
|
if 199 > resp.status_code < 300:
|
|
|
|
self.log_error_resp(resp)
|
|
|
|
return None
|
|
|
|
|
|
|
|
media_id = resp.json()['media_id']
|
|
|
|
media_id = self.upload_media_chunked(file, total_bytes,
|
|
|
|
media_id)
|
|
|
|
|
|
|
|
resp = self.upload_media_finalize(media_id)
|
|
|
|
if 199 > resp.status_code < 300:
|
|
|
|
self.log_error_resp(resp)
|
|
|
|
|
|
|
|
return media_id
|
|
|
|
|
|
|
|
def upload_media_init(self, media_type, total_bytes):
|
|
|
|
"""Upload media, INIT phase."""
|
|
|
|
resp = self.api.request('media/upload',
|
|
|
|
{'command': 'INIT', 'media_type': media_type,
|
|
|
|
'total_bytes': total_bytes})
|
|
|
|
return resp
|
|
|
|
|
|
|
|
def upload_media_chunked(self, file, total_bytes, media_id):
|
|
|
|
"""Upload media, chunked append."""
|
|
|
|
segment_id = 0
|
|
|
|
bytes_sent = 0
|
|
|
|
while bytes_sent < total_bytes:
|
|
|
|
chunk = file.read(4 * 1024 * 1024)
|
|
|
|
resp = self.upload_media_append(chunk, media_id, segment_id)
|
|
|
|
if resp.status_code not in range(200, 299):
|
|
|
|
self.log_error_resp_append(resp)
|
|
|
|
return None
|
|
|
|
segment_id = segment_id + 1
|
|
|
|
bytes_sent = file.tell()
|
|
|
|
self.log_bytes_sent(bytes_sent, total_bytes)
|
|
|
|
return media_id
|
|
|
|
|
|
|
|
def upload_media_append(self, chunk, media_id, segment_id):
|
|
|
|
"""Upload media, append phase."""
|
|
|
|
return self.api.request('media/upload',
|
|
|
|
{'command': 'APPEND', 'media_id': media_id,
|
|
|
|
'segment_index': segment_id},
|
|
|
|
{'media': chunk})
|
|
|
|
|
|
|
|
def upload_media_finalize(self, media_id):
|
|
|
|
"""Upload media, finalize phase."""
|
|
|
|
return self.api.request('media/upload',
|
|
|
|
{'command': 'FINALIZE', 'media_id': media_id})
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def log_bytes_sent(bytes_sent, total_bytes):
|
|
|
|
"""Log upload progress."""
|
|
|
|
_LOGGER.debug("%s of %s bytes uploaded", str(bytes_sent),
|
|
|
|
str(total_bytes))
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def log_error_resp(resp):
|
|
|
|
"""Log error response."""
|
|
|
|
obj = json.loads(resp.text)
|
|
|
|
error_message = obj['error']
|
|
|
|
_LOGGER.error("Error %s : %s", resp.status_code, error_message)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def log_error_resp_append(resp):
|
|
|
|
"""Log error response, during upload append phase."""
|
|
|
|
obj = json.loads(resp.text)
|
|
|
|
error_message = obj['errors'][0]['message']
|
|
|
|
error_code = obj['errors'][0]['code']
|
|
|
|
_LOGGER.error("Error %s : %s (Code %s)", resp.status_code,
|
|
|
|
error_message, error_code)
|