ChatGPT解决这个技术问题 Extra ChatGPT

如何使用 python smtplib 向多个收件人发送电子邮件?

经过大量搜索,我找不到如何使用 smtplib.sendmail 发送给多个收件人。问题是每次发送邮件时,邮件标头似乎都包含多个地址,但实际上只有第一个收件人会收到电子邮件。

问题似乎是 email.Message 模块需要与 smtplib.sendmail() 函数不同的东西。

简而言之,要发送给多个收件人,您应该将标头设置为以逗号分隔的电子邮件地址字符串。但是,sendmail() 参数 to_addrs 应该是电子邮件地址列表。

from email.MIMEMultipart import MIMEMultipart
from email.MIMEText import MIMEText
import smtplib

msg = MIMEMultipart()
msg["Subject"] = "Example"
msg["From"] = "me@example.com"
msg["To"] = "malcom@example.com,reynolds@example.com,firefly@example.com"
msg["Cc"] = "serenity@example.com,inara@example.com"
body = MIMEText("example email body")
msg.attach(body)
smtp = smtplib.SMTP("mailhost.example.com", 25)
smtp.sendmail(msg["From"], msg["To"].split(",") + msg["Cc"].split(","), msg.as_string())
smtp.quit()
看来 OP 回答了他自己的问题:sendmail 需要一个列表。
使用 Python3 我必须遍历收件人; for addr in recipients: msg['To'] = addr 然后它起作用了。多个分配实际上为每个分配一个新的“To”标题。这是一个非常奇怪的界面,我什至无法解释我是如何尝试它的。我什至考虑使用 subprocess 来调用 unix sendmail 包以挽救我的理智,然后才弄清楚这一点。

s
sorin

这确实有效,我花了很多时间尝试多种变体。

import smtplib
from email.mime.text import MIMEText

s = smtplib.SMTP('smtp.uk.xensource.com')
s.set_debuglevel(1)
msg = MIMEText("""body""")
sender = 'me@example.com'
recipients = ['john.doe@example.com', 'john.smith@example.co.uk']
msg['Subject'] = "subject line"
msg['From'] = sender
msg['To'] = ", ".join(recipients)
s.sendmail(sender, recipients, msg.as_string())

该文档确实有示例:tolist =["one@one.org","two@two.org","three@three.org","four@four.org"]
谢谢@sorin 这个脚本。我在从 python 脚本发送电子邮件时遇到问题,使用这段代码,我现在可以发送电子邮件了。
如果您使用的是 Python 3,这将不会发送给多个收件人,您需要 send_message 而不是 sendmail 根据 Antoine 下面的评论和 Python 文档 docs.python.org/3/library/email.examples.html
您必须为 sendmail 的收件人每次遍历使用,否则只有第一个元素会收到邮件。
A
Adrian Petrescu

msg['To'] 需要是一个字符串:

msg['To'] = "a@b.com, b@b.com, c@b.com"

虽然 sendmail(sender, recipients, message) 中的 recipients 需要是一个列表:

sendmail("a@a.com", ["a@b.com", "b@b.com", "c@b.com"], "Howdy")

对于 smtplib.,这是一个奇怪的设计决定
recipients 不必是列表 - 如果给出字符串,则将其视为具有一个元素的列表。 msg['To'] 字符串可以简单地省略。
我真的不明白,'a@a.com, b@b.com' 是如何解析的,所以只有第一个地址才能收到电子邮件。但是,谢谢!这就是答案,必须把列表放在那里。
为我工作,它与 docs.python.org/2/library/email-examples.html 中的文档一致
字符串 "Howdy" 不是有效的 RFC822 消息。一些邮件服务器会简单地拒绝它,其他邮件服务器可能会猜测缺少的部分是什么,并且可能会发现令人惊讶的东西放在那里。最终,如果某些东西最终被交付到某个地方,它可能不会有用。
H
Has QUIT--Anony-Mousse

您需要了解电子邮件的可见地址和交付之间的区别。

msg["To"] 本质上就是印在信上的内容。它实际上没有任何作用。除了您的电子邮件客户端,就像普通邮递员一样,会假设这是您要将电子邮件发送给的人。

然而,实际交付的工作方式可能完全不同。因此,您可以将电子邮件(或副本)放入完全不同的人的邮箱中。

这有多种原因。例如转发To: 标头字段在转发时不会更改,但电子邮件会被放入不同的邮箱。

smtp.sendmail 命令现在负责实际 交付。 email.Message 只是信件的内容,而不是交付的内容。

在低级 SMTP 中,您需要一一提供收件人,这就是为什么地址列表(不包括名称!)是明智的 API。

对于标题,它还可以包含例如名称,例如 To: First Last <email@addr.tld>, Other User <other@mail.tld>因此不推荐您的代码示例,因为它将无法传递此邮件,因为仅通过在 , 上拆分它仍然没有有效地址!


RFC 2822 规定给定标题的最大宽度为 988 个字符,建议的宽度为 78 个字符。如果地址太多,您需要确保“折叠”标题。
这应该是公认的答案,因为它实际上解释了原因和方式。
很好的答案。 CC 和 BCC 电子邮件字段呢?我假设我们还必须在 smtp.send 中包含 CC 和 BCC 电子邮件。并且在 msg 字段中只有 CC 列表(而不是 BCC 列表)?
是的,它就是这样工作的。邮件服务器可能会删除密件抄送字段(以防止此字段可见,我认为它们都不会这样做),但它们不会解析它。
c
coolguy

这个对我有用。

import smtplib
from email.mime.text import MIMEText

s = smtplib.SMTP('smtp.uk.xensource.com')
s.set_debuglevel(1)
msg = MIMEText("""body""")
sender = 'me@example.com'
recipients = 'john.doe@example.com,john.smith@example.co.uk'
msg['Subject'] = "subject line"
msg['From'] = sender
msg['To'] = recipients
s.sendmail(sender, recipients.split(','), msg.as_string())

你用的是什么版本的python?我遇到了与原始海报相同的问题,我使用的是 python 2.7.9
为什么不简单地 recipients = ['john.doe@example.com','john.smith@example.co.uk'] 而不是将其设为字符串,然后将其拆分为列表?
给 Woj,因为 msg['To'] 应该是一个字符串,而 s.sendmail 应该有一个列表:(sender,>>>LIST HERE<<<,msg.as_string())。这意味着,尽管看起来很烦人,但您不能对两个字段使用同一种类型 [字符串或列表]
对我来说就像一个魅力。蟒蛇 3.7.3。
3
3pitt

下面的解决方案对我有用。它成功地向多个收件人发送了一封电子邮件,包括“CC”和“BCC”。

toaddr = ['mailid_1','mailid_2']
cc = ['mailid_3','mailid_4']
bcc = ['mailid_5','mailid_6']
subject = 'Email from Python Code'
fromaddr = 'sender_mailid'
message = "\n  !! Hello... !!"

msg['From'] = fromaddr
msg['To'] = ', '.join(toaddr)
msg['Cc'] = ', '.join(cc)
msg['Bcc'] = ', '.join(bcc)
msg['Subject'] = subject

s.sendmail(fromaddr, (toaddr+cc+bcc) , message)

这部分有效,要真正隐藏密件抄送,您必须省略密件抄送行 bcc = ['mailid_5','mailid_6'] 否则这将显示在标题中,破坏密件抄送的目的。用 gmail 和其他邮件服务器测试。
@Wil 在这种情况下您将如何实施密件抄送?
@3pitt 有点晚了,但您只需使用 s.sendmail(fromaddr, bcc, message) 向他们发送相同的电子邮件。
一些 MTA 会在发送前去除 Bcc: 标头,而其他则不会。 SMTP 并不真正关心标头中的内容。它会实际上尝试发送到您在信封中提供的列表(这里是 smtplibsendmail 方法)并完全忽略标题中的内容。
您的 message 不是有效的 RFC822 消息,因此这应该会非常失败。
S
Syscall

所以实际上问题在于 SMTP.sendmail 和 email.MIMEText 需要两个不同的东西。

email.MIMEText 设置电子邮件正文的“收件人:”标题。它仅用于向另一端的人显示结果,并且像所有电子邮件标题一样,必须是单个字符串。 (请注意,它实际上与实际接收消息的人没有任何关系。)

另一方面,SMTP.sendmail 为 SMTP 协议设置消息的“信封”。它需要一个 Python 字符串列表,每个字符串都有一个地址。

所以,你需要做的是结合你收到的两个回复。将 msg['To'] 设置为单个字符串,但将原始列表传递给 sendmail:

emails = ['a.com','b.com', 'c.com']
msg['To'] = ', '.join( emails ) 
....
s.sendmail( msg['From'], emails, msg.as_string())

M
MasterMind

我尝试了以下方法,它就像一个魅力:)

rec_list =  ['first@example.com', 'second@example.com']
rec =  ', '.join(rec_list)

msg['To'] = rec

send_out = smtplib.SMTP('localhost')
send_out.sendmail(me, rec_list, msg.as_string())

FYR 整个简单代码如下:import smtplib from email.mime.multipart import MIMEMultipart from email.mime.text import MIMEText sender = 'myEmailAddress@example.com' rec_list = ['first@example.com', 'second@example.com'] rec = ', '.join(rec_list) msg = MIMEMultipart('alternative') msg['Subject'] = 'The required subject' msg['From'] = sender msg['To'] = rec html = ('whatever html code') htm_part = MIMEText(html, 'html') msg.attach(htm_part) send_out = smtplib.SMTP('localhost') send_out.sendmail(sender, rec_list, msg.as_string()) send_out.quit()
G
Guilherme Henrique Mendes
import smtplib
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText

def sender(recipients): 

    body = 'Your email content here'
    msg = MIMEMultipart()

    msg['Subject'] = 'Email Subject'
    msg['From'] = 'your.email@gmail.com'
    msg['To'] = (', ').join(recipients.split(','))

    msg.attach(MIMEText(body,'plain'))

    server = smtplib.SMTP('smtp.gmail.com', 587)
    server.starttls()
    server.login('your.email@gmail.com', 'yourpassword')
    server.send_message(msg)
    server.quit()

if __name__ == '__main__':
    sender('email_1@domain.com,email_2@domain.com')

它仅适用于我的 send_message 函数并在收件人列表中使用 join 函数,python 3.6。


r
radtek

我想出了这个可导入的模块功能。在此示例中,它使用 gmail 电子邮件服务器。它分为标题和消息,因此您可以清楚地看到发生了什么:

import smtplib

def send_alert(subject=""):

    to = ['email@one.com', 'email2@another_email.com', 'a3rd@email.com']
    gmail_user = 'me@gmail.com'
    gmail_pwd = 'my_pass'
    smtpserver = smtplib.SMTP("smtp.gmail.com", 587)
    smtpserver.ehlo()
    smtpserver.starttls()
    smtpserver.ehlo
    smtpserver.login(gmail_user, gmail_pwd)
    header = 'To:' + ", ".join(to) + '\n' + 'From: ' + gmail_user + '\n' + 'Subject: ' + subject + '\n'
    msg = header + '\n' + subject + '\n\n'
    smtpserver.sendmail(gmail_user, to, msg)
    smtpserver.close()

只要输入字符串是简单的短 ASCII 文本片段,像这样通过将它们粘贴在一起来组装电子邮件消息实际上是可行的;但除非您确切知道自己在做什么,否则最好使用 email 库,它知道极端情况是什么以及如何处理不完全是纯文本的内容类型。
R
Robie

我使用 python 3.6,以下代码适用于我

email_send = 'xxxxx@xxx.xxx,xxxx@xxx.xxx'
server.sendmail(email_user,email_send.split(','),text)    

m
miken32

几个月前我发现了这一点,blogged about it。总结是:

如果您想使用 smtplib 向多个收件人发送电子邮件,请使用 email.Message.add_header('To', eachRecipientAsString) 添加他们,然后在调用 sendmail 方法时,use email.Message.get_all('To') 将邮件发送给所有收件人。抄送和密送收件人同上。


Python 3.7 抛出异常并显示消息:发生异常:ValueError 消息中可能最多有 1 个 To 标头
C
Community

好吧,this asnwer 方法中的方法对我不起作用。我不知道,也许这是 Python3(我使用的是 3.4 版本)或 gmail 相关问题,但经过一些尝试,对我有用的解决方案是行

s.send_message(msg)

代替

s.sendmail(sender, recipients, msg.as_string())

没有详细信息,这是相当可疑的,但您似乎设法在现代 EmailMessage API 正式发布之前使用它。 (它已经在 3.3 中引入,但在 3.6 中成为官方并记录了一个。)
S
Skiller Dz

您可以在将收件人电子邮件写入文本文件时尝试此操作

from email.mime.text import MIMEText
from email.header import Header
import smtplib

f =  open('emails.txt', 'r').readlines()
for n in f:
     emails = n.rstrip()
server = smtplib.SMTP('smtp.uk.xensource.com')
server.ehlo()
server.starttls()
body = "Test Email"
subject = "Test"
from = "me@example.com"
to = emails
msg = MIMEText(body,'plain','utf-8')
msg['Subject'] = Header(subject, 'utf-8')
msg['From'] =  Header(from, 'utf-8')
msg['To'] = Header(to, 'utf-8')
text = msg.as_string()
try:
   server.send(from, emails, text)
   print('Message Sent Succesfully')
except:
   print('There Was An Error While Sending The Message')

这里的许多装饰对旧版 email.message.message / MIMEText API 很有用,但对于现代 Python 3.6+ email.message.EmailMessage API 不再需要。
G
Gwi7d31

这里有很多在技术上或部分正确的答案。在阅读了每个人的答案后,我想出了一个更可靠/通用的电子邮件功能。我已经确认它可以工作,您可以为正文传递 HTML 或纯文本。请注意,此代码不包括附件代码:

import smtplib
import socket

# Import the email modules we'll need
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart

#
# @param [String] email_list
# @param [String] subject_line
# @param [String] error_message
def sendEmailAlert(email_list="default@email.com", subject_line="Default Subject", error_message="Default Error Message"):
    hostname = socket.gethostname()
    # Create message
    msg = MIMEMultipart()
    msg['Subject'] = subject_line
    msg['From'] = f'no-reply@{hostname}'
    msg['To'] = email_list
    msg.attach(MIMEText(error_message, 'html'))
    # Send the message via SMTP server
    s = smtplib.SMTP('localhost') # Change for remote mail server!
    # Verbose debugging
    s.set_debuglevel(2)
    try:
        s.sendmail(msg['From'], msg['To'].split(","), msg.as_string())
    except Exception as e:
        print(f'EMAIL ISSUE: {e}')
    s.quit()

这显然可以修改为使用本机 Python 日志记录。我只是提供一个坚实的核心功能。我也不能强调这一点,sendmail() 想要一个 List 而不是 String!函数适用于 Python3.6+


顺便说一句,您的代码似乎是为 Python 3.5 或更早版本编写的。 email 库在 3.6 中进行了大修,现在更加通用和合乎逻辑。可能扔掉你所拥有的并从examples from the email documentation.重新开始
K
Kevin G

尝试将所有收件人和 cc_recipients 的列表变量声明为字符串,而不是循环遍历它们,如下所示:

from email.MIMEMultipart import MIMEMultipart
from email.MIMEText import MIMEText
import smtplib

recipients = ["malcom@example.com","reynolds@example.com", "firefly@example.com"]
cc_recipients=["serenity@example.com", "inara@example.com"]
msg = MIMEMultipart()
msg["Subject"] = "Example"
msg["From"] = "me@example.com"
msg["To"] = ', '.join(recipients)
msg["Cc"] = ', '.join(cc_recipients)
body = MIMEText("example email body")
msg.attach(body)
smtp = smtplib.SMTP("mailhost.example.com", 25)
for recipient in recipients:
    smtp.sendmail(msg["From"], recipient, msg.as_string())
for cc_recipient in cc_recipients:
    smtp.sendmail(msg["From"], cc_recipient, msg.as_string())
smtp.quit()

这里的几个示例不必要地创建了一个只有一个主体部分的多部分。显然,当没有多个身体部位时,不需要多部分容器。然而,这也受到在相当新的代码中使用遗留的 pre-3.6 API 的影响;你可能应该扔掉你所拥有的并重新开始使用新的 API。
t
tripleee

这是一个老问题。我发布新答案的主要原因是解释如何使用 Python 3.6+ 中的现代 email 库解决问题,以及它与旧版本有何不同;但首先,让我们回顾一下 Anony-Mousse 在 their answer from 2012. 中所写的内容

SMTP 根本不关心标头中的内容。您传递给 sendmail 方法的收件人列表实际上决定了邮件将被传递到哪里。

在 SMTP 术语中,这称为消息的 信封。 在协议级别,您连接到服务器,然后告诉它消息来自谁(MAIL FROM: SMTP 动词)以及将其发送给谁(RCPT TO:),然后将消息本身 (DATA) 与 headers 和 body 作为一个倾斜的字符串 blob 单独传输。

现代的 smtplib 通过提供一个 send_message 方法来简化这方面的 Python 方面,该方法实际上发送到消息标头中指定的收件人。

现代的 email 库提供了一个 EmailMessage 对象,它替换了您过去必须使用的所有各种单独的 MIME 类型,以从较小的部分组装消息。您可以在不单独构建附件的情况下添加附件,并在需要时构建各种更复杂的多部分结构,但通常不必这样做。只需创建一条消息并填充您想要的部分。

请注意,以下内容被大量评论;总体而言,新的 EmailMessage API 比旧 API 更简洁更通用。

from email.message import EmailMessage

msg = EmailMessage()

# This example uses explicit strings to emphasize that
# that's what these header eventually get turned into
msg["From"] = "me@example.org"
msg["To"] = "main.recipient@example.net, other.main.recipient@example.org"
msg["Cc"] = "secondary@example.com, tertiary@example.eu"
msg["Bcc"] = "invisible@example.int, undisclosed@example.org.au"
msg["Subject"] = "Hello from the other side"

msg.set_content("This is the main text/plain message.")
# You can put an HTML body instead by adding a subtype string argument "html"
# msg.set_content("<p>This is the main text/html message.</p>", "html")

# You can add attachments of various types as you see fit;
# if there are no other parts, the message will be a simple
# text/plain or text/html, but Python will change it into a
# suitable multipart/related or etc if you add more parts
with open("image.png", "rb") as picture:
    msg.add_attachment(picture.read(), maintype="image", subtype="png")

# Which port to use etc depends on the mail server.
# Traditionally, port 25 is SMTP, but modern SMTP MSA submission uses 587.
# Some servers accept encrypted SMTP_SSL on port 465.
# Here, we use SMTP instead of SMTP_SSL, but pivot to encrypted
# traffic with STARTTLS after the initial handshake.
with smtplib.SMTP("smtp.example.org", 587) as server:
    # Some servers insist on this, others are more lenient ...
    # It is technically required by ESMTP, so let's do it
    # (If you use server.login() Python will perform an EHLO first
    # if you haven't done that already, but let's cover all bases)
    server.ehlo()
    # Whether or not to use STARTTLS depends on the mail server
    server.starttls()
    # Bewilderingly, some servers require a second EHLO after STARTTLS!
    server.ehlo()
    # Login is the norm rather than the exception these days
    # but if you are connecting to a local mail server which is
    # not on the public internet, this might not be useful or even possible
    server.login("me.myself@example.org", "xyzzy")

    # Finally, send the message
    server.send_message(msg)

Bcc: 标头的最终可见性取决于邮件服务器。如果您想真正确保收件人彼此不可见,也许根本不要放置 Bcc: 标头,并像以前那样单独枚举信封中的信封收件人必须使用 sendmailsend_message 也允许您这样做,但如果您只想发送给标题中指定的收件人,则不必这样做)。

这显然会一次性向所有收件人发送一条消息。如果您要向很多人发送相同的信息,这通常是您应该做的。但是,如果每条消息都是唯一的,您将需要遍历收件人并为每个消息创建和发送一条新消息。 (仅仅希望将收件人的姓名和地址放在 To: 标头中可能不足以保证发送比所需更多的消息,但当然,有时您在正文中也有每个收件人的唯一内容。)