Support Slack URLs missing channel references (#108)

This commit is contained in:
Chris Caron 2019-05-11 15:02:56 -04:00 committed by GitHub
parent 333926c359
commit 51a5ce344a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 54 additions and 68 deletions

View File

@ -57,7 +57,7 @@ The table below identifies the services this tool supports and some example serv
| [Pushover](https://github.com/caronc/apprise/wiki/Notify_pushover) | pover:// | (TCP) 443 | pover://user@token<br />pover://user@token/DEVICE<br />pover://user@token/DEVICE1/DEVICE2/DEVICEN<br />**Note**: you must specify both your user_id and token | [Pushover](https://github.com/caronc/apprise/wiki/Notify_pushover) | pover:// | (TCP) 443 | pover://user@token<br />pover://user@token/DEVICE<br />pover://user@token/DEVICE1/DEVICE2/DEVICEN<br />**Note**: you must specify both your user_id and token
| [Rocket.Chat](https://github.com/caronc/apprise/wiki/Notify_rocketchat) | rocket:// or rockets:// | (TCP) 80 or 443 | rocket://user:password@hostname/RoomID/Channel<br />rockets://user:password@hostname:443/Channel1/Channel1/RoomID<br />rocket://user:password@hostname/Channel | [Rocket.Chat](https://github.com/caronc/apprise/wiki/Notify_rocketchat) | rocket:// or rockets:// | (TCP) 80 or 443 | rocket://user:password@hostname/RoomID/Channel<br />rockets://user:password@hostname:443/Channel1/Channel1/RoomID<br />rocket://user:password@hostname/Channel
| [Ryver](https://github.com/caronc/apprise/wiki/Notify_ryver) | ryver:// | (TCP) 443 | ryver://Organization/Token<br />ryver://botname@Organization/Token | [Ryver](https://github.com/caronc/apprise/wiki/Notify_ryver) | ryver:// | (TCP) 443 | ryver://Organization/Token<br />ryver://botname@Organization/Token
| [Slack](https://github.com/caronc/apprise/wiki/Notify_slack) | slack:// | (TCP) 443 | slack://TokenA/TokenB/TokenC/Channel<br />slack://botname@TokenA/TokenB/TokenC/Channel<br />slack://user@TokenA/TokenB/TokenC/Channel1/Channel2/ChannelN | [Slack](https://github.com/caronc/apprise/wiki/Notify_slack) | slack:// | (TCP) 443 | slack://TokenA/TokenB/TokenC/<br />slack://TokenA/TokenB/TokenC/Channel<br />slack://botname@TokenA/TokenB/TokenC/Channel<br />slack://user@TokenA/TokenB/TokenC/Channel1/Channel2/ChannelN
| [Telegram](https://github.com/caronc/apprise/wiki/Notify_telegram) | tgram:// | (TCP) 443 | tgram://bottoken/ChatID<br />tgram://bottoken/ChatID1/ChatID2/ChatIDN | [Telegram](https://github.com/caronc/apprise/wiki/Notify_telegram) | tgram:// | (TCP) 443 | tgram://bottoken/ChatID<br />tgram://bottoken/ChatID1/ChatID2/ChatIDN
| [Twilio](https://github.com/caronc/apprise/wiki/Notify_twilio) | twilio:// | (TCP) 443 | twilio://AccountSid:AuthToken@FromPhoneNo<br/>twilio://AccountSid:AuthToken@FromPhoneNo/ToPhoneNo<br/>twilio://AccountSid:AuthToken@FromPhoneNo/ToPhoneNo1/ToPhoneNo2/ToPhoneNoN/<br/>twilio://AccountSid:AuthToken@ShortCode/ToPhoneNo<br/>twilio://AccountSid:AuthToken@ShortCode/ToPhoneNo1/ToPhoneNo2/ToPhoneNoN/ | [Twilio](https://github.com/caronc/apprise/wiki/Notify_twilio) | twilio:// | (TCP) 443 | twilio://AccountSid:AuthToken@FromPhoneNo<br/>twilio://AccountSid:AuthToken@FromPhoneNo/ToPhoneNo<br/>twilio://AccountSid:AuthToken@FromPhoneNo/ToPhoneNo1/ToPhoneNo2/ToPhoneNoN/<br/>twilio://AccountSid:AuthToken@ShortCode/ToPhoneNo<br/>twilio://AccountSid:AuthToken@ShortCode/ToPhoneNo1/ToPhoneNo2/ToPhoneNoN/
| [Twitter](https://github.com/caronc/apprise/wiki/Notify_twitter) | tweet:// | (TCP) 443 | tweet://user@CKey/CSecret/AKey/ASecret | [Twitter](https://github.com/caronc/apprise/wiki/Notify_twitter) | tweet:// | (TCP) 443 | tweet://user@CKey/CSecret/AKey/ASecret

View File

@ -36,7 +36,6 @@
# #
# #
import re import re
import six
import requests import requests
from json import dumps from json import dumps
from time import time from time import time
@ -46,6 +45,7 @@ from ..common import NotifyImageSize
from ..common import NotifyType from ..common import NotifyType
from ..common import NotifyFormat from ..common import NotifyFormat
from ..utils import parse_bool from ..utils import parse_bool
from ..utils import parse_list
# Token required as part of the API request # Token required as part of the API request
# /AAAAAAAAA/........./........................ # /AAAAAAAAA/........./........................
@ -155,21 +155,13 @@ class NotifySlack(NotifyBase):
self.logger.warning( self.logger.warning(
'No user was specified; using %s.' % SLACK_DEFAULT_USER) 'No user was specified; using %s.' % SLACK_DEFAULT_USER)
if isinstance(targets, six.string_types): # Build list of channels
self.channels = [x for x in filter(bool, CHANNEL_LIST_DELIM.split( self.channels = parse_list(targets)
targets,
))]
elif isinstance(targets, (set, tuple, list)):
self.channels = targets
else:
self.channels = list()
if len(self.channels) == 0: if len(self.channels) == 0:
msg = 'No channel(s) were specified.' # No problem; the webhook is smart enough to just notify the
self.logger.warning(msg) # channel it was created for; adding 'None' is just used as
raise TypeError(msg) # a flag lower to not set the channels
self.channels.append(None)
# Formatting requirements are defined here: # Formatting requirements are defined here:
# https://api.slack.com/docs/message-formatting # https://api.slack.com/docs/message-formatting
@ -218,47 +210,48 @@ class NotifySlack(NotifyBase):
self.token_c, self.token_c,
) )
# prepare JSON Object
payload = {
'username': self.user if self.user else SLACK_DEFAULT_USER,
# Use Markdown language
'mrkdwn': (self.notify_format == NotifyFormat.MARKDOWN),
'attachments': [{
'title': title,
'text': body,
'color': self.color(notify_type),
# Time
'ts': time(),
'footer': self.app_id,
}],
}
# Create a copy of the channel list # Create a copy of the channel list
channels = list(self.channels) channels = list(self.channels)
while len(channels): while len(channels):
channel = channels.pop(0) channel = channels.pop(0)
if not IS_CHANNEL_RE.match(channel):
self.logger.warning(
"The specified channel '%s' is invalid; skipping." % (
channel,
)
)
# Mark our failure
has_error = True
continue
if len(channel) > 1 and channel[0] == '+': if channel is not None:
# Treat as encoded id if prefixed with a + # Channel over-ride was specified
_channel = channel[1:] if not IS_CHANNEL_RE.match(channel):
self.logger.warning(
"The specified target {} is invalid;"
"skipping.".format(channel))
elif len(channel) > 1 and channel[0] == '@': # Mark our failure
# Treat @ value 'as is' has_error = True
_channel = channel continue
else: if len(channel) > 1 and channel[0] == '+':
# Prefix with channel hash tag # Treat as encoded id if prefixed with a +
_channel = '#%s' % channel payload['channel'] = channel[1:]
# prepare JSON Object elif len(channel) > 1 and channel[0] == '@':
payload = { # Treat @ value 'as is'
'channel': _channel, payload['channel'] = channel
'username': self.user if self.user else SLACK_DEFAULT_USER,
# Use Markdown language else:
'mrkdwn': (self.notify_format == NotifyFormat.MARKDOWN), # Prefix with channel hash tag
'attachments': [{ payload['channel'] = '#%s' % channel
'title': title,
'text': body,
'color': self.color(notify_type),
# Time
'ts': time(),
'footer': self.app_id,
}],
}
# Acquire our to-be footer icon if configured to do so # Acquire our to-be footer icon if configured to do so
image_url = None if not self.include_image \ image_url = None if not self.include_image \
@ -288,9 +281,10 @@ class NotifySlack(NotifyBase):
r.status_code, SLACK_HTTP_ERROR_MAP) r.status_code, SLACK_HTTP_ERROR_MAP)
self.logger.warning( self.logger.warning(
'Failed to send Slack notification to {}: ' 'Failed to send Slack notification{}: '
'{}{}error={}.'.format( '{}{}error={}.'.format(
channel, ' to {}'.format(channel)
if channel is not None else '',
status_str, status_str,
', ' if status_str else '', ', ' if status_str else '',
r.status_code)) r.status_code))
@ -304,13 +298,16 @@ class NotifySlack(NotifyBase):
else: else:
self.logger.info( self.logger.info(
'Sent Slack notification to {}.'.format(channel)) 'Sent Slack notification{}.'.format(
' to {}'.format(channel)
if channel is not None else ''))
except requests.RequestException as e: except requests.RequestException as e:
self.logger.warning( self.logger.warning(
'A Connection error occured sending Slack:%s ' % ( 'A Connection error occured sending Slack '
channel) + 'notification.' 'notification{}.'.format(
) ' to {}'.format(channel)
if channel is not None else ''))
self.logger.debug('Socket Exception: %s' % str(e)) self.logger.debug('Socket Exception: %s' % str(e))
# Mark our failure # Mark our failure

View File

@ -1671,8 +1671,8 @@ TEST_URLS = (
'instance': plugins.NotifySlack, 'instance': plugins.NotifySlack,
}), }),
('slack://username@T1JJ3T3L2/A1BRTD4JD/TIiajkdnlazkcOXrIdevi7FQ', { ('slack://username@T1JJ3T3L2/A1BRTD4JD/TIiajkdnlazkcOXrIdevi7FQ', {
# Missing a channel # Missing a channel, falls back to webhook channel bindings
'instance': TypeError, 'instance': plugins.NotifySlack,
}), }),
('slack://username@INVALID/A1BRTD4JD/TIiajkdnlazkcOXrIdevi7FQ/#cool', { ('slack://username@INVALID/A1BRTD4JD/TIiajkdnlazkcOXrIdevi7FQ/#cool', {
# invalid 1st Token # invalid 1st Token
@ -3157,17 +3157,6 @@ def test_notify_slack_plugin(mock_post, mock_get):
mock_post.return_value.status_code = requests.codes.ok mock_post.return_value.status_code = requests.codes.ok
mock_get.return_value.status_code = requests.codes.ok mock_get.return_value.status_code = requests.codes.ok
# Empty Channel list
try:
plugins.NotifySlack(
token_a=token_a, token_b=token_b, token_c=token_c,
targets=None)
assert False
except TypeError:
# we'll thrown because an empty list of channels was provided
assert True
# Missing first Token # Missing first Token
try: try:
plugins.NotifySlack( plugins.NotifySlack(