X-Git-Url: https://git.distorted.org.uk/~mdw/stgit/blobdiff_plain/f2944a74ffacb504b6fa092e138c3e7af3ed5ffb..bd427e467ee41df85a761d4d6001b5f3797644cd:/stgit/commands/mail.py diff --git a/stgit/commands/mail.py b/stgit/commands/mail.py index 566c669..3cc71f6 100644 --- a/stgit/commands/mail.py +++ b/stgit/commands/mail.py @@ -15,13 +15,12 @@ along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA """ -import sys, os, re, time, smtplib, email.Utils +import sys, os, re, time, datetime, smtplib, email.Utils from optparse import OptionParser, make_option -from time import gmtime, strftime from stgit.commands.common import * from stgit.utils import * -from stgit import stack, git +from stgit import stack, git, version, templates from stgit.config import config @@ -31,16 +30,18 @@ usage = """%prog [options] [ [\s,]+', string); - if (m == None): - return [] - return [ m.group() ] + __addr_list(string[m.end():]) + def __addr_list(addrs): + m = re.search('[^@\s<,]+@[^>\s,]+', addrs); + if (m == None): + return [] + return [ m.group() ] + __addr_list(addrs[m.end():]) from_addr_list = [] to_addr_list = [] - for line in string.split('\n'): + for line in addresses.split('\n'): if re.match('from:\s+', line, re.I): from_addr_list += __addr_list(line) elif re.match('(to|cc|bcc):\s+', line, re.I): @@ -171,25 +175,44 @@ def __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, s.quit() +def __write_mbox(from_addr, msg): + """Write an mbox like file to the standard output + """ + r = re.compile('^From ', re.M) + msg = r.sub('>\g<0>', msg) + + print 'From %s %s' % (from_addr, datetime.datetime.today().ctime()) + print msg + print + 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_extra_headers(): + """Build extra headers like content-type etc. + """ + headers = 'Content-Type: text/plain; charset=utf-8; format=fixed\n' + headers += 'Content-Transfer-Encoding: 8bit\n' + headers += 'User-Agent: StGIT/%s\n' % version.version + + return headers + def __build_cover(tmpl, total_nr, msg_id, options): """Build the cover message (series description) to be sent via SMTP """ @@ -199,6 +222,10 @@ def __build_cover(tmpl, total_nr, msg_id, options): headers_end = __build_address_headers(options) headers_end += 'Message-Id: %s\n' % msg_id + if options.refid: + headers_end += "In-Reply-To: %s\n" % options.refid + headers_end += "References: %s\n" % options.refid + headers_end += __build_extra_headers() if options.version: version_str = ' %s' % options.version @@ -255,7 +282,7 @@ def __build_cover(tmpl, total_nr, msg_id, options): msg = f.read() f.close() - return msg + return msg.strip('\n') def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): """Build the message to be sent via SMTP @@ -278,6 +305,7 @@ def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): if ref_id: headers_end += "In-Reply-To: %s\n" % ref_id headers_end += "References: %s\n" % ref_id + headers_end += __build_extra_headers() if options.version: version_str = ' %s' % options.version @@ -296,10 +324,10 @@ def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): 'shortdescr': short_descr, 'longdescr': long_descr, 'endofheaders': headers_end, - 'diff': git.diff(rev1 = git_id('%s/bottom' % patch), - rev2 = git_id('%s/top' % patch)), - 'diffstat': git.diffstat(rev1 = git_id('%s/bottom'%patch), - rev2 = git_id('%s/top' % patch)), + 'diff': git.diff(rev1 = git_id('%s//bottom' % patch), + rev2 = git_id('%s//top' % patch)), + 'diffstat': git.diffstat(rev1 = git_id('%s//bottom'%patch), + rev2 = git_id('%s//top' % patch)), 'date': email.Utils.formatdate(localtime = True), 'version': version_str, 'patchnr': patch_nr_str, @@ -323,14 +351,12 @@ def __build_message(tmpl, patch, patch_nr, total_nr, msg_id, ref_id, options): raise CmdException, 'Only "%(name)s" variables are ' \ 'supported in the patch template' - return msg + return msg.strip('\n') def func(parser, options, args): """Send the patches by e-mail using the patchmail.tmpl file as a template """ - if not config.has_option('stgit', 'smtpserver'): - raise CmdException, 'smtpserver not defined' smtpserver = config.get('stgit', 'smtpserver') smtpuser = None @@ -411,25 +437,17 @@ def func(parser, options, args): if options.sleep != None: sleep = options.sleep else: - sleep = 2 + sleep = config.getint('stgit', 'smtpdelay') # send the cover message (if any) if options.cover or options.edit: # find the template file if options.cover: - tfile_list = [options.cover] + tmpl = file(options.template).read() 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' + tmpl = templates.get_template('covermail.tmpl') + if not tmpl: + raise CmdException, 'No cover message template file found' msg_id = email.Utils.make_msgid('stgit') msg = __build_cover(tmpl, total_nr, msg_id, options) @@ -438,28 +456,22 @@ def func(parser, options, args): # subsequent e-mails are seen as replies to the first one ref_id = msg_id - print 'Sending the cover message...', - sys.stdout.flush() - - __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, - smtpuser, smtppassword) - - print 'done' + if options.mbox: + __write_mbox(from_addr, msg) + else: + print 'Sending the cover message...', + sys.stdout.flush() + __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, + smtpuser, smtppassword) + print 'done' # send the patches if options.template: - tfile_list = [options.template] + tmpl = file(options.template).read() else: - 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 found' + tmpl = templates.get_template('patchmail.tmpl') + if not tmpl: + 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') @@ -471,10 +483,11 @@ def func(parser, options, args): if not ref_id: ref_id = msg_id - print 'Sending patch "%s"...' % p, - sys.stdout.flush() - - __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, - smtpuser, smtppassword) - - print 'done' + if options.mbox: + __write_mbox(from_addr, msg) + else: + print 'Sending patch "%s"...' % p, + sys.stdout.flush() + __send_message(smtpserver, from_addr, to_addr_list, msg, sleep, + smtpuser, smtppassword) + print 'done'