Простой SMTP-сервер (на Python) - PullRequest
20 голосов
/ 22 апреля 2010

Не могли бы вы предложить простой SMTP-сервер с очень простыми API-интерфейсами (под «простыми», я имею в виду, для чтения, записи, удаления электронной почты), который мог бы быть запущен на Linux?Мне просто нужно преобразовать суть письма в формат XML и передать его на другой компьютер.

Ответы [ 8 ]

40 голосов
/ 22 апреля 2010

Посмотрите на этот SMTP-сервер-приемник :

from datetime import datetime
import asyncore
from smtpd import SMTPServer

class EmlServer(SMTPServer):
    no = 0
    def process_message(self, peer, mailfrom, rcpttos, data):
        filename = '%s-%d.eml' % (datetime.now().strftime('%Y%m%d%H%M%S'),
                self.no)
        f = open(filename, 'w')
        f.write(data)
        f.close
        print '%s saved.' % filename
        self.no += 1


def run():
    foo = EmlServer(('localhost', 25), None)
    try:
        asyncore.loop()
    except KeyboardInterrupt:
        pass


if __name__ == '__main__':
    run()

Используется smtpd.SMTPServer для выгрузки электронных писем в файлы.

21 голосов
/ 22 апреля 2010

Для отправки электронного письма требуется 2 вещи:

  • SMTP-сервер - это может быть Python SMTP-сервер , либо вы можете использовать GMail или сервер вашего провайдера. Скорее всего, вам не нужно работать самостоятельно.
  • Библиотека SMTP - то, что отправит запрос по электронной почте на сервер SMTP. Python поставляется с библиотекой smtplib, которая может сделать это за вас. Здесь есть масса информации о том, как его использовать: http://docs.python.org/library/smtplib.html

Для чтения есть два варианта в зависимости от того, с какого сервера вы читаете письмо.

4 голосов
/ 03 августа 2017

Два SMTP-сервера Python, которые я использовал с успехом:

  1. Twisted's Mail - Очень гибкая почтовая библиотека для SMTP, IMAP, ...
  2. python-slimta - Полный MTA (сервер ретрансляции / пересылки SMTP)

Пример Twisted показан ниже

# Copyright (c) Twisted Matrix Laboratories.
# See LICENSE for details.

# You can run this module directly with:
#    twistd -ny emailserver.tac

"""
A toy email server.
"""
from __future__ import print_function

from zope.interface import implementer

from twisted.internet import defer
from twisted.mail import smtp
from twisted.mail.imap4 import LOGINCredentials, PLAINCredentials

from twisted.cred.checkers import InMemoryUsernamePasswordDatabaseDontUse
from twisted.cred.portal import IRealm
from twisted.cred.portal import Portal



@implementer(smtp.IMessageDelivery)
class ConsoleMessageDelivery:
    def receivedHeader(self, helo, origin, recipients):
        return "Received: ConsoleMessageDelivery"


    def validateFrom(self, helo, origin):
        # All addresses are accepted
        return origin


    def validateTo(self, user):
        # Only messages directed to the "console" user are accepted.
        if user.dest.local == "console":
            return lambda: ConsoleMessage()
        raise smtp.SMTPBadRcpt(user)



@implementer(smtp.IMessage)
class ConsoleMessage:
    def __init__(self):
        self.lines = []


    def lineReceived(self, line):
        self.lines.append(line)


    def eomReceived(self):
        print("New message received:")
        print("\n".join(self.lines))
        self.lines = None
        return defer.succeed(None)


    def connectionLost(self):
        # There was an error, throw away the stored lines
        self.lines = None



class ConsoleSMTPFactory(smtp.SMTPFactory):
    protocol = smtp.ESMTP

    def __init__(self, *a, **kw):
        smtp.SMTPFactory.__init__(self, *a, **kw)
        self.delivery = ConsoleMessageDelivery()


    def buildProtocol(self, addr):
        p = smtp.SMTPFactory.buildProtocol(self, addr)
        p.delivery = self.delivery
        p.challengers = {"LOGIN": LOGINCredentials, "PLAIN": PLAINCredentials}
        return p



@implementer(IRealm)
class SimpleRealm:
    def requestAvatar(self, avatarId, mind, *interfaces):
        if smtp.IMessageDelivery in interfaces:
            return smtp.IMessageDelivery, ConsoleMessageDelivery(), lambda: None
        raise NotImplementedError()



def main():
    from twisted.application import internet
    from twisted.application import service    

    portal = Portal(SimpleRealm())
    checker = InMemoryUsernamePasswordDatabaseDontUse()
    checker.addUser("guest", "password")
    portal.registerChecker(checker)

    a = service.Application("Console SMTP Server")
    internet.TCPServer(2500, ConsoleSMTPFactory(portal)).setServiceParent(a)

    return a

application = main()
2 голосов
/ 30 декабря 2017

Более современный подход заключается в использовании библиотеки aiosmtpd (документация доступна здесь ).

Хороший пример вы можете найти здесь: https://aiosmtpd.readthedocs.io/en/latest/aiosmtpd/docs/controller.html.

2 голосов
/ 23 августа 2013

Это хорошие примеры для начала.

smtpd - Образцы SMTP-серверов

http://pymotw.com/2/smtpd/index.html

smtplib - клиент простого протокола пересылки почты

http://pymotw.com/2/smtplib/index.html

1 голос
/ 22 апреля 2010

Существует Python SMTP-сервер .

Этот модуль предлагает несколько классов для реализации SMTP-серверов. Один универсальная реализация ничего не делающая, которая может быть переопределена, в то время как другие два предлагают конкретные стратегии отправки почты.

0 голосов
/ 02 мая 2018

Если вы хотите быстро протестировать send_mail Джанго с ответом Хасена выше:

# Skip lines 3 and 4 if not using virtualenv.
# At command prompt

mkdir django1
cd django1
virtualenv venv
source venv/bin/activate
pip install django==1.11
django-admin startproject django1 .

# run the Django shell

python manage.py shell

# paste into shell following:

from django.core.mail import send_mail

send_mail(
    'Subject here',
    'Here is the message.',
    'from@example.com',
    ['to@example.com'],
    fail_silently=False,
)
# This should write an email like the following:

Content-Type: text/plain; charset="utf-8"
MIME-Version: 1.0
Content-Transfer-Encoding: 7bit
Subject: Subject here
From: from@example.com
To: to@example.com
Date: Wed, 02 May 2018 02:12:09 -0000
Message-ID: <20180502021209.32641.51865@i1022>

Here is the message.

Не обязательно иметь допустимые значения в функции send_mail. Приведенные выше значения будут хорошо работать с примером Хасена.

0 голосов
/ 12 июня 2017

Чтобы заставить скрипт Хасена работать в Python 3, мне пришлось немного его настроить:

from datetime import datetime
import asyncore
from smtpd import SMTPServer

class EmlServer(SMTPServer):
    no = 0
    def process_message(self, peer, mailfrom, rcpttos, data, **kwargs):
        filename = '%s-%d.eml' % (datetime.now().strftime('%Y%m%d%H%M%S'),
            self.no)
        print(filename)
        f = open(filename, 'wb')
        f.write(data)
        f.close
        print('%s saved.' % filename)
        self.no += 1

def run():
    EmlServer(('localhost', 25), None)
    try:
        asyncore.loop()
    except KeyboardInterrupt:
        pass

if __name__ == '__main__':
    run()
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...