##// END OF EJS Templates
merges for stable
merges for stable

File last commit:

r1231:9f656066 default
r1232:0dc8d578 default
Show More
smtp_mailer.py
143 lines | 5.0 KiB | text/x-python | PythonLexer
rhodecode release 1.1.3 changes
r1057 # -*- coding: utf-8 -*-
"""
rhodecode.lib.smtp_mailer
~~~~~~~~~~~~~~~~~~~~~~~~~
fixes for stable
r1231
rhodecode release 1.1.3 changes
r1057 Simple smtp mailer used in RhodeCode
fixes for stable
r1231
rhodecode release 1.1.3 changes
r1057 :created_on: Sep 13, 2010
:copyright: (c) 2011 by marcink.
:license: LICENSE_NAME, see LICENSE_FILE for more details.
"""
renamed project to rhodecode
r547 import logging
import smtplib
import mimetypes
rhodecode release 1.1.3 changes
r1057 from socket import sslerror
renamed project to rhodecode
r547 from email.mime.multipart import MIMEMultipart
from email.mime.image import MIMEImage
from email.mime.audio import MIMEAudio
from email.mime.base import MIMEBase
from email.mime.text import MIMEText
from email.utils import formatdate
from email import encoders
class SmtpMailer(object):
rhodecode release 1.1.3 changes
r1057 """SMTP mailer class
fixes for stable
r1231
renamed project to rhodecode
r547 mailer = SmtpMailer(mail_from, user, passwd, mail_server, mail_port, ssl, tls)
fixes for stable
r1231 mailer.send(recipients, subject, body, attachment_files)
renamed project to rhodecode
r547 :param recipients might be a list of string or single string
fixes for stable
r1231 :param attachment_files is a dict of {filename:location}
it tries to guess the mimetype and attach the file
renamed project to rhodecode
r547 """
def __init__(self, mail_from, user, passwd, mail_server,
fixes for stable
r1231 mail_port=None, ssl=False, tls=False, debug=False):
fixes #59, notifications for user registrations + some changes to mailer
r689
renamed project to rhodecode
r547 self.mail_from = mail_from
self.mail_server = mail_server
self.mail_port = mail_port
self.user = user
self.passwd = passwd
self.ssl = ssl
self.tls = tls
fixes for stable
r1231 self.debug = debug
fixes #59, notifications for user registrations + some changes to mailer
r689
rhodecode release 1.1.3 changes
r1057 def send(self, recipients=[], subject='', body='', attachment_files=None):
renamed project to rhodecode
r547
if isinstance(recipients, basestring):
recipients = [recipients]
if self.ssl:
smtp_serv = smtplib.SMTP_SSL(self.mail_server, self.mail_port)
else:
smtp_serv = smtplib.SMTP(self.mail_server, self.mail_port)
if self.tls:
rhodecode release 1.1.3 changes
r1057 smtp_serv.ehlo()
renamed project to rhodecode
r547 smtp_serv.starttls()
fixes #59, notifications for user registrations + some changes to mailer
r689
if self.debug:
renamed project to rhodecode
r547 smtp_serv.set_debuglevel(1)
rhodecode release 1.1.3 changes
r1057 smtp_serv.ehlo()
renamed project to rhodecode
r547
#if server requires authorization you must provide login and password
rhodecode release 1.1.3 changes
r1057 #but only if we have them
if self.user and self.passwd:
smtp_serv.login(self.user, self.passwd)
renamed project to rhodecode
r547
date_ = formatdate(localtime=True)
msg = MIMEMultipart()
msg['From'] = self.mail_from
msg['To'] = ','.join(recipients)
msg['Date'] = date_
msg['Subject'] = subject
msg.preamble = 'You will not see this in a MIME-aware mail reader.\n'
msg.attach(MIMEText(body))
if attachment_files:
self.__atach_files(msg, attachment_files)
smtp_serv.sendmail(self.mail_from, recipients, msg.as_string())
logging.info('MAIL SEND TO: %s' % recipients)
rhodecode release 1.1.3 changes
r1057
try:
smtp_serv.quit()
except sslerror:
# sslerror is raised in tls connections on closing sometimes
pass
renamed project to rhodecode
r547
def __atach_files(self, msg, attachment_files):
if isinstance(attachment_files, dict):
for f_name, msg_file in attachment_files.items():
ctype, encoding = mimetypes.guess_type(f_name)
logging.info("guessing file %s type based on %s" , ctype, f_name)
if ctype is None or encoding is not None:
# No guess could be made, or the file is encoded (compressed), so
# use a generic bag-of-bits type.
ctype = 'application/octet-stream'
maintype, subtype = ctype.split('/', 1)
if maintype == 'text':
# Note: we should handle calculating the charset
fixes #59, notifications for user registrations + some changes to mailer
r689 file_part = MIMEText(self.get_content(msg_file),
renamed project to rhodecode
r547 _subtype=subtype)
elif maintype == 'image':
fixes #59, notifications for user registrations + some changes to mailer
r689 file_part = MIMEImage(self.get_content(msg_file),
renamed project to rhodecode
r547 _subtype=subtype)
elif maintype == 'audio':
fixes #59, notifications for user registrations + some changes to mailer
r689 file_part = MIMEAudio(self.get_content(msg_file),
renamed project to rhodecode
r547 _subtype=subtype)
else:
file_part = MIMEBase(maintype, subtype)
file_part.set_payload(self.get_content(msg_file))
# Encode the payload using Base64
encoders.encode_base64(msg)
# Set the filename parameter
fixes #59, notifications for user registrations + some changes to mailer
r689 file_part.add_header('Content-Disposition', 'attachment',
renamed project to rhodecode
r547 filename=f_name)
file_part.add_header('Content-Type', ctype, name=f_name)
msg.attach(file_part)
else:
fixes #59, notifications for user registrations + some changes to mailer
r689 raise Exception('Attachment files should be'
'a dict in format {"filename":"filepath"}')
renamed project to rhodecode
r547
def get_content(self, msg_file):
rhodecode release 1.1.3 changes
r1057 """Get content based on type, if content is a string do open first
renamed project to rhodecode
r547 else just read because it's a probably open file object
fixes for stable
r1231
fixed @repo into :repo for docs...
r604 :param msg_file:
rhodecode release 1.1.3 changes
r1057 """
renamed project to rhodecode
r547 if isinstance(msg_file, str):
return open(msg_file, "rb").read()
else:
#just for safe seek to 0
msg_file.seek(0)
return msg_file.read()