2020-08-24 18:07:13 +02:00
|
|
|
#! /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.
|
2020-08-24 18:15:45 +02:00
|
|
|
#
|
2020-08-24 18:07:13 +02:00
|
|
|
# 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.
|
2020-08-24 18:15:45 +02:00
|
|
|
#
|
2020-08-24 18:07:13 +02:00
|
|
|
# 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.
|
|
|
|
|
2020-08-24 18:34:32 +02:00
|
|
|
import argparse, sys, re
|
2020-08-24 18:07:13 +02:00
|
|
|
import smtplib, time, mimetypes
|
|
|
|
from email.header import Header
|
|
|
|
from email.message import EmailMessage
|
|
|
|
|
2020-08-24 18:34:32 +02:00
|
|
|
|
2020-08-24 19:05:59 +02:00
|
|
|
def parse_args():
|
|
|
|
parser = argparse.ArgumentParser(description="Simple mailing script.")
|
|
|
|
ma = parser.add_argument_group(title="mandatory arguments", description=None)
|
|
|
|
ma.add_argument(
|
|
|
|
"-t",
|
|
|
|
"--tofile",
|
|
|
|
metavar="TO.FILE",
|
|
|
|
type=str,
|
|
|
|
required=True,
|
|
|
|
help="Sort of CSV file containing addresses of recipients.",
|
|
|
|
)
|
|
|
|
ma.add_argument(
|
|
|
|
"-b",
|
|
|
|
"--bodyfile",
|
|
|
|
metavar="BODY.FILE",
|
|
|
|
type=str,
|
|
|
|
required=True,
|
|
|
|
help="Template of the mail to be sent",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-d",
|
|
|
|
"--dry-run",
|
|
|
|
action="store_true",
|
|
|
|
help="Load data but don't send anything.",
|
|
|
|
)
|
2020-08-24 20:02:27 +02:00
|
|
|
parser.add_argument(
|
|
|
|
"-v", "--verbose", action="store_true",
|
|
|
|
)
|
2020-08-24 19:05:59 +02:00
|
|
|
parser.add_argument(
|
|
|
|
"-a",
|
|
|
|
"--attachement",
|
|
|
|
metavar="ATTACHED.FILE",
|
|
|
|
type=str,
|
|
|
|
nargs="?",
|
|
|
|
help="Optionnal attachment.",
|
|
|
|
)
|
|
|
|
return parser.parse_args()
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
|
|
|
|
# read the recipients file where values are separated by | characters
|
2020-08-24 19:05:59 +02:00
|
|
|
def read_recipients(args):
|
|
|
|
recipientfile = open(args.tofile)
|
2020-08-24 18:07:13 +02:00
|
|
|
lines = recipientfile.readlines()
|
2020-08-24 18:15:45 +02:00
|
|
|
return [line[:-1].split("|") for line in lines]
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
|
2020-08-24 19:05:59 +02:00
|
|
|
def read_body(args):
|
|
|
|
bodyfile = open(args.bodyfile)
|
2020-08-24 18:07:13 +02:00
|
|
|
body = ""
|
2020-08-24 18:15:45 +02:00
|
|
|
for line in bodyfile.readlines():
|
2020-08-24 18:07:13 +02:00
|
|
|
body = body + line
|
|
|
|
return body
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
|
2020-08-24 19:05:59 +02:00
|
|
|
def read_join(args):
|
2020-08-24 18:07:13 +02:00
|
|
|
try:
|
2020-08-24 19:05:59 +02:00
|
|
|
with open(args.attachement, "rb") as fp:
|
|
|
|
ctype, encoding = mimetypes.guess_type(args.attachement)
|
2020-08-24 18:07:13 +02:00
|
|
|
if ctype is None or encoding is not None:
|
2020-08-24 18:15:45 +02:00
|
|
|
ctype = "application/octet-stream"
|
|
|
|
maintype, subtype = ctype.split("/", 1)
|
2020-08-24 18:07:13 +02:00
|
|
|
metadata = {
|
2020-08-24 19:05:59 +02:00
|
|
|
"filename": args.attachement,
|
2020-08-24 18:15:45 +02:00
|
|
|
"maintype": maintype,
|
|
|
|
"subtype": subtype,
|
|
|
|
}
|
2020-08-24 18:07:13 +02:00
|
|
|
return (fp.read(), metadata)
|
|
|
|
except Exception as e:
|
|
|
|
print("read error: %s" % e)
|
|
|
|
exit(1)
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
|
|
|
|
def replace_values(bodytemplate, values):
|
2020-08-24 18:07:13 +02:00
|
|
|
body = bodytemplate
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
for i in range(len(values)):
|
|
|
|
i = i + 1
|
2020-08-24 18:07:13 +02:00
|
|
|
pattern = "${{%02d}}" % i
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
body = body.replace(pattern, values[i - 1])
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
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: *(.*)")
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
|
|
|
|
def qencode_subject(message):
|
|
|
|
subjectmatches = re.search(r"^Subject: *(.*)$", message, re.MULTILINE)
|
|
|
|
if subjectmatches == None:
|
|
|
|
return message
|
2020-08-24 18:07:13 +02:00
|
|
|
subjectstr = subjectmatches.group(1)
|
2020-08-24 18:15:45 +02:00
|
|
|
h = Header(subjectstr, "utf-8")
|
|
|
|
qsubjectstr = h.encode()
|
|
|
|
return message.replace(subjectstr, qsubjectstr, 1)
|
|
|
|
|
2020-08-24 18:07:13 +02:00
|
|
|
|
2020-08-24 18:56:58 +02:00
|
|
|
def send_message(message, to, attachment, args):
|
2020-08-24 18:07:13 +02:00
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
lines = message.split("\n")
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# identify headers in the template
|
|
|
|
fromvalue = None
|
|
|
|
subjectvalue = None
|
|
|
|
ccvalue = None
|
|
|
|
bccvalue = None
|
|
|
|
replytovalue = None
|
2020-08-24 18:15:45 +02:00
|
|
|
|
|
|
|
for index, line in enumerate(lines):
|
|
|
|
if not line:
|
|
|
|
body = "\n".join(lines[index + 1 :])
|
2020-08-24 18:07:13 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
frommatches = FROMPATTERN.match(line)
|
|
|
|
subjectmatches = SUBJECTPATTERN.match(line)
|
|
|
|
ccmatches = CCPATTERN.match(line)
|
|
|
|
bccmatches = BCCPATTERN.match(line)
|
|
|
|
replytomatches = REPLYTOPATTERN.match(line)
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
if frommatches:
|
2020-08-24 18:07:13 +02:00
|
|
|
fromvalue = frommatches.group(1)
|
2020-08-24 18:15:45 +02:00
|
|
|
if subjectmatches:
|
2020-08-24 18:07:13 +02:00
|
|
|
subjectvalue = subjectmatches.group(1)
|
2020-08-24 18:15:45 +02:00
|
|
|
if ccmatches:
|
2020-08-24 18:07:13 +02:00
|
|
|
ccvalue = ccmatches.group(1)
|
2020-08-24 18:15:45 +02:00
|
|
|
if bccmatches:
|
2020-08-24 18:07:13 +02:00
|
|
|
bccvalue = bccmatches.group(1)
|
2020-08-24 18:15:45 +02:00
|
|
|
if replytomatches:
|
2020-08-24 18:07:13 +02:00
|
|
|
replytovalue = replytomatches.group(1)
|
|
|
|
|
|
|
|
# lists all addresses to which the mail will be sent
|
|
|
|
dests = [to]
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
if ccvalue:
|
2020-08-24 18:07:13 +02:00
|
|
|
dests.append(ccvalue)
|
2020-08-24 18:15:45 +02:00
|
|
|
if bccvalue:
|
2020-08-24 18:07:13 +02:00
|
|
|
dests.append(bccvalue)
|
|
|
|
|
|
|
|
msg = EmailMessage()
|
2020-08-24 18:15:45 +02:00
|
|
|
msg["Subject"] = subjectvalue
|
|
|
|
msg["From"] = fromvalue
|
|
|
|
msg["To"] = to
|
2020-08-24 18:07:13 +02:00
|
|
|
if replytovalue:
|
2020-08-24 18:15:45 +02:00
|
|
|
msg["Reply-To"] = replytovalue
|
|
|
|
if ccvalue:
|
|
|
|
msg["Cc"] = ccvalue
|
2020-08-24 18:07:13 +02:00
|
|
|
msg.set_content(body)
|
|
|
|
if attachment:
|
|
|
|
msg.add_attachment(attachment[0], **attachment[1])
|
|
|
|
|
2020-08-24 20:02:27 +02:00
|
|
|
if args.verbose:
|
|
|
|
print(msg)
|
|
|
|
print("-" * 80)
|
|
|
|
else:
|
|
|
|
print(
|
|
|
|
"Sending : %s, from %s to %s, dests : %s"
|
|
|
|
% (subjectvalue, fromvalue, dests[0], repr(dests))
|
|
|
|
)
|
2020-08-24 18:07:13 +02:00
|
|
|
|
2020-08-24 18:56:58 +02:00
|
|
|
if not args.dry_run:
|
|
|
|
# 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()
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
|
2020-08-24 18:15:45 +02:00
|
|
|
if __name__ == "__main__":
|
2020-08-24 18:07:13 +02:00
|
|
|
|
2020-08-24 19:05:59 +02:00
|
|
|
args = parse_args()
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# read the recipients file
|
2020-08-24 19:05:59 +02:00
|
|
|
sets = read_recipients(args)
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# read the template of the mail
|
2020-08-24 19:05:59 +02:00
|
|
|
bodytemplate = read_body(args)
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# optionnaly read the attachment of the mail
|
|
|
|
attachment = None
|
2020-08-24 19:05:59 +02:00
|
|
|
if args.attachement:
|
|
|
|
attachment = read_join(args)
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# counter to be able to pause each 10 mails send
|
|
|
|
counter = 0
|
|
|
|
|
|
|
|
# send mail to each recipient
|
2020-08-24 18:15:45 +02:00
|
|
|
for set in sets:
|
|
|
|
|
2020-08-24 18:07:13 +02:00
|
|
|
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
|
2020-08-24 18:56:58 +02:00
|
|
|
send_message(body, recipient, attachment, args)
|
2020-08-24 18:07:13 +02:00
|
|
|
|
|
|
|
# pause every 10 mails for 5 secs so that the MTA doesn't explode
|
|
|
|
counter = counter + 1
|
2020-08-24 18:15:45 +02:00
|
|
|
if counter >= 10:
|
2020-08-24 18:07:13 +02:00
|
|
|
counter = 0
|
|
|
|
print("suspending execution for 5 secs")
|
|
|
|
time.sleep(5)
|