-#!/usr/bin/python
+#!/usr/bin/python3
# -*- coding: utf-8 -*-
#
-# mailplate — apply templates to mail drafts.
+# mailplate — reformat mail drafts according to templates
#
-# This script reformats mail drafts according to a given template. The
-# template may be specified on the command line, but mailplate can also use
-# control information from the template files to automatically select an
-# appropriate template (--auto). A selection menu feature is planned (--menu).
-#
-# Applying a template means obtainined select data from an existing mail
-# message (unless --new is specified) and to fill it into appropriate slots in
-# the template. Messages are processed in three parts: headers, body, and
-# signature.
-#
-# The template can define two types of headers: mandatory and preservatory.
-# Mandatory headers take precedence over headers in the existing message and
-# thus overwrite them. Preservatory headers instruct mailplate to port their
-# data from the existing mail message. Headers in the existing message but not
-# defined in the template are dropped, unless --keep-unknown is given.
-#
-# Body and signature are separated by '-- '. If this sentinel is not found,
-# no signature is extracted.
-#
-# Templates can be interpolated and data filled into slots. Helper slots are
-# filled with the output of helper commands (which must be defined in the
-# configuration), environment variable slots are just that, and mail variable
-# slots can be filled with data obtained by running regexps or commands over
-# the message.
-#
-# This script can be run in multiple ways:
-#
-# As a filter, it applies a template to data from stdin and writes the result
-# to stdout.
-#
-# Given a file, it modifies the file, unless it cannot write to the file, in
-# which case it writes to stdout.
-#
-# When --editor is passed, the script spawns sensible-editor on the result. It
-# may thus be used as the editor for your mail user agent.
+# Please see the mailplate(1) manpage or the homepage for more information:
+# http://madduck.net/code/mailplate/
#
# TODO: if headers like From are absent from the mail, they should not be kept
# but replaced with a default.
#
-# Copyright © martin f. krafft <madduck@debian.org>
+# Copyright © martin f. krafft <madduck@madduck.net>
# Released under the terms of the Artistic Licence 2.0
#
+__name__ = 'mailplate'
+__description__ = 'reformat mail drafts according to templates'
+__version__ = '0.3'
+__author__ = 'martin f. krafft <madduck@madduck.net>'
+__copyright__ = 'Copyright © ' + __author__
+__licence__ = 'Artistic Licence 2.0'
+
import email
import os
import posix
import re
import sys
import subprocess
-import ConfigParser
+import configparser
from optparse import OptionParser
###
STD_HEADERS = ('From', 'To', 'Cc', 'Bcc', 'Subject', 'Reply-To', 'In-Reply-To')
KEEP_HEADERS = { 'KEEP_FROM_HEADER' : STD_HEADERS[:1]
, 'KEEP_STD_HEADERS' : STD_HEADERS[1:]
+ , 'KEEP_ALL_HEADERS' : STD_HEADERS
}
SIG_DELIM='\n-- \n'
###
def err(s):
- sys.stderr.write(s + '\n')
+ sys.stderr.write('E: ' + s + '\n')
+
+def warn(s):
+ sys.stderr.write('W: ' + s + '\n')
+
+def info(s):
+ if not options.verbose: return
+ sys.stderr.write('I: ' + s + '\n')
# obtain a regexp from a line, run it, return score/dict if matched
def exec_regexp(line, rawmsg, name):
else:
return 0
except OSError:
- err("W: command '%s' (template '%s') failed to run." % (r, name))
+ warn("command '%s' (template '%s') failed to run." % (r, name))
return 0
def interpolate_helpers(s):
proc = subprocess.Popen(helpers[helper], shell=True,
stdout=subprocess.PIPE, stderr=sys.stderr)
out = proc.communicate()[0]
- s = s[:helper_begin] + out.strip() + s[helper_end+1:]
+ s = s[:helper_begin] + out.strip().decode() + s[helper_end+1:]
except KeyError:
- err('E: unknown helper: ' + helper)
+ err('unknown helper: ' + helper)
sys.exit(posix.EX_DATAERR)
return s
headers = {}
payload = None
+###
+### COMMAND LINE PARSING
+###
+
+parser = OptionParser()
+parser.prog = __name__
+parser.version = __version__
+parser.description = __description__
+parser.usage = '%prog [options] <message>'
+parser.add_option('-a', '--auto', dest='auto',
+ default=False, action='store_true',
+ help='turn on template auto-discovery')
+parser.add_option('-m', '--menu', dest='menu',
+ default=False, action='store_true',
+ help='choose from a list of templates (not yet implemented)')
+parser.add_option('-n', '--new', dest='new',
+ default=False, action='store_true',
+ help='create a new message')
+parser.add_option('-e', '--editor', dest='edit',
+ default=False, action='store_true',
+ help='spawn editor once template is applied')
+parser.add_option('-k', '--keep-unknown', dest='keep_unknown',
+ default=False, action='store_true',
+ help='preserve mail headers not specified in template')
+parser.add_option('-v', '--verbose', dest='verbose',
+ default=False, action='store_true',
+ help='write informational messages to stderr')
+parser.add_option('-d', '--debug', dest='debug',
+ default=False, action='store_true',
+ help='start a debugger after initialisation')
+parser.add_option('-V', '--version', dest='version',
+ default=False, action='store_true',
+ help='display version information')
+
+options, args = parser.parse_args()
+
+if options.version:
+ print(__name__, __version__ + ' — ' + __description__)
+ print('')
+ print('Written by ' + __author__)
+ print(__copyright__)
+ print('Released under the ' + __licence__)
+ sys.exit(posix.EX_OK)
+
###
### CONFIGURATION FILE PARSING
###
MAILPLATEDIR = os.path.expanduser(MAILPLATEDIR)
# defaults
-config = { 'default_template' : None
+config = { 'default_template' : 'default'
, 'template_path' : TEMPLATEDIR
}
helpers = { 'get_quote' : 'fortune -s' }
if not os.path.exists(CONFFILE):
# conffile does not exist, let's create it with defaults.
+ options.verbose = True
if not os.path.isdir(MAILPLATEDIR):
- os.mkdir(MAILPLATEDIR, 0700)
+ info('configuration directory not found, creating: ' + MAILPLATEDIR)
+ os.mkdir(MAILPLATEDIR, o0700)
if not os.path.isfile(CONFFILE):
+ info('creating a default configuration file: ' + CONFFILE)
f = file(CONFFILE, 'w')
f.write('# mailplate configuration\n[%s]\n' % SECTION_GENERAL)
for kvpair in config.iteritems():
- if kvpair[1] and len(kvpair[1]) > 0:
+ if len(kvpair[1]) > 0:
f.write('%s = %s\n' % kvpair)
if len(helpers) > 0:
f.close()
if not os.access(CONFFILE, os.R_OK):
- err('E: cannot read configuration file: %s' % CONFFILE)
+ err('cannot read configuration file: %s' % CONFFILE)
sys.exit(posix.EX_OSFILE)
# now parse
-parser = ConfigParser.SafeConfigParser()
+parser = configparser.ConfigParser()
parser.read(CONFFILE)
# first the GENERAL section into the config dict for all keys with defaults
for key in config.keys():
try:
config[key] = parser.get(SECTION_GENERAL, key)
- except ConfigParser.NoSectionError, ConfigParser.MissingSectionHeaderError:
- err("E: no section '%s' in %s" % (SECTION_GENERAL, CONFFILE))
+ except (configparser.NoSectionError,
+ configparser.MissingSectionHeaderError):
+ err("no section '%s' in %s" % (SECTION_GENERAL, CONFFILE))
sys.exit(posix.EX_CONFIG)
- except ConfigParser.NoOptionError:
+ except configparser.NoOptionError:
continue
- except ConfigParser.DuplicateSectionError, ConfigParser.ParseError:
- err('E: parse error on %s' % CONFFILE)
+ except (configparser.DuplicateSectionError,
+ configparser.ParseError):
+ err('parse error on %s' % CONFFILE)
sys.exit(posix.EX_CONFIG)
# all HELPERS into the helpers dict
TPATH = os.path.expanduser(config['template_path'])
if not os.path.isdir(TPATH):
- os.mkdir(TPATH, 0700)
-
-###
-### COMMAND LINE PARSING
-###
-
-parser = OptionParser()
-parser.usage = '%prog [options] <message>'
-parser.add_option('-a', '--auto', dest='auto',
- default=False, action='store_true',
- help='turn on template auto-discovery')
-parser.add_option('-m', '--menu', dest='menu',
- default=False, action='store_true',
- help='choose from a list of template')
-parser.add_option('-n', '--new', dest='new',
- default=False, action='store_true',
- help='create a new message')
-parser.add_option('-e', '--editor', dest='edit',
- default=False, action='store_true',
- help='spawn editor once template is applied')
-parser.add_option('-k', '--keep-unknown', dest='keep_unknown',
- default=False, action='store_true',
- help='preserve mail headers not specified in template')
-parser.add_option('-d', '--debug', dest='debug',
- default=False, action='store_true',
- help='start a debugger after initialisation')
-
-options, args = parser.parse_args()
+ info('creating template directory: ' + TPATH)
+ os.mkdir(TPATH, o0700)
+
+default_templname = config['default_template']
+if default_templname is not None:
+ default_templpath = os.path.join(TPATH, default_templname)
+ if not os.path.isfile(default_templpath):
+ info('creating the default template: ' + default_templpath)
+ f = open(default_templpath, 'w')
+ f.write('@KEEP_STD_HEADERS\n\n@KEEP_BODY\n')
+ f.close()
if options.debug:
import pdb
# parse the arguments
for arg in args:
if arg == '-':
- # filename is -, so do nothing, since stdin/stdout are default
- continue
+ infname = arg
+ outfname = arg
+ elif arg.find(os.path.sep) == -1 and os.access(os.path.join(TPATH, arg), os.R_OK):
+ if templname is not None:
+ err("template already specified (%s), unsure what to do with '%s'"
+ % (templname, arg))
+ sys.exit(posix.EX_USAGE)
+ # argument references an existing template
+ templname = arg
elif os.path.isfile(arg):
+ if infname is not None:
+ err("input file already specified (%s), unsure what to do with '%s'"
+ % (infname, arg))
+ sys.exit(posix.EX_USAGE)
# the file exists, so use it as in/out if read/writeable
if os.access(arg, os.R_OK):
infname = arg
if os.access(arg, os.W_OK):
outfname = arg
- elif os.access(os.path.join(TPATH, arg), os.R_OK):
- # argument referenced an existing template
- templname = arg
else:
- err('E: unknown argument: %s' % arg)
+ err('unknown argument: %s' % arg)
sys.exit(posix.EX_USAGE)
# sanity checks
if options.auto and options.menu:
- err('E: cannot combine --auto and --menu')
+ err('cannot combine --auto and --menu')
sys.exit(posix.EX_USAGE)
elif (options.auto or options.menu) and templname:
- err('E: cannot specify a template with --auto or --menu')
+ err('cannot specify a template with --auto or --menu')
sys.exit(posix.EX_USAGE)
elif not templname and not (options.auto or options.menu):
- err('E: no template specified')
- sys.exit(posix.EX_USAGE)
+ if default_templname is not None:
+ templname = default_templname
+ else:
+ err('no template specified')
+ sys.exit(posix.EX_USAGE)
elif options.menu:
- err('E: --menu mode not yet implemented')
+ err('--menu mode not yet implemented')
sys.exit(posix.EX_USAGE)
###
# read in the message from a file, if a filename is given.
if infname is not None:
- inf = file(infname, 'r', 1)
+ inf = open(infname, 'r', 1)
# read message into buffer, or preinitialise the buffer if --new is given
if options.new:
rawmsg = ''.join(inf.readlines())
if options.auto:
- best_score = (0, config['default_template'], {})
+ best_score = (0, default_templname, {})
for tf in os.listdir(TPATH):
tp = os.path.join(TPATH, tf)
if not os.path.isfile(tp) or not os.access(tp, os.R_OK): continue
templname = best_score[1]
if templname is None:
- err('E: could not determine a template to use and no default is set')
+ err('could not determine a template to use and no default is set')
sys.exit(posix.EX_CONFIG)
- print >>sys.stderr, \
- 'I: Chose profile %s with score %d.' % (templname, best_score[0])
+ info('chose profile %s with score %d.' % (templname, best_score[0]))
vars = best_score[2]
# now read in the template
templpath = os.path.join(TPATH, templname)
if not os.path.isfile(templpath):
- err('E: not a template: ' + templpath)
+ err('not a template: ' + templpath)
sys.exit(posix.EX_OSFILE)
elif not os.access(templpath, os.R_OK):
- err('E: template ' + templpath + ' could not be read.')
+ err('template ' + templpath + ' could not be read.')
sys.exit(posix.EX_OSFILE)
-templ = file(templpath, 'r', 1)
+templ = open(templpath, 'r', 1)
for line in templ:
if not options.auto and line[0] == REGEXPCHAR:
l = line[:-1]
if len(l) == 0:
payload = '' # end of headers
- elif l[0] == KEEP_SLOT_LEADER and KEEP_HEADERS.has_key(l[1:]):
- # found predefined header slot keyword
- for header in KEEP_HEADERS[l[1:]]:
- headers[header.lower()] = (header, _keep_header)
+ elif l[0] == KEEP_SLOT_LEADER:
+ if l[1:] in KEEP_HEADERS:
+ # found predefined header slot keyword
+ for header in KEEP_HEADERS[l[1:]]:
+ headers[header.lower()] = (header, _keep_header)
+ else:
+ err('unknown header slot ' + l + ' found')
+ sys.exit(posix.EX_CONFIG)
else:
header, content = l.split(':', 1)
content = content.strip()
for header, content in msg.items():
# iterate all existing mail headers
lheader = header.lower()
- if headers.has_key(lheader):
+ if lheader in headers:
# the template defines this header
if headers[lheader][1] == _keep_header:
# it's marked as keep, thus use content from email message
# open the output file
if outfname is not None:
- outf = file(outfname, 'w', 0)
+ outf = open(outfname, 'w')
# print the headers, starting with the standard headers in order
for header in STD_HEADERS:
if headers.get(lheader, (None, _keep_header))[1] is not _keep_header:
# the template header contains mandatory data, let's print it.
hpair = headers[lheader]
- print >>outf, ': '.join(hpair)
+ print(': '.join(hpair), file=outf)
# and remove it from the dict
del headers[lheader]
-for i, (header, content) in headers.iteritems():
+for i, (header, content) in headers.items():
# print all remaining headers
if content == _keep_header: continue
- print >>outf, ': '.join((header, content))
+ print(': '.join((header, content)), file=outf)
# print empty line to indicate end of headers.
-print >>outf
+print('', file=outf)
# split payload of existing message into body and signature
body = msg.get_payload().rsplit(SIG_DELIM, 1)
if keep_sig:
payload = payload.replace('@KEEP_SIGNATURE', signature, 1)
-print >>outf, payload.rstrip()
+print(payload.rstrip(), file=outf)
outf.close()
if options.edit:
# finally, spawn the editor, if we wrote into a file
if outfname is None:
- err('E: cannot use --edit without an output file.')
+ err('cannot use --edit without an output file.')
sys.exit(posix.EX_USAGE)
os.execlp('sensible-editor', 'sensible-editor', outfname)