#! /usr/bin/env python3 # (C) Olivier Berger 2001-2002 # (C) François Poulain 2019 # # 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 2 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, write to the Free Software # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA # 02111-1307, USA. # # This program is used to mail a message to several recipients, thus # providing customization for each recipient. import sys, getopt, re import smtplib, time, mimetypes from email.header import Header from email.message import EmailMessage # # displays usage of the program, then quit # def usage(returncode): print("""SYNTAX: """, sys.argv[0], """ -t tofile -b bodyfile [-p attachedfile] tofile : sort of CSV file containing addresses of recipients bodyfile : template of the mail to be sent attachedfile : optionnal attachment """) sys.exit(returncode) TOFILE = None BODYFILE = None JOINFILE = None # read the recipients file where values are separated by | characters def read_recipients() : recipientfile = open(TOFILE) lines = recipientfile.readlines() return [line[:-1].split('|') for line in lines] def read_body() : bodyfile = open(BODYFILE) body = "" for line in bodyfile.readlines() : body = body + line return body def read_join() : try: with open(JOINFILE, 'rb') as fp: ctype, encoding = mimetypes.guess_type(JOINFILE) if ctype is None or encoding is not None: ctype = 'application/octet-stream' maintype, subtype = ctype.split('/', 1) metadata = { 'filename': JOINFILE, 'maintype': maintype, 'subtype': subtype, } return (fp.read(), metadata) except Exception as e: print("read error: %s" % e) exit(1) def replace_values(bodytemplate, values) : body = bodytemplate for i in range(len(values)) : i = i+1 pattern = "${{%02d}}" % i body = body.replace(pattern, values[i-1]) return body # patterns for header of the mail FROMPATTERN = re.compile(r"^From: *(.*)") SUBJECTPATTERN = re.compile(r"^Subject: *(.*)") CCPATTERN = re.compile(r"^Cc: *(.*)") BCCPATTERN = re.compile(r"^Bcc: *(.*)") REPLYTOPATTERN = re.compile(r"^Reply-To: *(.*)") def qencode_subject (message): subjectmatches = re.search (r"^Subject: *(.*)$", message, re.MULTILINE) if (subjectmatches == None): return message subjectstr = subjectmatches.group(1) h = Header(subjectstr, 'utf-8') qsubjectstr=h.encode() return message.replace (subjectstr, qsubjectstr, 1) def send_message(message, to, attachment) : lines = message.split('\n') # identify headers in the template fromvalue = None subjectvalue = None ccvalue = None bccvalue = None replytovalue = None for index, line in enumerate(lines) : if not line : body = '\n'.join(lines[index+1:]) break frommatches = FROMPATTERN.match(line) subjectmatches = SUBJECTPATTERN.match(line) ccmatches = CCPATTERN.match(line) bccmatches = BCCPATTERN.match(line) replytomatches = REPLYTOPATTERN.match(line) if frommatches : fromvalue = frommatches.group(1) if subjectmatches : subjectvalue = subjectmatches.group(1) if ccmatches : ccvalue = ccmatches.group(1) if bccmatches : bccvalue = bccmatches.group(1) if replytomatches : replytovalue = replytomatches.group(1) # lists all addresses to which the mail will be sent dests = [to] if ccvalue : dests.append(ccvalue) if bccvalue : dests.append(bccvalue) msg = EmailMessage() msg['Subject'] = subjectvalue msg['From'] = fromvalue msg['To'] = to if replytovalue: msg['Reply-To'] = replytovalue if ccvalue : msg['Cc'] = ccvalue msg.set_content(body) if attachment: msg.add_attachment(attachment[0], **attachment[1]) print("Sending : %s, from %s to %s, dests : %s" % (subjectvalue, fromvalue, dests[0], repr(dests))) # sending the mail to its recipients using the local mailer via SMTP server = smtplib.SMTP('localhost','25') server.set_debuglevel(1) server.send_message(msg) server.quit() if __name__ == '__main__' : # handle options of the program try: opts, args = getopt.getopt(sys.argv[1:], 't:b:p:') except getopt.error as msg: print("getopt error: %s" % msg) usage(1) for name, value in opts: if name == '-t': TOFILE = value elif name == '-b': BODYFILE = value elif name == '-p': JOINFILE = value else: print("argument: ", name, " invalid") usage(1) if not TOFILE or not BODYFILE: usage(1) # read the recipients file sets = read_recipients() # read the template of the mail bodytemplate = read_body() # optionnaly read the attachment of the mail attachment = None if JOINFILE: attachment = read_join() # counter to be able to pause each 10 mails send counter = 0 # send mail to each recipient for set in sets : values = set # the recipient is always in first column recipient = values[0] # replace substitution variables in the template for each recipient body = replace_values(bodytemplate, values) # send message send_message(body, recipient, attachment) # pause every 10 mails for 5 secs so that the MTA doesn't explode counter = counter + 1 if counter >= 10 : counter = 0 print("suspending execution for 5 secs") time.sleep(5)