Подтвердить что ты не робот

Как отправить электронную почту через gmail без включения "небезопасного доступа"?

Google подталкивает нас к повышению безопасности script доступа к их gmail smtp-серверам. У меня нет проблем с этим. На самом деле я рад помочь.

Но они не облегчают. Все хорошо и полезно предложить нам Upgrade to a more secure app that uses the most up to date security measures, но это не помогает мне понять, как обновлять биты кода, которые выглядят следующим образом:

server = smtplib.SMTP("smtp.gmail.com", 587)
server.ehlo()
server.starttls()
server.login(GMAIL_USER, GMAIL_PASSWORD)
server.sendmail(FROM, TO, MESSAGE)
server.close()

Конечно, я поеду и включу "Доступ к менее безопасным приложениям", но если кто-то разработает, что заменить этот код, я буду благодарен.

4b9b3361

Ответ 1

Кажется, что ответ Джона Ми был устаревшим. Он не работает в июле 2016 года. Возможно, из-за обновления API Gmail. Я обновляю его код (python 2), как показано ниже:

    """Send an email message from the user account.
"""

import base64
from email.mime.audio import MIMEAudio
from email.mime.base import MIMEBase
from email.mime.image import MIMEImage
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
import mimetypes
import os

#from __future__ import print_function
import httplib2
import os

from apiclient import discovery
import oauth2client
from oauth2client import client
from oauth2client import tools

from apiclient import errors

SCOPES = 'https://www.googleapis.com/auth/gmail.compose'
CLIENT_SECRET_FILE = 'client_secret.json'
APPLICATION_NAME = 'Gmail API Python Quickstart'

try:
    import argparse
    flags = argparse.ArgumentParser(parents=[tools.argparser]).parse_args()
except ImportError:
    flags = None

def SendMessage(service, user_id, message):
  """Send an email message.

  Args:
    service: Authorized Gmail API service instance.
    user_id: User 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


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 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,
                                   'sendEmail.json')

    store = oauth2client.file.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

if __name__ == "__main__":

    try:
        credentials = get_credentials()
        http = credentials.authorize(httplib2.Http())
        service = discovery.build('gmail', 'v1', http=http)
        SendMessage(service, "me", CreateMessage("[email protected]", "[email protected]", "Test gmail automation", "Hello world"))

    except Exception, e:
        print e
        raise

Обратите внимание, что если вы столкнулись с ошибкой Insufficient Permission, одна из возможных причин заключается в том, что область в программе установлена ​​неправильно. Другая возможная причина может заключаться в том, что вам нужно удалить файл json для хранения ( "sendEmail.json" в этой программе) и обновить свою программу. Более подробную информацию можно найти в этом сообщении .

Ответ 2

Это было больно, но, похоже, сейчас у меня что-то происходит...

Python3 не поддерживается (пока)

Я не думаю, что это будет слишком сложно достичь, поскольку я спотыкался об конвертации пакетов, не нажимая ничего массивного: просто обычный материал 2to3. Но через пару часов я устал плавать вверх по течению. На момент написания статьи я не смог найти опубликованный пакет для общественного потребления для Python 3. Опыт python 2 был прямым (в сравнении).

Навигация по сайту Google - половина битвы

Без сомнения, со временем это изменится. В конечном итоге вам необходимо загрузить файл client_secret.json. Вы можете (возможно) выполнить эту настройку через веб-браузер:

  • Вам нужна учетная запись google - либо приложения Google, либо gmail. Итак, если у вас его нет, найдите один.
  • Вернитесь в консоль разработчиков
  • Создайте новый проект и подождите 4 или 400 секунд для завершения.
  • Перейдите к API and AuthCredentials
  • В OAuth выберите Create New Client ID
  • Выберите Installed Application в качестве типа приложения и Другое
  • Теперь у вас должна быть кнопка Download JSON. Сделай это. Это ваш client_secret.json - пароли, так сказать.

Но подождите, что не все!

Вы должны указать вашему приложению "Имя продукта", чтобы избежать некоторых нечетных ошибок. (см., сколько я страдал, чтобы дать вам это; -)

  • Перейдите к API & authConsent Screen
  • Выберите свой адрес электронной почты
  • Введите имя PRODUCT. Неважно, что это такое. "Foobar" прекрасно справится.
  • Сохранить

Newsflash! Вау. Теперь еще больше!

  • Перейдите в API и auth → API → API Gmail
  • Нажмите кнопку "Включить API"

Yay. Теперь мы можем обновить электронную почту script.

Python 2

Вам нужно запустить интерактивный script в первый раз. Он откроет веб-браузер на вашем компьютере, и вы предоставите разрешения (нажмите кнопку). Это упражнение сохранит файл на вашем компьютере gmail.storage, который содержит многоразовый токен.

[Мне не повезло передать токен машине, которая не имеет графических функций браузера - возвращает HTTPError. Я попытался пройти через графический браузер lynx. Это также потерпело неудачу, потому что Google установил окончательную кнопку "accept" на "disabled"!? Я подниму еще один вопрос, чтобы преодолеть это препятствие (более ворчание)]

Сначала вам нужны библиотеки:

pip install --upgrade google-api-python-client
pip install --upgrade python-gflags
  • вам нужно изменить адреса и из них.
  • убедитесь, что у вас есть файл client_token.json, если команды Storage ожидают его
  • каталог должен быть доступен для записи, чтобы он мог сохранить файл gmail.storage

Наконец, некоторый код:

import base64
import httplib2

from email.mime.text import MIMEText

from apiclient.discovery import build
from oauth2client.client import flow_from_clientsecrets
from oauth2client.file import Storage
from oauth2client.tools import run


# Path to the client_secret.json file downloaded from the Developer Console
CLIENT_SECRET_FILE = 'client_secret.json'

# Check https://developers.google.com/gmail/api/auth/scopes for all available scopes
OAUTH_SCOPE = 'https://www.googleapis.com/auth/gmail.compose'

# Location of the credentials storage file
STORAGE = Storage('gmail.storage')

# Start the OAuth flow to retrieve credentials
flow = flow_from_clientsecrets(CLIENT_SECRET_FILE, scope=OAUTH_SCOPE)
http = httplib2.Http()

# Try to retrieve credentials from storage or run the flow to generate them
credentials = STORAGE.get()
if credentials is None or credentials.invalid:
  credentials = run(flow, STORAGE, http=http)

# Authorize the httplib2.Http object with our credentials
http = credentials.authorize(http)

# Build the Gmail service from discovery
gmail_service = build('gmail', 'v1', http=http)

# create a message to send
message = MIMEText("Message goes here.")
message['to'] = "[email protected]"
message['from'] = "[email protected]"
message['subject'] = "your subject goes here"
body = {'raw': base64.b64encode(message.as_string())}

# send it
try:
  message = (gmail_service.users().messages().send(userId="me", body=body).execute())
  print('Message Id: %s' % message['id'])
  print(message)
except Exception as error:
  print('An error occurred: %s' % error)

Надеюсь, что мы все начнем. Не так просто, как по-старому, но выглядит намного менее сложным, теперь я вижу его во плоти.

Ответ 4

Я включаю некоторый код, который был обновлен для использования python 3 - кажется, он отправляет электронные письма после получения необходимых разрешений и токенов OAuth. Это в основном основано на примерах веб-сайта google api.

    from __future__ import print_function

import base64
import os
from email.mime.text import MIMEText

import httplib2
from apiclient import discovery
from googleapiclient import errors
from oauth2client import client
from oauth2client import tools
from oauth2client.file import Storage

try:
    import argparse

    flags = argparse.ArgumentParser(parents=[tools.argparser]).parse_args()
except ImportError:
    flags = None

# If modifying these scopes, delete your previously saved credentials
# at ~/.credentials/gmail-python-quickstart.json
SCOPES = 'https://www.googleapis.com/auth/gmail.send'
CLIENT_SECRET_FILE = 'client_secret.json'
APPLICATION_NAME = 'Gmail API Python Quickstart'


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


to = '[email protected]'
sender = '[email protected]'
subject = 'test emails'
message_text = 'hello this is a text test message'
user_id = 'me'

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_bytes()).decode())}


def send_message(service, user_id, message):
    """Send an email message.

    Args:
      service: Authorized Gmail API service instance.
      user_id: User 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: {}'.format(message['id']))
        return message
    except errors.HttpError as error:
        print('An error occurred: {}'.format(error))


def main():
    """Shows basic usage of the Gmail API.

    Creates a Gmail API service object and outputs a list of label names
    of the user Gmail account.
    """
    credentials = get_credentials()
    http = credentials.authorize(httplib2.Http())
    service = discovery.build('gmail', 'v1', http=http)

    msg = create_message(sender,to,subject,message_text)
    message = (service.users().messages().send(userId=user_id, body=msg)
               .execute())
    print('Message Id: {}'.format(message['id']))
    results = service.users().messages().list(userId='me').execute()
    labels = results.get('labels', [])

    if not labels:
        print('No labels found.')
    else:
        print('Labels:')
        for label in labels:
            print(label['name'])


if __name__ == '__main__':
    main()

Ответ 5

  protected string SendEmail(string toAddress, string subject, string body)
    {
        string result = "Message Sent Successfully..!!";

        string senderID = "...........";// use sender email id here..
        const string senderPassword = "........."; // sender password here...

        try
        {
            SmtpClient smtp = new SmtpClient
            {
                Host = "smtp.gmail.com", // smtp server address here...
                Port = 587,
                EnableSsl = true,
                DeliveryMethod = SmtpDeliveryMethod.Network,
                Credentials = new System.Net.NetworkCredential(senderID, senderPassword),
                Timeout = 30000,

            };

            MailMessage message = new MailMessage(senderID, toAddress, subject, body);

            smtp.Send(message);
        }
        catch (Exception ex)
        {
            result = "Error sending email.!!!";
        }

        return result;
    }