# -*- coding: utf-8 -*- # # anotify.py # Copyright (c) 2012 magnific0 # # based on: # growl.py # Copyright (c) 2011 Sorin Ionescu # # Permission is hereby granted, free of charge, to any person obtaining a copy # of this software and associated documentation files (the "Software"), to deal # in the Software without restriction, including without limitation the rights # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell # copies of the Software, and to permit persons to whom the Software is # furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included in # all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE # SOFTWARE. SCRIPT_NAME = 'anotify' SCRIPT_AUTHOR = 'magnific0' SCRIPT_VERSION = '1.0.2' SCRIPT_LICENSE = 'MIT' SCRIPT_DESC = 'Sends libnotify notifications upon events.' # Changelog # 2014-05-10: v1.0.1 Change hook_print callback argument type of # displayed/highlight (WeeChat >= 1.0) # 2012-09-20: v1.0.0 Forked from original and adapted for libnotify. # ----------------------------------------------------------------------------- # Settings # ----------------------------------------------------------------------------- SETTINGS = { 'show_public_message': 'off', 'show_private_message': 'on', 'show_public_action_message': 'off', 'show_private_action_message': 'on', 'show_notice_message': 'off', 'show_invite_message': 'on', 'show_highlighted_message': 'on', 'show_server': 'on', 'show_channel_topic': 'on', 'show_dcc': 'on', 'show_upgrade_ended': 'on', 'sticky': 'off', 'sticky_away': 'on', 'icon': '/usr/share/pixmaps/weechat.xpm', } # ----------------------------------------------------------------------------- # Imports # ----------------------------------------------------------------------------- try: import re import weechat import gi import notify2 import subprocess IMPORT_OK = True except ImportError as error: IMPORT_OK = False if str(error).find('weechat') != -1: print('This script must be run under WeeChat.') print('Get WeeChat at http://www.weechat.org.') else: weechat.prnt('', 'anotify: {0}'.format(error)) # ----------------------------------------------------------------------------- # Globals # ----------------------------------------------------------------------------- TAGGED_MESSAGES = { 'public message or action': set(['irc_privmsg', 'notify_message']), 'private message or action': set(['irc_privmsg', 'notify_private']), 'notice message': set(['irc_notice', 'notify_private']), 'invite message': set(['irc_invite', 'notify_highlight']), 'channel topic': set(['irc_topic', ]), #'away status': set(['away_info', ]), } UNTAGGED_MESSAGES = { 'away status': re.compile(r'^You ((\w+).){2,3}marked as being away', re.UNICODE), 'dcc chat request': re.compile(r'^xfer: incoming chat request from (\w+)', re.UNICODE), 'dcc chat closed': re.compile(r'^xfer: chat closed with (\w+)', re.UNICODE), 'dcc get request': re.compile( r'^xfer: incoming file from (\w+) [^:]+: ((?:,\w|[^,])+),', re.UNICODE), 'dcc get completed': re.compile(r'^xfer: file ([^\s]+) received from \w+: OK', re.UNICODE), 'dcc get failed': re.compile( r'^xfer: file ([^\s]+) received from \w+: FAILED', re.UNICODE), 'dcc send completed': re.compile(r'^xfer: file ([^\s]+) sent to \w+: OK', re.UNICODE), 'dcc send failed': re.compile(r'^xfer: file ([^\s]+) sent to \w+: FAILED', re.UNICODE), } DISPATCH_TABLE = { 'away status': 'set_away_status', 'public message or action': 'notify_public_message_or_action', 'private message or action': 'notify_private_message_or_action', 'notice message': 'notify_notice_message', 'invite message': 'notify_invite_message', 'channel topic': 'notify_channel_topic', 'dcc chat request': 'notify_dcc_chat_request', 'dcc chat closed': 'notify_dcc_chat_closed', 'dcc get request': 'notify_dcc_get_request', 'dcc get completed': 'notify_dcc_get_completed', 'dcc get failed': 'notify_dcc_get_failed', 'dcc send completed': 'notify_dcc_send_completed', 'dcc send failed': 'notify_dcc_send_failed', } STATE = { 'icon': None, 'is_away': False } # ----------------------------------------------------------------------------- # Notifiers # ----------------------------------------------------------------------------- def cb_irc_server_connected(data, signal, signal_data): '''Notify when connected to IRC server.''' if weechat.config_get_plugin('show_server') == 'on': a_notify( 'Server', 'Server Connected', 'Connected to network {0}.'.format(signal_data)) return weechat.WEECHAT_RC_OK def cb_irc_server_disconnected(data, signal, signal_data): '''Notify when disconnected to IRC server.''' if weechat.config_get_plugin('show_server') == 'on': a_notify( 'Server', 'Server Disconnected', 'Disconnected from network {0}.'.format(signal_data)) return weechat.WEECHAT_RC_OK def cb_notify_upgrade_ended(data, signal, signal_data): '''Notify on end of WeeChat upgrade.''' if weechat.config_get_plugin('show_upgrade_ended') == 'on': a_notify( 'WeeChat', 'WeeChat Upgraded', 'WeeChat has been upgraded.') return weechat.WEECHAT_RC_OK def notify_highlighted_message(prefix, message): '''Notify on highlighted message.''' if weechat.config_get_plugin("show_highlighted_message") == "on": a_notify( 'Highlight', 'Highlighted Message', "{0}: {1}".format(prefix, message), priority=notify2.URGENCY_CRITICAL) def notify_public_message_or_action(prefix, message, highlighted): '''Notify on public message or action.''' if prefix == ' *': regex = re.compile(r'^(\w+) (.+)$', re.UNICODE) match = regex.match(message) if match: prefix = match.group(1) message = match.group(2) notify_public_action_message(prefix, message, highlighted) else: if highlighted: notify_highlighted_message(prefix, message) elif weechat.config_get_plugin("show_public_message") == "on": a_notify( 'Public', 'Public Message', '{0}: {1}'.format(prefix, message)) def notify_private_message_or_action(prefix, message, highlighted): '''Notify on private message or action.''' regex = re.compile(r'^CTCP_MESSAGE.+?ACTION (.+)$', re.UNICODE) match = regex.match(message) if match: notify_private_action_message(prefix, match.group(1), highlighted) else: if prefix == ' *': regex = re.compile(r'^(\w+) (.+)$', re.UNICODE) match = regex.match(message) if match: prefix = match.group(1) message = match.group(2) notify_private_action_message(prefix, message, highlighted) else: if highlighted: notify_highlighted_message(prefix, message) elif weechat.config_get_plugin("show_private_message") == "on": a_notify( 'Private', 'Private Message', '{0}: {1}'.format(prefix, message)) def notify_public_action_message(prefix, message, highlighted): '''Notify on public action message.''' if highlighted: notify_highlighted_message(prefix, message) elif weechat.config_get_plugin("show_public_action_message") == "on": a_notify( 'Action', 'Public Action Message', '{0}: {1}'.format(prefix, message), priority=notify2.URGENCY_NORMAL) def notify_private_action_message(prefix, message, highlighted): '''Notify on private action message.''' if highlighted: notify_highlighted_message(prefix, message) elif weechat.config_get_plugin("show_private_action_message") == "on": a_notify( 'Action', 'Private Action Message', '{0}: {1}'.format(prefix, message), priority=notify2.URGENCY_NORMAL) def notify_notice_message(prefix, message, highlighted): '''Notify on notice message.''' regex = re.compile(r'^([^\s]*) [^:]*: (.+)$', re.UNICODE) match = regex.match(message) if match: prefix = match.group(1) message = match.group(2) if highlighted: notify_highlighted_message(prefix, message) elif weechat.config_get_plugin("show_notice_message") == "on": a_notify( 'Notice', 'Notice Message', '{0}: {1}'.format(prefix, message)) def notify_invite_message(prefix, message, highlighted): '''Notify on channel invitation message.''' if weechat.config_get_plugin("show_invite_message") == "on": regex = re.compile( r'^You have been invited to ([^\s]+) by ([^\s]+)$', re.UNICODE) match = regex.match(message) if match: channel = match.group(1) nick = match.group(2) a_notify( 'Invite', 'Channel Invitation', '{0} has invited you to join {1}.'.format(nick, channel)) def notify_channel_topic(prefix, message, highlighted): '''Notify on channel topic change.''' if weechat.config_get_plugin("show_channel_topic") == "on": regex = re.compile( r'^\w+ has (?:changed|unset) topic for ([^\s]+)' + '(?:(?: from "(?:(?:"\w|[^"])+)")? to "((?:"\w|[^"])+)")?', re.UNICODE) match = regex.match(message) if match: channel = match.group(1) topic = match.group(2) or '' a_notify( 'Channel', 'Channel Topic', "{0}: {1}".format(channel, topic)) def notify_dcc_chat_request(match): '''Notify on DCC chat request.''' if weechat.config_get_plugin("show_dcc") == "on": nick = match.group(1) a_notify( 'DCC', 'Direct Chat Request', '{0} wants to chat directly.'.format(nick)) def notify_dcc_chat_closed(match): '''Notify on DCC chat termination.''' if weechat.config_get_plugin("show_dcc") == "on": nick = match.group(1) a_notify( 'DCC', 'Direct Chat Ended', 'Direct chat with {0} has ended.'.format(nick)) def notify_dcc_get_request(match): 'Notify on DCC get request.' if weechat.config_get_plugin("show_dcc") == "on": nick = match.group(1) file_name = match.group(2) a_notify( 'DCC', 'File Transfer Request', '{0} wants to send you {1}.'.format(nick, file_name)) def notify_dcc_get_completed(match): 'Notify on DCC get completion.' if weechat.config_get_plugin("show_dcc") == "on": file_name = match.group(1) a_notify('DCC', 'Download Complete', file_name) def notify_dcc_get_failed(match): 'Notify on DCC get failure.' if weechat.config_get_plugin("show_dcc") == "on": file_name = match.group(1) a_notify('DCC', 'Download Failed', file_name) def notify_dcc_send_completed(match): 'Notify on DCC send completion.' if weechat.config_get_plugin("show_dcc") == "on": file_name = match.group(1) a_notify('DCC', 'Upload Complete', file_name) def notify_dcc_send_failed(match): 'Notify on DCC send failure.' if weechat.config_get_plugin("show_dcc") == "on": file_name = match.group(1) a_notify('DCC', 'Upload Failed', file_name) # ----------------------------------------------------------------------------- # Utility # ----------------------------------------------------------------------------- def set_away_status(match): status = match.group(1) if status == 'been ': STATE['is_away'] = True if status == 'longer ': STATE['is_away'] = False def cb_process_message( data, wbuffer, date, tags, displayed, highlight, prefix, message ): '''Delegates incoming messages to appropriate handlers.''' tags = set(tags.split(',')) functions = globals() is_public_message = tags.issuperset( TAGGED_MESSAGES['public message or action']) buffer_name = weechat.buffer_get_string(wbuffer, 'name') dcc_buffer_regex = re.compile(r'^irc_dcc\.', re.UNICODE) dcc_buffer_match = dcc_buffer_regex.match(buffer_name) highlighted = False if int(highlight): highlighted = True # Private DCC message identifies itself as public. if is_public_message and dcc_buffer_match: notify_private_message_or_action(prefix, message, highlighted) return weechat.WEECHAT_RC_OK # Pass identified, untagged message to its designated function. for key, value in UNTAGGED_MESSAGES.items(): match = value.match(message) if match: functions[DISPATCH_TABLE[key]](match) return weechat.WEECHAT_RC_OK # Pass identified, tagged message to its designated function. for key, value in TAGGED_MESSAGES.items(): if tags.issuperset(value): functions[DISPATCH_TABLE[key]](prefix, message, highlighted) return weechat.WEECHAT_RC_OK return weechat.WEECHAT_RC_OK def a_notify(notification, title, description, priority=notify2.URGENCY_LOW): '''Returns whether notifications should be sticky.''' is_away = STATE['is_away'] icon = STATE['icon'] time_out = 5000 if weechat.config_get_plugin('sticky') == 'on': time_out = 0 if weechat.config_get_plugin('sticky_away') == 'on' and is_away: time_out = 0 try: # notify2.init("wee-notifier") # wn = notify2.Notification(title, description, icon) # wn.set_urgency(priority) # wn.set_timeout(time_out) # wn.show() subprocess.run(["notify-send", "-a", " WeeChat", title, description]) if title != "Server Connected" and title != "Server Disconnected": subprocess.run(["canberra-gtk-play", "-i", "message-new-instant", "-V", "15"]) except Exception as error: weechat.prnt('', 'anotify: {0}'.format(error)) # ----------------------------------------------------------------------------- # Main # ----------------------------------------------------------------------------- def main(): '''Sets up WeeChat notifications.''' # Initialize options. for option, value in SETTINGS.items(): if not weechat.config_is_set_plugin(option): weechat.config_set_plugin(option, value) # Initialize. name = "WeeChat" icon = "/usr/share/pixmaps/weechat.xpm" notifications = [ 'Public', 'Private', 'Action', 'Notice', 'Invite', 'Highlight', 'Server', 'Channel', 'DCC', 'WeeChat' ] STATE['icon'] = icon # Register hooks. weechat.hook_signal( 'irc_server_connected', 'cb_irc_server_connected', '') weechat.hook_signal( 'irc_server_disconnected', 'cb_irc_server_disconnected', '') weechat.hook_signal('upgrade_ended', 'cb_upgrade_ended', '') weechat.hook_print('', '', '', 1, 'cb_process_message', '') if __name__ == '__main__' and IMPORT_OK and weechat.register( SCRIPT_NAME, SCRIPT_AUTHOR, SCRIPT_VERSION, SCRIPT_LICENSE, SCRIPT_DESC, '', '' ): main()