Python: Error sending more than one email in a row - python
I have made a development who let me to send emails. If I try to send an email to one receipt there is no problem. But, If I try to send the same email to more than one receipts then I've got an error. I don't want to send an email to several receipts, what I want is to send one email to each of the receipts.
The error that I've got is:
File "/home/josecarlos/Workspace/python/reports/reports/pregame.py", line 22, in __init__
self.send_mail(subject, message, fileName)
File "/home/josecarlos/Workspace/python/reports/reports/report.py", line 48, in send_mail
message = mail.create_message()
File "/home/josecarlos/Workspace/python/reports/com/mail/mail.py", line 100, in create_message
message.attach(MIMEText(self.params["message"], "plain"))
File "/usr/lib/python3.6/email/mime/text.py", line 34, in __init__
_text.encode('us-ascii')
AttributeError: 'dict' object has no attribute 'encode'
To send an email I have this class:
import base64
import logging
import os
import os.path
import pickle
from email import encoders
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.base import MIMEBase
from google_auth_oauthlib.flow import InstalledAppFlow
from google.auth.transport.requests import Request
from googleapiclient import errors
from googleapiclient.discovery import build
class Mail:
def __init__(self, params):
'''
:param params: It's a dictionary with these keys:
from: Email account from the email is sended
to: Email account who will receive the email
subject: Subject of the email
message: Message of the email.
game: Next games
'''
self.params = params
#staticmethod
def get_service():
"""Gets an authorized Gmail API service instance.
Returns:
An authorized Gmail API service instance..
"""
# If modifying these scopes, delete the file token.pickle.
SCOPES = [
#'https://www.googleapis.com/auth/gmail.readonly',
'https://www.googleapis.com/auth/gmail.send',
]
creds = None
# The file token.pickle stores the user's access and refresh tokens, and is
# created automatically when the authorization flow completes for the first
# time.
if os.path.exists('token.pickle'):
with open('token.pickle', 'rb') as token:
creds = pickle.load(token)
# If there are no (valid) credentials available, let the user log in.
if not creds or not creds.valid:
if creds and creds.expired and creds.refresh_token:
creds.refresh(Request())
else:
flow = InstalledAppFlow.from_client_secrets_file(
'com/mail/credentials.json', SCOPES)
creds = flow.run_local_server(port=0)
# Save the credentials for the next run
with open('token.pickle', 'wb') as token:
pickle.dump(creds, token)
service = build('gmail', 'v1', credentials=creds)
return service
#staticmethod
def send_message(service, sender, message):
"""Send an email message.
Args:
service: Authorized Gmail API service instance.
sender: User's email address. The special value "me"
can be used to indicate the authenticated user.
message: Message to be sent.
Returns:
Sent Message.
"""
try:
sent_message = (service.users().messages().send(userId=sender, body=message)
.execute())
logging.info('Message Id: %s', sent_message['id'])
return sent_message
except errors.HttpError as error:
logging.error('An HTTP error occurred: %s', error)
def create_message(self):
"""Create a message for an email.
Args:
sender: Email address of the sender.
to: Email address of the receiver.
subject: The subject of the email message.
message_text: The text of the email message.
Returns:
An object containing a base64url encoded email object.
"""
#message = MIMEText(message_text)
message = MIMEMultipart()
message['from'] = self.params["from"]
message['to'] = self.params["to"]
message['subject'] = self.params["subject"]
message.attach(MIMEText(self.params["message"], "plain"))
routeFile = self.params["routeFile"] + self.params["fileName"]
fileName = self.params["fileName"]
# Open PDF file in binary mode
with open(routeFile, "rb") as attachment:
# Add file as application/octet-stream
# Email client can usually download this automatically as attachment
part = MIMEBase("application", "octet-stream")
part.set_payload(attachment.read())
# Encode file in ASCII characters to send by email
encoders.encode_base64(part)
# Add header as key/value pair to attachment part
part.add_header(
"Content-Disposition",
f"attachment; filename= {fileName}",
)
# Add attachment to message and convert message to string
message.attach(part)
s = message.as_string()
b = base64.urlsafe_b64encode(s.encode('utf-8'))
return {'raw': b.decode('utf-8')}
To send an email, I'll do it with this method:
def send_mail(self, subject, message, fileName):
args = self.params["destiny"]
if self.params["competition"] == COMPETITIONS.LF1 or self.params["competition"] == COMPETITIONS.LF2:
data = SearchData(args, "subscriptors.emails=")
else:
data = SearchDataFIBA(args, "subscriptors.emails=")
emails = data.get_result().getData()
for item in emails:
print(f"Enviamos informe a la cuenta: {item['email']}")
params = {
"from" : "basketmetrics#gmail.com",
"to" : item["email"],
"subject": subject,
"message" : message,
"fileName" : fileName,
"routeFile" : f"output/reports/{self.params['destiny']}/"
}
mail = Mail(params)
message = mail.create_message()
service = mail.get_service()
mail.send_message(service, "basketmetrics#gmail.com", message)
My application has secure access to google account.
I don't know how I cannot send more than an email in a row when I have no
problem to send only one e-mail.
Am I doing something wrong?
Edit I:
To reproduce the error, you can test it with this test code:
import unittest
import os
from com.mail.mail import Mail
class TestSendMail(unittest.TestCase):
def setUp(self) -> None:
os.chdir(os.path.abspath(os.path.join(os.path.dirname( __file__ ), '..')))
def test_send_mail(self):
message = "¡¡¡Hola!!!\n\nOs enviamos el informe pre partido previo a vuestro próximo partido.\n\nSaludos,\n\nBasketmetrics.com"
subject = "Informe pre partido"
fileName = "name_of_the_file"
emails = [{"email" : "receipt1#gmail.com"}, {"email" : "receipt2#gmail.com"}]
for item in emails:
print(f"Enviamos informe a la cuenta: {item['email']}")
params = {
"from" : "sender#gmail.com",
"to" : item["email"],
"subject": subject,
"message" : message,
"fileName" : fileName,
"routeFile" : "route to the file"
}
mail = Mail(params)
message = mail.create_message()
service = mail.get_service()
mail.send_message(service, "sender#gmail.com", message)
Also, you have to change some values for your own values and the file credentials.json file of your own google account
Edit II:
I have found where it produces the error but not why. The problem comes up when I invoque for second time the class Mail. In that moment I pass some parameters to the constructor with params variable. In that variable I pass the text of the message. This message is create outside of the loop.
If I read the first 120 characters of what receives the Mail class in params["message"] in the constructor:
def __init__(self, params):
'''
:param params: It's a dictionary with these keys:
from: Email account from the email is sended
to: Email account who will receive the email
subject: Subject of the email
message: Message of the email.
game: Next games
'''
self.params = params
print(f"message received: {params['message'][:120]}")
For the first time, I've got the content of message variable:
message received: ¡¡¡Hola!!!
Os enviamos el informe pre partido previo a vuestro próximo partido.
Saludos,
Basketmetrics.com
But the second time, I should receive the same text!!! But I receive an error:
Error
Traceback (most recent call last):
File "/usr/lib/python3.6/unittest/case.py", line 59, in testPartExecutor
yield
File "/usr/lib/python3.6/unittest/case.py", line 605, in run
testMethod()
File "/home/josecarlos/Workspace/python/reports/test/test_send_mail.py", line 26, in test_send_mail
mail = Mail(params)
File "/home/josecarlos/Workspace/python/reports/com/mail/mail.py", line 27, in __init__
print(f"message received: {params['message'][:120]}")
TypeError: unhashable type: 'slice'
If I read all the message without any limit or characers. For the first time, I receive the content of the variable message. But in the second time, I receive a very big string os characters, here is a little example:
lGWFBVZzhQc3pDNWdiNmhnMW1odDZHcmlFZWsyClZvTTBRT1R2dXpWaWlyZkpleS9mZXhtR3V3V2hTV0JjWERtSUNnWENTQVJ1QjdrN1Nzd3BrZ0c1Rkl3MXVDMmNyZk95ZUhySVM1dHQKSUh2T1YvWW1Pd2YzL3B2WEpLaEMza
Why I receive this string of charanters instead of the value of message variable?
I have check it that If I put message variable inside the for loop, instead of outside of the loop ... It works!!! I receive the two mails!!!
But, this solution isn't usefull because I want to reuse my code and I need to pass some values through variables.
So, why in the second time I don't receive the value of message variable and I receive a long string of characters?
How can I fix this error? Why it happens this error?
Edit III:
Checking the type of the value that I receive in the constructor of Mail, for the first time is "string":
typeof: <class 'str'>
But in the second time is "dict":
typeof: <class 'dict'>
And checking the keys of self.params["message"] are:
keys: dict_keys(['raw'])
I dont't understand anything ... How is it possible that params["message"] has the value of message variable and the second time params["message"] has modified its type to raw?
Edit IV:
I have modified the content of message variable from ...
message= ""
To ...
mesage = "¡¡¡Hola!!!0x0a0x0aOs enviamos el informe pre partido previo a vuestro próximo partido.0x0a0x0aSaludos,0x0a0x0aBasketmetrics.com"
But it doesn't work. I've got the same error.
Edit V:
I have modified the contento of message variable. Now, instead of plain text, I'm going to send an html ...
message = """\
<html>
<head></head>
<body>
<p>
Hi, this is a test!!!
</p>
<p>
Best regards!!!
</p>
</body>
</html>
"""
To do send this message, you have to modify this instruction in method create_message of Mail class:
message.attach(MIMEText(self.params["message"], "plain"))
to this:
message.attach(MIMEText(self.params["message"], "html"))
And ... I've got the same error!!!
I don't know what to do anymore ...
Edit VI:
Last attempt ... I have modified the text of the message removing "strange" character like "" or "<" and I have send a simple text with the message "Hello".
So, my message variable now is:
message = "Hello"
And I have modified again the format of the email, from "html" to "plain"
And ... I've got the same error in my second email!!!
This is frustrating ... :((((((
You have:
def send_mail(self, subject, message, fileName):
Where argument message is the message text to be sent. But we have in your function `test_send_mail':
def test_send_mail(self):
message = "¡¡¡Hola!!!\n\nOs enviamos el informe pre partido previo a vuestro próximo partido.\n\nSaludos,\n\nBasketmetrics.com"
# code omitted
for item in emails:
# code omitted
message = mail.create_message() # overlaying message text with return value from create_message
service = mail.get_service()
mail.send_message(service, "sender#gmail.com", message)
In the for item in emails: loop, you have overlaid message with the return value from the call to mail.create_message(), so for the next iteration message is not a string anymore. This, I believe is your problem. You need to use a different variable name for either the return value from the call to mail.create_message() or for the message text.
So the best solution would be to have a small list of all the email ids, and run through the for loop and you can send the mails, Or you can actually use this package from python to help you assist you in your work,
A simple example would be like this
def send_mail(names, emails, senderMail, password, subject):
port = 587
password = password
total = len(names)
server = smtplib.SMTP('smtp.gmail.com', port)
server.starttls()
server.login(senderMail, password)
cc = 0
for i in range(len(emails)):
body = '' #body of the mail, you can actually add html also please check the documentation
msg = MIMEMultipart()
msg['From'] = senderMail
msg['To'] = emails[i]
msg['Subject'] = subject
msg.attach(MIMEText(body, 'plain'))
text = msg.as_string()
server.sendmail(senderMail,emails[i], text)
cc += 1
print(cc , " / ", total)
server.quit()
If you want a easier approach attaching images you can check this package also,
Please check this github repo for detailed work flow
Related
e-mail automation using python
I use an automation of sending e-mails using python, the problem is that sometimes the code works and sometimes it doesn't. In the code python basically takes my contacts from the txt and then my e-mail message in another txt and sends the message from my txt of messages to the contacts stored in my contact.txt. When the problem is most of the time when I update mycontacts.txt and the error that appears is: Traceback (most recent call last): File "c:/emailbot/botmessav1.py", line 70, in <module> main() File "c:/emailbot/botmessav1.py", line 36, in main names, emails = get_contacts('mycontacts.txt') # read contacts File "c:/emailbot/botmessav1.py", line 21, in get_contacts names.append(a_contact.split()[0]) IndexError: list index out of range ``` The following is an example of the contact list, mycontacts.txt: marina marinam#gmail.com luis luis#gmail.com carlos carlos#gmail.com marcelo marcelom#gmail.com And this is the code I'm using: import smtplib from string import Template from email.mime.multipart import MIMEMultipart from email.mime.text import MIMEText MY_ADDRESS = 'e-mail' PASSWORD = 'password' def get_contacts(mycontacts): """ Return two lists names, emails containing names and email addresses read from a file specified by filename. """ names = [] emails = [] with open(mycontacts, mode='r', encoding='utf-8') as contacts_file: for a_contact in contacts_file: names.append(a_contact.split()[0]) emails.append(a_contact.split()[1]) return names, emails def read_template(message): """ Returns a Template object comprising the contents of the file specified by filename. """ with open(message, 'r', encoding='utf-8') as template_file: template_file_content = template_file.read() return Template(template_file_content) def main(): names, emails = get_contacts('mycontacts.txt') # read contacts message_template = read_template('message.txt') # set up the SMTP server s = smtplib.SMTP(host='smtp.gmail.com', port=587) s.starttls() s.login(MY_ADDRESS, PASSWORD) # For each contact, send the email: for name, email in zip(names, emails): msg = MIMEMultipart() # create a message # add in the actual person name to the message template message = message_template.substitute(PERSON_NAME=name.title()) # Prints out the message body for our sake print(message) # setup the parameters of the message msg['From']=MY_ADDRESS msg['To']=email msg['Subject']="Subject" # add in the message body msg.attach(MIMEText(message, 'plain')) # send the message via the server set up earlier. s.send_message(msg) del msg # Terminate the SMTP session and close the connection s.quit() if __name__ == '__main__': main()
You should not have any empty lines in contacts.txt file marina marinam#gmail.com luis luis#gmail.com carlos carlos#gmail.com marcelo marcelom#gmail.com
Why is my code opening another file and handling error exceptions?
I am trying to run a file (asana_get_user.py) in C:\Users\WilsonNg\Documents\Internship via python. However, when I run it, I get an attribute error: error message However, there is another exception error happening after this code is finished for some reason: exception error So it opened another file (email.py) within the same folder and created an exception error. Any idea on why it referenced my email.py file? asana_get_user.py import asana # replace with your personal access token. personal_access_token = '0/2c...' # Construct an Asana client client = asana.Client.access_token(personal_access_token) # Set things up to send the name of this script to us to show that you succeeded! This is optional. client.options['client_name'] = "hello_world_python" # Get your user info me = client.users.me() # Print out your information print ("Hello world! " + "My name is " + me['name'] + " and I my primary Asana workspace is " + me['workspaces'][0]['name'] + ".") email.py def create_message(sender, to, subject, message_text): """Create a message for an email. Args: sender: Email address of the sender. to: Email address of the receiver. subject: The subject of the email message. message_text: The text of the email message. Returns: An object containing a base64url encoded email object. """ message = MIMEText(message_text) message['to'] = to message['from'] = sender message['subject'] = subject return {'raw': base64.urlsafe_b64encode(message.as_string())} def send_message(service, user_id, message): """Send an email message. Args: service: Authorized Gmail API service instance. user_id: User's email address. The special value "me" can be used to indicate the authenticated user. message: Message to be sent. Returns: Sent Message. """ try: message = (service.users().messages().send(userId=user_id, body=message) .execute()) print ('Message Id: %s' % message['id']) return message except (errors.HttpError, error): print ('An error occurred: %s' % error) subject = "This is a test subject" message = "This is a test content" sent_to = "test#test.com" sender = "test#test.com" sending_msg = create_message(sender,sent_to,subject,message) send_message()
Please put a check in email.py as if __name__ == '__main__': before subject = "This is a test subject". email.py is being run as the import asana is called. if __name__ == '__main__': subject = "This is a test subject" message = "This is a test content" sent_to = "test#test.com" sender = "test#test.com" sending_msg = create_message(sender,sent_to,subject,message) send_message() Read about it here.
How to send email via python to several recipients with contacts on text file
When I write the script below I get the error IndexError: list index out of range: import smtplib from string import Template from email.mime.multipart import MIMEMultipart from email.mime.text import MIMEText MY_ADDRESS = 'joaosoundcloudpython#gmail.com' PASSWORD = 'XXXX' def get_contacts(test1): """ Return two lists names, emails containing names and email addresses read from a file specified by filename. """ names = [] emails = [] with open('/Users/joaopimenta/Documents/joaoPython/soundcloud_email.txt', mode='r', encoding='utf-8') as contacts_file: for a_contact in contacts_file: names.append(a_contact.split()[0]) emails.append(a_contact.split()[100]) return names, emails def read_template(): """ Returns a Template object comprising the contents of the file specified by filename. """ with open(soundcloud_email_message.txt, 'r', encoding='utf-8') as template_file: template_file_content = template_file.read() return Template(template_file_content) def main(): names, emails = get_contacts('/Users/joaopimenta/Documents/joaoPython/soundcloud_email.rtf') # read contacts message_template = read_template('/Users/joaopimenta/Documents/joaoPython/soundcloud_email_message.txt') # set up the SMTP server s = smtplib.SMTP(host='smtp.gmail.com', port=587) s.starttls() s.login(joaosoundcloudpython#gmail.com, XXXX) # For each contact, send the email: for name, email in zip(names, emails): msg = MIMEMultipart() # create a message # add in the actual person name to the message template message = message_template.substitute(PERSON_NAME=name.title()) # Prints out the message body for our sake print(message) # setup the parameters of the message msg['From']=MY_ADDRESS msg['To']=email msg['Subject']="This is TEST" # add in the message body msg.attach(MIMEText(message, 'plain')) # send the message via the server set up earlier. s.send_message(msg) del msg # Terminate the SMTP session and close the connection s.quit() if __name__ == '__main__': main()
How do I send HTML Formatted emails, through the gmail-api for python
Using the sample code from the GMail API Example: Send Mail, and after following rules for authentication, it's simple enough to send a programmatically generated email, via a gmail account. What isn't obvious from the example is how to set that email to be HTML formatted. The Question How do I get HTML formatting in my gmail-api send messages, using python? I have this... message_body = "Hello!\nYou've just received a test message!\n\nSincerely,\n-Test Message Generator\n" and I want it to be this... Hello! You've just received a test message! Sincerely, -Test Message Generator Example Source Code from GMail-API Below is a slightly modified version of the example, but still works: import argparse import base64 from pprint import pformat from pprint import pprint import httplib2 import os from email.MIMEMultipart import MIMEMultipart from email.MIMEText import MIMEText from apiclient import discovery from oauth2client import client from oauth2client import tools from oauth2client.file import Storage SCOPES = 'https://mail.google.com/' CLIENT_SECRET_FILE = 'client_secret.json' APPLICATION_NAME = 'Test EMail App' def get_credentials(): """Gets valid user credentials from storage. If nothing has been stored, or if the stored credentials are invalid, the OAuth2 flow is completed to obtain the new credentials. Returns: Credentials, the obtained credential. """ home_dir = os.path.expanduser('~') credential_dir = os.path.join(home_dir, '.credentials') if not os.path.exists(credential_dir): os.makedirs(credential_dir) credential_path = os.path.join(credential_dir, 'gmail-python-quickstart.json') store = Storage(credential_path) credentials = store.get() if not credentials or credentials.invalid: flow = client.flow_from_clientsecrets(CLIENT_SECRET_FILE, SCOPES) flow.user_agent = APPLICATION_NAME if flags: credentials = tools.run_flow(flow, store, flags) else: # Needed only for compatibility with Python 2.6 credentials = tools.run(flow, store) print('Storing credentials to ' + credential_path) return credentials def create_message(sender, to, cc, subject, message_text): """Create a message for an email. Args: sender: Email address of the sender. to: Email address of the receiver. subject: The subject of the email message. message_text: The text of the email message. Returns: An object containing a base64url encoded email object. """ print(sender + ', ' + to + ', ' + subject + ', ' + message_text) message = MIMEText(message_text) message['to'] = to message['from'] = sender message['subject'] = subject message['cc'] = cc pprint(message) return {'raw': base64.urlsafe_b64encode(message.as_string())} def send_message(service, user_id, message_in): """Send an email message. Args: service: Authorized Gmail API service instance. user_id: User's email address. The special value "me" can be used to indicate the authenticated user. message: Message to be sent. Returns: Sent Message. """ pprint(message_in) try: message = (service.users().messages().send(userId=user_id, body=message_in).execute()) pprint(message) print ('Message Id: %s' % message['id']) return message except errors.HttpError, error: print ('An error occurred: %s' % error) def main(cli): """Shows basic usage of the Gmail API. Creates a Gmail API service object and outputs a list of label names of the user's Gmail account. """ credentials = get_credentials() http = credentials.authorize(httplib2.Http()) service = discovery.build('gmail', 'v1', http=http) email_msg = create_message(cli.addr_from, cli.addr_to, cli.addr_cc, cli.subject, cli.message) msg_out = service.users().messages().send(userId = 'me', body = email_msg).execute() pprint(msg_out) if __name__ == '__main__': parser = argparse.ArgumentParser() parser.add_argument('-m', '--message', help = 'The message to send in the email', default='<MESSAGE = unfinished>') parser.add_argument('-t', '--addr_to', help = 'the list of comma separated emails to send', default='cbsd.tools#gmail.com') parser.add_argument('-s', '--subject', help = 'the email subject', default='<SUBJECT = undefined>') parser.add_argument('-c', '--addr_cc', help = 'email CC\'s', default='') parser.add_argument('-f', '--addr_from', help = 'Email address to send from', default='cbsd.tools#gmail.com') cli = parser.parse_args() pprint(dir(cli)) main(cli) Tried as I might, with this code, and variations on it, I could not get html formatted code, nor could I get simple escape characters to create carriage returns where they needed to be. Here's what didn't work Trying the following didn't work either: modifying line 69 to add additional message dictionary parameters... ie. {'raw': base64.urlsafe_b64encode(message.as_string()), 'payload': {'mimeType': 'text/html'}} As documented here in the GMail API Docs Adding a variety of escaped backslashes into the message text: \n... ie: \\n nor \\\n and just rendered as those exact characters Adding <br> </br> <br/> did not add new lines and just rendered as those exact characters Adding \r did not add new lines and just rendered as that exact character Reasons this is not a duplicate question This SO link deals with multipart/signed messages I am uninterested in multipart messaging because it is an inelegant solution. I want the WHOLE message to be HTML. Additionally, this link has no accepted answer, and the one answer present is a non-answer as it simply states that they are sending a problem statement to Google. This one deals with c#, which I am not interested in since I'm writing in python This one also uses SMTP which I'm not interested in This one is for Rails
After doing a lot of digging around, I started looking in to the python side of the message handling, and noticed that a python object is actually constructing the message to be sent for base64 encoding into the gmail-api message object constructor. See line 63 from above: message = MIMEText(message_text) The one trick that finally worked for me, after all the attempts to modify the header values and payload dict (which is a member of the message object), was to set (line 63): message = MIMEText(message_text, 'html') <-- add the 'html' as the second parameter of the MIMEText object constructor The default code supplied by Google for their gmail API only tells you how to send plain text emails, but they hide how they're doing that. ala... message = MIMEText(message_text) I had to look up the python class email.mime.text.MIMEText object. That's where you'll see this definition of the constructor for the MIMEText object: class email.mime.text.MIMEText(_text[, _subtype[, _charset]]) We want to explicitly pass it a value to the _subtype. In this case, we want to pass: 'html' as the _subtype. Now, you won't have anymore unexpected word wrapping applied to your messages by Google, or the Python mime.text.MIMEText object The Fixed Code def create_message(sender, to, cc, subject, message_text): """Create a message for an email. Args: sender: Email address of the sender. to: Email address of the receiver. subject: The subject of the email message. message_text: The text of the email message. Returns: An object containing a base64url encoded email object. """ print(sender + ', ' + to + ', ' + subject + ', ' + message_text) message = MIMEText(message_text,'html') message['to'] = to message['from'] = sender message['subject'] = subject message['cc'] = cc pprint(message) return {'raw': base64.urlsafe_b64encode(message.as_string())}
Try this: def CreateMessage(emailSubject, emailTo, emailFrom, message_body, emailCc, html_content=None): try: message = MIMEMultipart('alternative') message['to'] = emailTo message['from'] = emailFrom message['subject'] = emailSubject message['Cc'] = emailCc body_mime = MIMEText(message_body, 'plain') message.attach(body_mime) if html_content: html_mime = MIMEText(html_content, 'html') message.attach(html_mime) return { 'raw': base64.urlsafe_b64encode( bytes( message.as_string(), "utf-8")).decode("utf-8")} except Exception as e: print('Error in CreateMessage()', e) return '400'
The accepted answer works, but in my journey down this rabbit hole, I found the right place to put the payload from the original question, so here's a full example. I'm using a service account with domain-wide delegation: import base64 from googleapiclient.discovery import build from google.oauth2 import service_account from email.message import EmailMessage SCOPES = ['https://www.googleapis.com/auth/gmail.send'] CREDS = service_account.Credentials.from_service_account_file( serviceAcct, scopes=SCOPES, subject=senderEmail) with build('gmail', 'v1', credentials=CREDS) as service: msg = EmailMessage() content="Message body in <b>html</b> format!" msg['To'] = recipientEmail msg['From'] = senderEmail msg['Subject'] = 'Gmail API test' # Use this for plain text # msg.set_content(content) # Otherwise, use this for html msg.add_header('Content-Type','text/html') msg.set_payload(content) encodedMsg = base64.urlsafe_b64encode(msg.as_bytes()).decode() try: sendMsg = service.users().messages().send( userId=senderEmail, body={ 'raw': encodedMsg } ).execute() print('Msg id:', sendMsg['id']) except Exception as e: print('Error:', e)
403 error sending email with gmail API (python)
I'm trying to send an email using the Gmail API in python. I think I followed the relevant documentation and youtube vids. I'm running into this error: googleapiclient.errors.HttpError: HttpError 403 when requesting https://www.googleapis.com/gmail/v1/users/me/messages/send?alt=json returned "Insufficient Permission" Here is my script: #!/usr/bin/env python from googleapiclient.discovery import build from httplib2 import Http from oauth2client import file, client, tools from email.mime.text import MIMEText import base64 import errors SCOPES = 'https://mail.google.com/' CLIENT_SECRET = 'client_secret.json' store = file.Storage('storage.json') credz = store.get() if not credz or credz.invalid: flags = tools.argparser.parse_args(args=[]) flow = client.flow_from_clientsecrets(CLIENT_SECRET, SCOPES) credz = tools.run_flow(flow, store, flags) GMAIL = build('gmail', 'v1', http=credz.authorize(Http())) def CreateMessage(sender, to, subject, message_text): """Create a message for an email. Args: sender: Email address of the sender. to: Email address of the receiver. subject: The subject of the email message. message_text: The text of the email message. Returns: An object containing a base64url encoded email object. """ message = MIMEText(message_text) message['to'] = to message['from'] = sender message['subject'] = subject return {'raw': base64.urlsafe_b64encode(message.as_string())} def SendMessage(service, user_id, message): """Send an email message. Args: service: Authorized Gmail API service instance. user_id: User's email address. The special value "me" can be used to indicate the authenticated user. message: Message to be sent. Returns: Sent Message. """ try: message = (service.users().messages().send(userId=user_id, body=message) .execute()) print 'Message Id: %s' % message['id'] return message except errors.HttpError, error: print 'An error occurred: %s' % error message = CreateMessage('testemail#gmail.com', 'testemail#gmail.com', 'test_subject', 'foo') print message SendMessage(GMAIL, 'me', message) I tried adding scopes, trying different emails, etc. I have authenticated by logging into my browser as well. (The testemail#gmail.com is a dummy email btw)
Try deleting generated storage.json file and then try again afresh. you might be trying this script with different scopes so "storage.json" might be having wrong details.
I had the same problem. I solved it by running again the quickstart.py that provides google and change SCOPE so that google can give you all permissions you want. After that don't need to have SCOPE or CLIENT_SECRET on your new code to send a message, just get_credentials(), CreateMessage() and SendMessage() methods.