123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 |
- #!/usr/bin/python3
- # Copyright (C) 2013 Raymond Wagenmaker <raymondwagenmaker@gmail.com>
- # Copyright (C) 2020 Distopico <distopico@riseup.net> and contributors
- #
- # This program is free software: you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation, either version 3 of the License, or
- # (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
- """Run OfflineIMAP after adding notification sending to its UIs.
- When an account finishes syncing, messages copied to the local repository will
- be reported using D-Bus (through notifypy) or a fallback notifier command.
- """
- import html
- from collections import defaultdict, OrderedDict
- import configparser
- from datetime import datetime
- import email.header
- import email.parser
- import email.utils
- import functools
- import inspect
- import locale
- import operator
- import os
- import shlex
- import string
- import subprocess
- import sys
- import textwrap
- import offlineimap
- try:
- import notifypy
- except ImportError:
- pass
- __copyright__ = """
- Copyright 2013, Raymond Wagenmaker <raymondwagenmaker@gmail.com>
- Copyright 2024, Distopico <distopico@riseup.net>
- """
- __author__ = 'Raymond Wagenmaker and Distopico'
- __maintainer__ = 'Distopico <distopico@riseup.net>'
- __license__ = "GPLv3"
- __version__ = '0.7.1'
- CONFIG_SECTION = 'notifications'
- CONFIG_DEFAULTS = OrderedDict((
- ('summary', 'New mail for {account} in {folder}'),
- ('body', 'From: {h[from]}\nSubject: {h[subject]}'),
- ('icon', 'mail-unread'),
- ('urgency', 'normal'),
- ('timeout', '-1'),
- ('max', '2'),
- ('digest-summary', 'New mail for {account} ({count})'),
- ('digest-body', '{count} in {folder}'),
- ('notifier', 'notify-send -a {appname} -i {icon} -c {category}'
- ' -u {urgency} -t {timeout} {summary} {body}'),
- ('failstr', '')
- ))
- def send_notification(ui, conf, summary, body):
- appname = 'OfflineIMAP'
- category = 'email.arrived'
- icon = conf['icon']
- urgency = conf['urgency']
- timeout = conf['timeout']
- encoding = locale.getpreferredencoding(False)
- if not icon or not os.path.isfile(icon):
- icon = os.path.join(os.getcwd(), 'icon.svg')
- try:
- notification = notifypy.Notify()
- notification.application_name = appname
- notification.title = summary
- notification.message = body
- notification.icon = icon
- notification.urgency = urgency
- notification.timeout = timeout
- notification.send()
- except (NameError, RuntimeError) as e: # no notify-py or no notification service
- try:
- format_args = {'appname': appname, 'category': category,
- 'summary': summary, 'body': body, 'icon': icon,
- 'urgency': urgency, 'timeout': timeout}
- subprocess.call([word.format(**format_args).encode(encoding)
- for word in shlex.split(conf['notifier'])])
- except ValueError as exc:
- ui.error(exc, msg='While parsing fallback notifier command')
- except OSError as exc:
- ui.error(exc, msg='While calling fallback notifier')
- def add_notifications(ui_cls):
- def extension(method):
- old = getattr(ui_cls, method.__name__)
- uibase_spec = inspect.getfullargspec(getattr(offlineimap.ui.UIBase.UIBase,
- method.__name__))
- @functools.wraps(old)
- def new(*args, **kwargs):
- old(*args, **kwargs)
- old_args = inspect.getcallargs(old, *args, **kwargs)
- method(**{arg: old_args[arg] for arg in uibase_spec.args})
- setattr(ui_cls, method.__name__, new)
- @extension
- def __init__(self, *args, **kwargs):
- self.local_repo_names = {}
- self.new_messages = defaultdict(lambda: defaultdict(list))
- @extension
- def acct(self, account):
- self.local_repo_names[account] = account.localrepos.getname()
- @extension
- def acctdone(self, account):
- if self.new_messages[account]:
- notify(self, account)
- self.new_messages[account].clear()
- @extension
- def copyingmessage(self, uid, num, num_to_copy, src, destfolder):
- repository = destfolder.getrepository()
- account = repository.getaccount()
- if (repository.getname() == self.local_repo_names[account] and
- 'S' not in src.getmessageflags(uid)):
- content = { 'uid': uid, 'message': src.getmessage(uid) }
- folder = destfolder.getname()
- self.new_messages[account][folder].append(content)
- return ui_cls
- class MailNotificationFormatter(string.Formatter):
- _FAILED_DATE_CONVERSION = object()
- def __init__(self, escape=False, failstr=''):
- self.escape = escape
- self.failstr = failstr
- def convert_field(self, value, conversion):
- if conversion == 'd':
- datetuple = email.utils.parsedate_tz(value)
- if datetuple is None:
- return MailNotificationFormatter._FAILED_DATE_CONVERSION
- return datetime.fromtimestamp(email.utils.mktime_tz(datetuple))
- elif conversion in ('a', 'n', 'N'):
- name, address = email.utils.parseaddr(value)
- if not address:
- address = value
- if conversion == 'a':
- return address
- return name if name or conversion == 'n' else address
- return super(MailNotificationFormatter, self).convert_field(value,
- conversion)
- def format_field(self, value, format_spec):
- if value is MailNotificationFormatter._FAILED_DATE_CONVERSION:
- result = self.failstr
- else:
- result = super(MailNotificationFormatter, self).format_field(value,
- format_spec)
- return html.escape(result, quote=True) if self.escape else result
- class HeaderDecoder(object):
- def __init__(self, message, failstr=''):
- self.message = message
- self.failstr = failstr
- def __getitem__(self, key):
- header = self.message[key]
- if header is None:
- return self.failstr
- return ' '.join(word.decode(charset, errors='replace')
- if charset is not None else word
- for word, charset in email.header.decode_header(header))
- def get_config(ui):
- conf = CONFIG_DEFAULTS.copy()
- try:
- for item in ui.config.items(CONFIG_SECTION):
- option, value = item
- if option in ('max', 'timeout'):
- try:
- conf[option] = int(value)
- except ValueError:
- ui.warn('value "{}" for "{}" is not a valid integer; '
- 'ignoring'.format(value, option))
- else:
- conf[option] = value
- except configparser.NoSectionError:
- pass
- return conf
- def notify(ui, account):
- account_name = account.getname()
- conf = get_config(ui)
- notify_send = functools.partial(send_notification, ui, conf)
- summary_formatter = MailNotificationFormatter(escape=False, failstr=conf['failstr'])
- body_formatter = MailNotificationFormatter(escape=True, failstr=conf['failstr'])
- count = 0
- body = []
- for folder, contents in ui.new_messages[account].items():
- count += len(contents)
- body.append(body_formatter.format(conf['digest-body'], count=len(contents),
- folder=folder))
- if count > conf['max']:
- summary = summary_formatter.format(conf['digest-summary'], count=count,
- account=account_name)
- return notify_send(summary, '\n'.join(body))
- need_body = '{body' in conf['body'] or '{body' in conf['summary']
- parser = email.parser.Parser()
- encoding = locale.getpreferredencoding(False)
- for folder, contents in ui.new_messages[account].items():
- format_args = {'account': account_name,
- 'folder': folder}
- for content in contents:
- message = parser.parsestr(content.get('message').as_string(),
- headersonly=not need_body)
- format_args['h'] = HeaderDecoder(message, failstr=conf['failstr'])
- if need_body:
- for part in message.walk():
- if part.get_content_type() == 'text/plain':
- charset = part.get_content_charset()
- payload = part.get_payload(decode=True)
- format_args['body'] = payload.decode(charset)
- break
- else:
- format_args['body'] = conf['failstr']
- try:
- notify_send(summary_formatter.vformat(conf['summary'], (), format_args),
- body_formatter.vformat(conf['body'], (), format_args))
- except (AttributeError, KeyError, TypeError, ValueError) as exc:
- ui.error(exc, msg='In notification format specification')
- def print_help():
- try:
- text_width = int(os.environ['COLUMNS'])
- except (KeyError, ValueError):
- text_width = 80
- tw = textwrap.TextWrapper(width=text_width)
- print('Notification wrapper v{} -- {}\n'.format(__version__, __copyright__))
- print(tw.fill(__doc__))
- print('\nDefault configuration:\n')
- default_config = offlineimap.CustomConfig.CustomConfigParser()
- default_config.add_section(CONFIG_SECTION)
- for option, value in CONFIG_DEFAULTS.items():
- default_config.set(CONFIG_SECTION, option, value)
- default_config.write(sys.stdout)
- def main():
- locale.setlocale(locale.LC_ALL, 'C')
- for name, cls in offlineimap.ui.UI_LIST.items():
- offlineimap.ui.UI_LIST[name] = add_notifications(cls)
- try:
- offlineimap.OfflineImap().run()
- except SystemExit:
- if '-h' in sys.argv or '--help' in sys.argv:
- print('\n')
- print_help()
- raise
- if __name__ == '__main__':
- main()
|