Um servidor SMTP simples (em Python)
poderia sugerir um servidor SMTP simples com as APIs muito básicas (por muito básicas, quero dizer, ler, escrever, apagar e-mail), que poderia ser executado numa caixa linux? Eu só preciso converter o crux do E-mail em formato XML e FTP ele para outra máquina.
8 answers
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()
Ele usa smtpd.SMTPServer
para enviar e-mails para ficheiros.
- um servidor de SMTP - este pode ser o servidor de SMTP em Python ou pode usar o GMail ou o servidor do seu ISP. É provável que não precises de gerir os teus.
- uma biblioteca SMTP-algo que irá enviar um pedido de E-mail para o servidor SMTP. Naves Python com uma biblioteca chamada smtplib que pode fazer isso por você. Há toneladas de informações sobre como usá-lo aqui: http://docs.python.org/library/smtplib.html
Para ler, existem duas opções dependendo de qual servidor você está lendo o e-mail.
- para um servidor de E-mail POP - pode usar a biblioteca poplib python: http://docs.python.org/library/poplib.html
- para um servidor de E-mail de IMAP-poderá usar a biblioteca de python de imaplib: http://docs.python.org/library/imaplib.html
Smtpd-servidores SMTP de exemplo
Http://pymotw.com/2/smtpd/index.html
Smtplib-cliente de Protocolo de transferência de correio simples
Existe o servidor de SMTP em Python .
Este módulo oferece várias classes para implementar servidores SMTP. Um é um implementação genérica do nada, que pode ser sobreposta, enquanto o outros dois oferecem estratégias específicas de envio de correio.
O Twisted tem uma compilação do servidor de SMTP. Ver http://twistedmatrix.com/documents/current/mail/examples/index.html
# 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()
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()
Uma abordagem mais moderna é usar a bibliotecaaiosmtpd (documentação disponívelaqui ).
Você pode encontrar um bom exemplo aqui: https://aiosmtpd.readthedocs.io/en/latest/aiosmtpd/docs/controller.html.
Se quiser testar rapidamente a resposta de Django send_mail com a resposta de hasen acima:
# 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.',
'[email protected]',
['[email protected]'],
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: [email protected]
To: [email protected]
Date: Wed, 02 May 2018 02:12:09 -0000
Message-ID: <20180502021209.32641.51865@i1022>
Here is the message.
Não é necessário ter valores válidos na função send_mail. Acima dos valores funcionará muito bem com o exemplo de hasen.