X-Git-Url: https://git.distorted.org.uk/~mdw/stgit/blobdiff_plain/e83b314952d7acfd55fa5bb1adb9623acb908566..93aa248c1f7d093c77959a70942e1ae6208750c8:/stgit/commands/mail.py diff --git a/stgit/commands/mail.py b/stgit/commands/mail.py index 8474c78..6a7d5d2 100644 --- a/stgit/commands/mail.py +++ b/stgit/commands/mail.py @@ -36,10 +36,15 @@ or /usr/share/stgit/templates/patchmail.tmpl). The To/Cc/Bcc addresses can either be added to the template file or passed via the corresponding command line options. -A preamble e-mail can be sent using the '--first' option. All the -subsequent e-mails appear as replies to the first e-mail sent (either -the preamble or the first patch). E-mails can be seen as replies to a -different e-mail by using the '--refid' option. +A preamble e-mail can be sent using the '--cover' and/or '--edit' +options. The first allows the user to specify a file to be used as a +template. The latter option will invoke the editor on the specified +file (defaulting to .git/covermail.tmpl or +/usr/share/stgit/templates/covermail.tmpl). + +All the subsequent e-mails appear as replies to the first e-mail sent +(either the preamble or the first patch). E-mails can be seen as +replies to a different e-mail by using the '--refid' option. SMTP authentication is also possible with '--smtp-user' and '--smtp-password' options, also available as configuration settings: @@ -91,8 +96,11 @@ options = [make_option('-a', '--all', help = 'add VERSION to the [PATCH ...] prefix'), make_option('-t', '--template', metavar = 'FILE', help = 'use FILE as the message template'), - make_option('-f', '--first', metavar = 'FILE', - help = 'send FILE as the first message'), + make_option('-c', '--cover', metavar = 'FILE', + help = 'send FILE as the cover message'), + make_option('-e', '--edit', + help = 'edit the cover message before sending', + action = 'store_true'), make_option('-s', '--sleep', type = 'int', metavar = 'SECONDS', help = 'sleep for SECONDS between e-mails sending'), make_option('--refid', @@ -120,10 +128,10 @@ def __parse_addresses(string): """Return a two elements tuple: (from, [to]) """ def __addr_list(string): - m = re.search('[^@\s<,]+@[^>\s,]+', string); - if (m == None): - return [] - return [ m.group() ] + __addr_list(string[m.end():]) + m = re.search('[^@\s<,]+@[^>\s,]+', string); + if (m == None): + return [] + return [ m.group() ] + __addr_list(string[m.end():]) from_addr_list = [] to_addr_list = [] @@ -166,24 +174,24 @@ def __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, def __build_address_headers(options): headers_end = '' if options.to: - headers_end += 'To: ' - for to in options.to: - headers_end += '%s,' % to - headers_end = headers_end[:-1] + '\n' + headers_end += 'To: ' + for to in options.to: + headers_end += '%s, ' % to + headers_end = headers_end[:-2] + '\n' if options.cc: - headers_end += 'Cc: ' - for cc in options.cc: - headers_end += '%s,' % cc - headers_end = headers_end[:-1] + '\n' + headers_end += 'Cc: ' + for cc in options.cc: + headers_end += '%s, ' % cc + headers_end = headers_end[:-2] + '\n' if options.bcc: - headers_end += 'Bcc: ' - for bcc in options.bcc: - headers_end += '%s,' % bcc - headers_end = headers_end[:-1] + '\n' + headers_end += 'Bcc: ' + for bcc in options.bcc: + headers_end += '%s, ' % bcc + headers_end = headers_end[:-2] + '\n' return headers_end -def __build_first(tmpl, total_nr, msg_id, options): - """Build the first message (series description) to be sent via SMTP +def __build_cover(tmpl, total_nr, msg_id, options): + """Build the cover message (series description) to be sent via SMTP """ maintainer = __get_maintainer() if not maintainer: @@ -194,6 +202,8 @@ def __build_first(tmpl, total_nr, msg_id, options): if options.version: version_str = ' %s' % options.version + else: + version_str = '' total_nr_str = str(total_nr) patch_nr_str = '0'.zfill(len(total_nr_str)) @@ -219,6 +229,32 @@ def __build_first(tmpl, total_nr, msg_id, options): raise CmdException, 'Only "%(name)s" variables are ' \ 'supported in the patch template' + if options.edit: + fname = '.stgitmail.txt' + + # create the initial file + f = file(fname, 'w+') + f.write(msg) + f.close() + + # the editor + if config.has_option('stgit', 'editor'): + editor = config.get('stgit', 'editor') + elif 'EDITOR' in os.environ: + editor = os.environ['EDITOR'] + else: + editor = 'vi' + editor += ' %s' % fname + + print 'Invoking the editor: "%s"...' % editor, + sys.stdout.flush() + print 'done (exit code: %d)' % os.system(editor) + + # read the message back + f = file(fname) + msg = f.read() + f.close() + return msg def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): @@ -245,6 +281,8 @@ def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): if options.version: version_str = ' %s' % options.version + else: + version_str = '' total_nr_str = str(total_nr) patch_nr_str = str(patch_nr).zfill(len(total_nr_str)) @@ -303,12 +341,15 @@ def func(parser, options, args): smtppassword = config.get('stgit', 'smtppassword') applied = crt_series.get_applied() + unapplied = crt_series.get_unapplied() if len(args) >= 1: for patch in args: - if not patch in applied: + if patch in unapplied: raise CmdException, 'Patch "%s" not applied' % patch - patches = args + if not patch in applied: + raise CmdException, 'Patch "%s" does not exist' % patch + patches = args elif options.all: patches = applied elif options.range: @@ -331,11 +372,17 @@ def func(parser, options, args): if start in applied: start_idx = applied.index(start) else: - raise CmdException, 'Patch "%s" not applied' % start + if start in unapplied: + raise CmdException, 'Patch "%s" not applied' % start + else: + raise CmdException, 'Patch "%s" does not exist' % start if stop in applied: stop_idx = applied.index(stop) + 1 else: - raise CmdException, 'Patch "%s" not applied' % stop + if stop in unapplied: + raise CmdException, 'Patch "%s" not applied' % stop + else: + raise CmdException, 'Patch "%s" does not exist' % stop if start_idx >= stop_idx: raise CmdException, 'Incorrect patch range order' @@ -366,18 +413,32 @@ def func(parser, options, args): else: sleep = 2 - # send the first message (if any) - if options.first: - tmpl = file(options.first).read() + # send the cover message (if any) + if options.cover or options.edit: + # find the template file + if options.cover: + tfile_list = [options.cover] + else: + tfile_list = [os.path.join(git.base_dir, 'covermail.tmpl'), + os.path.join(sys.prefix, + 'share/stgit/templates/covermail.tmpl')] + + tmpl = None + for tfile in tfile_list: + if os.path.isfile(tfile): + tmpl = file(tfile).read() + break + if not tmpl: + raise CmdException, 'No cover message template file found' msg_id = email.Utils.make_msgid('stgit') - msg = __build_first(tmpl, total_nr, msg_id, options) + msg = __build_cover(tmpl, total_nr, msg_id, options) from_addr, to_addr_list = __parse_addresses(msg) # subsequent e-mails are seen as replies to the first one ref_id = msg_id - print 'Sending file "%s"...' % options.first, + print 'Sending the cover message...', sys.stdout.flush() __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, @@ -389,19 +450,16 @@ def func(parser, options, args): if options.template: tfile_list = [options.template] else: - tfile_list = [] - - tfile_list += [os.path.join(git.base_dir, 'patchmail.tmpl'), - os.path.join(sys.prefix, - 'share/stgit/templates/patchmail.tmpl')] + tfile_list = [os.path.join(git.base_dir, 'patchmail.tmpl'), + os.path.join(sys.prefix, + 'share/stgit/templates/patchmail.tmpl')] tmpl = None for tfile in tfile_list: if os.path.isfile(tfile): tmpl = file(tfile).read() break if not tmpl: - raise CmdException, 'No e-mail template file: %s or %s' \ - % (tfile_list[-1], tfile_list[-2]) + raise CmdException, 'No e-mail template file found' for (p, patch_nr) in zip(patches, range(1, len(patches) + 1)): msg_id = email.Utils.make_msgid('stgit')