添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
Collectives™ on Stack Overflow

Find centralized, trusted content and collaborate around the technologies you use most.

Learn more about Collectives

Teams

Q&A for work

Connect and share knowledge within a single location that is structured and easy to search.

Learn more about Teams

Sending an email via the Python email library throws error "expected string or bytes-like object"

Ask Question

I am trying to send a csv file as an attachment via a simple function in python 3.6.

from email.message import Message
from email.mime.base import MIMEBase
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
def email():
    msg = MIMEMultipart()
    msg['Subject'] = 'test'
    msg['From'] = 'test@gmail.com'
    msg['To'] = 'testee@gmail.com'
    msg.preamble = 'preamble'
    with open("test.csv") as fp:
        record = MIMEText(fp.read())
        msg.attach(record)
    server = smtplib.SMTP('smtp.gmail.com', 587)
    server.ehlo()
    server.starttls()
    server.login("test@gmail.com", "password")
    server.sendmail("test@gmail.com", "testee@gmail.com", msg)
    server.quit()

Calling email() produces the error expected string or bytes-like object. Redefining server.sendmail("test@gmail.com", "testee@gmail.com", msg) as server.sendmail("atest@gmail.com", "testee@gmail.com", msg.as_string()) causes an email to be sent, but sends the csv file in the body of the email, NOT as an attachment. can anyone give me some pointers on how to send the csv file as an attachment?

Have you tried adding a Content-Disposition header to your attachment? msg.add_header('Content-Disposition', 'attachment', filename='test.csv') – Birne94 Jan 4, 2017 at 17:37 Is this a copy/paste error or are you actually missing a ' on msg['From'] = 'test@gmail.com? – Andrew_CS Jan 4, 2017 at 17:37 @Andrew_CS, Copy paste error. Thank you for catching it! I've tried adding that header, and had the same issue- the email was sent successfully, but as plaintext in the email body. prntscr.com/drhazy – MeesterTeem Jan 4, 2017 at 17:46

1) You should use msg.as_string() if you call smtplib.SMTP.sendmail(). Alternatively, if you have Python 3.2 or newer, you can use server.send_message(msg).

2) You should add a body to your message. By design no one ever sees the preamble.

3) You should use content-disposition: attachment to indicate which parts are attachments and which are inline.

Try this:

def email():
    msg = MIMEMultipart()
    msg['Subject'] = 'test'
    msg['From'] = 'XXX'
    msg['To'] = 'XXX'
    msg.preamble = 'preamble'
    body = MIMEText("This is the body of the message")
    msg.attach(body)
    with open("test.csv") as fp:
        record = MIMEText(fp.read())
        record['Content-Disposition'] = 'attachment; filename="test.csv"'
        msg.attach(record)
    server = smtplib.SMTP('smtp.gmail.com', 587)
    server.ehlo()
    server.starttls()
    server.login("XXX", "XXX")
    server.sendmail("XXX", "XXX", msg.as_string())
    server.quit()
                This worked perfectly! Thank you so much. I'd tried a file header, but i attached it incorrectly.
– MeesterTeem
                Jan 4, 2017 at 17:57
                This is working. "If your gmail is secured by Two-Factor Authentication, you must first generate an application specific password security.google.com/settings/security/apppasswords?pli=1 then use that app-password for in the above example code (this is very important because then you aren't writing your password down anywhere in cleartext AND you can revoke the app-password at any time)". This is taken from this link: stackoverflow.com/a/12424439/1944314 comment section below.
– Alper
                Oct 6, 2022 at 19:14
        

Thanks for contributing an answer to Stack Overflow!

  • Please be sure to answer the question. Provide details and share your research!

But avoid

  • Asking for help, clarification, or responding to other answers.
  • Making statements based on opinion; back them up with references or personal experience.

To learn more, see our tips on writing great answers.