bzr branch
http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
1  | 
# Copyright (C) 2007 Canonical Ltd
 | 
2  | 
#
 | 
|
3  | 
# This program is free software; you can redistribute it and/or modify
 | 
|
4  | 
# it under the terms of the GNU General Public License as published by
 | 
|
5  | 
# the Free Software Foundation; either version 2 of the License, or
 | 
|
6  | 
# (at your option) any later version.
 | 
|
7  | 
#
 | 
|
8  | 
# This program is distributed in the hope that it will be useful,
 | 
|
9  | 
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
|
10  | 
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
|
11  | 
# GNU General Public License for more details.
 | 
|
12  | 
#
 | 
|
13  | 
# You should have received a copy of the GNU General Public License
 | 
|
14  | 
# along with this program; if not, write to the Free Software
 | 
|
| 
4183.7.1
by Sabin Iacob
 update FSF mailing address  | 
15  | 
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
16  | 
|
17  | 
"""A convenience class around smtplib."""
 | 
|
18  | 
||
| 
2535.2.3
by Adeodato Simó
 Import full email.Utils module instead of individual functions, as per  | 
19  | 
from email import Utils  | 
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
20  | 
import errno  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
21  | 
import smtplib  | 
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
22  | 
import socket  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
23  | 
|
| 
2900.2.11
by Vincent Ladeuil
 Make smtp aware of authentication config.  | 
24  | 
from bzrlib import (  | 
25  | 
config,  | 
|
| 
4147.1.3
by James Henstridge
 Switch to using osutils.safe_utf8() as suggested by Vincent. Also  | 
26  | 
osutils,  | 
| 
2900.2.11
by Vincent Ladeuil
 Make smtp aware of authentication config.  | 
27  | 
    )
 | 
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
28  | 
from bzrlib.errors import (  | 
29  | 
NoDestinationAddress,  | 
|
30  | 
SMTPError,  | 
|
31  | 
DefaultSMTPConnectionRefused,  | 
|
32  | 
SMTPConnectionRefused,  | 
|
33  | 
    )
 | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
34  | 
|
35  | 
||
36  | 
class SMTPConnection(object):  | 
|
37  | 
"""Connect to an SMTP server and send an email.  | 
|
38  | 
||
39  | 
    This is a gateway between bzrlib.config.Config and smtplib.SMTP. It
 | 
|
40  | 
    understands the basic bzr SMTP configuration information: smtp_server,
 | 
|
41  | 
    smtp_username, and smtp_password.
 | 
|
42  | 
    """
 | 
|
43  | 
||
44  | 
_default_smtp_server = 'localhost'  | 
|
45  | 
||
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
46  | 
def __init__(self, config, _smtp_factory=None):  | 
47  | 
self._smtp_factory = _smtp_factory  | 
|
48  | 
if self._smtp_factory is None:  | 
|
49  | 
self._smtp_factory = smtplib.SMTP  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
50  | 
self._config = config  | 
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
51  | 
self._config_smtp_server = config.get_user_option('smtp_server')  | 
52  | 
self._smtp_server = self._config_smtp_server  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
53  | 
if self._smtp_server is None:  | 
54  | 
self._smtp_server = self._default_smtp_server  | 
|
55  | 
||
56  | 
self._smtp_username = config.get_user_option('smtp_username')  | 
|
57  | 
self._smtp_password = config.get_user_option('smtp_password')  | 
|
58  | 
||
59  | 
self._connection = None  | 
|
60  | 
||
61  | 
def _connect(self):  | 
|
62  | 
"""If we haven't connected, connect and authenticate."""  | 
|
63  | 
if self._connection is not None:  | 
|
64  | 
            return
 | 
|
65  | 
||
66  | 
self._create_connection()  | 
|
| 
4222.3.10
by Jelmer Vernooij
 Avoid using the default username in the case of SMTP.  | 
67  | 
        # FIXME: _authenticate() should only be called when the server has
 | 
68  | 
        # refused unauthenticated access, so it can safely try to authenticate 
 | 
|
69  | 
        # with the default username. JRV20090407
 | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
70  | 
self._authenticate()  | 
71  | 
||
72  | 
def _create_connection(self):  | 
|
73  | 
"""Create an SMTP connection."""  | 
|
| 
2694.2.1
by Aaron Bentley
 Make error handling nicer when SMTP server not working  | 
74  | 
self._connection = self._smtp_factory()  | 
75  | 
try:  | 
|
76  | 
self._connection.connect(self._smtp_server)  | 
|
77  | 
except socket.error, e:  | 
|
78  | 
if e.args[0] == errno.ECONNREFUSED:  | 
|
79  | 
if self._config_smtp_server is None:  | 
|
80  | 
raise DefaultSMTPConnectionRefused(socket.error,  | 
|
81  | 
self._smtp_server)  | 
|
82  | 
else:  | 
|
83  | 
raise SMTPConnectionRefused(socket.error,  | 
|
84  | 
self._smtp_server)  | 
|
85  | 
else:  | 
|
86  | 
                raise
 | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
87  | 
|
| 
2898.2.3
by James Henstridge
 * Fix merge-directive blackbox test.  | 
88  | 
        # Say EHLO (falling back to HELO) to query the server's features.
 | 
| 
2898.2.1
by James Henstridge
 Update SMTPConnection._create_connection to better follow the SMTP  | 
89  | 
code, resp = self._connection.ehlo()  | 
90  | 
if not (200 <= code <= 299):  | 
|
91  | 
code, resp = self._connection.helo()  | 
|
92  | 
if not (200 <= code <= 299):  | 
|
93  | 
raise SMTPError("server refused HELO: %d %s" % (code, resp))  | 
|
94  | 
||
95  | 
        # Use TLS if the server advertised it:
 | 
|
96  | 
if self._connection.has_extn("starttls"):  | 
|
97  | 
code, resp = self._connection.starttls()  | 
|
98  | 
if not (200 <= code <= 299):  | 
|
99  | 
raise SMTPError("server refused STARTTLS: %d %s" % (code, resp))  | 
|
100  | 
            # Say EHLO again, to check for newly revealed features
 | 
|
101  | 
code, resp = self._connection.ehlo()  | 
|
102  | 
if not (200 <= code <= 299):  | 
|
103  | 
raise SMTPError("server refused EHLO: %d %s" % (code, resp))  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
104  | 
|
105  | 
def _authenticate(self):  | 
|
106  | 
"""If necessary authenticate yourself to the server."""  | 
|
| 
2900.2.15
by Vincent Ladeuil
 AuthenticationConfig can be queried for logins too (first step).  | 
107  | 
auth = config.AuthenticationConfig()  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
108  | 
if self._smtp_username is None:  | 
| 
4222.3.10
by Jelmer Vernooij
 Avoid using the default username in the case of SMTP.  | 
109  | 
            # FIXME: Since _authenticate gets called even when no authentication
 | 
110  | 
            # is necessary, it's not possible to use the default username 
 | 
|
111  | 
            # here yet.
 | 
|
| 
4304.2.1
by Vincent Ladeuil
 Fix bug #367726 by reverting some default user handling introduced  | 
112  | 
self._smtp_username = auth.get_user('smtp', self._smtp_server)  | 
113  | 
if self._smtp_username is None:  | 
|
| 
2900.2.15
by Vincent Ladeuil
 AuthenticationConfig can be queried for logins too (first step).  | 
114  | 
                return
 | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
115  | 
|
| 
2900.2.12
by Vincent Ladeuil
 Since all schemes query AuthenticationConfig then prompt user, make that  | 
116  | 
if self._smtp_password is None:  | 
117  | 
self._smtp_password = auth.get_password(  | 
|
118  | 
'smtp', self._smtp_server, self._smtp_username)  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
119  | 
|
| 
4147.1.1
by James Henstridge
 Ensure that byte strings are passed to SMTP.login(), as passing unicode  | 
120  | 
        # smtplib requires that the username and password be byte
 | 
| 
4147.1.2
by James Henstridge
 Encode usernames and passwords as UTF-8 rather than ASCII. While  | 
121  | 
        # strings.  The CRAM-MD5 spec doesn't give any guidance on
 | 
| 
4147.1.3
by James Henstridge
 Switch to using osutils.safe_utf8() as suggested by Vincent. Also  | 
122  | 
        # encodings, but the SASL PLAIN spec says UTF-8, so that's
 | 
123  | 
        # what we'll use.
 | 
|
124  | 
username = osutils.safe_utf8(self._smtp_username)  | 
|
125  | 
password = osutils.safe_utf8(self._smtp_password)  | 
|
| 
4147.1.1
by James Henstridge
 Ensure that byte strings are passed to SMTP.login(), as passing unicode  | 
126  | 
|
| 
4147.1.3
by James Henstridge
 Switch to using osutils.safe_utf8() as suggested by Vincent. Also  | 
127  | 
self._connection.login(username, password)  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
128  | 
|
129  | 
    @staticmethod
 | 
|
130  | 
def get_message_addresses(message):  | 
|
131  | 
"""Get the origin and destination addresses of a message.  | 
|
132  | 
||
| 
2625.6.1
by Adeodato Simó
 New EmailMessage class, façade around email.Message and MIMEMultipart.  | 
133  | 
        :param message: A message object supporting get() to access its
 | 
134  | 
            headers, like email.Message or bzrlib.email_message.EmailMessage.
 | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
135  | 
        :return: A pair (from_email, to_emails), where from_email is the email
 | 
136  | 
            address in the From header, and to_emails a list of all the
 | 
|
137  | 
            addresses in the To, Cc, and Bcc headers.
 | 
|
138  | 
        """
 | 
|
| 
2625.6.1
by Adeodato Simó
 New EmailMessage class, façade around email.Message and MIMEMultipart.  | 
139  | 
from_email = Utils.parseaddr(message.get('From', None))[1]  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
140  | 
to_full_addresses = []  | 
141  | 
for header in ['To', 'Cc', 'Bcc']:  | 
|
| 
2625.6.1
by Adeodato Simó
 New EmailMessage class, façade around email.Message and MIMEMultipart.  | 
142  | 
value = message.get(header, None)  | 
143  | 
if value:  | 
|
144  | 
to_full_addresses.append(value)  | 
|
| 
2535.2.3
by Adeodato Simó
 Import full email.Utils module instead of individual functions, as per  | 
145  | 
to_emails = [ pair[1] for pair in  | 
146  | 
Utils.getaddresses(to_full_addresses) ]  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
147  | 
|
148  | 
return from_email, to_emails  | 
|
149  | 
||
150  | 
def send_email(self, message):  | 
|
151  | 
"""Send an email message.  | 
|
152  | 
||
| 
2547.1.1
by Aaron Bentley
 Add SMTPConnection class (Adeodato Simó)  | 
153  | 
        The message will be sent to all addresses in the To, Cc and Bcc
 | 
154  | 
        headers.
 | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
155  | 
|
156  | 
        :param message: An email.Message or email.MIMEMultipart object.
 | 
|
157  | 
        :return: None
 | 
|
158  | 
        """
 | 
|
159  | 
from_email, to_emails = self.get_message_addresses(message)  | 
|
160  | 
||
161  | 
if not to_emails:  | 
|
162  | 
raise NoDestinationAddress  | 
|
163  | 
||
164  | 
try:  | 
|
165  | 
self._connect()  | 
|
| 
2547.1.1
by Aaron Bentley
 Add SMTPConnection class (Adeodato Simó)  | 
166  | 
self._connection.sendmail(from_email, to_emails,  | 
167  | 
message.as_string())  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
168  | 
except smtplib.SMTPRecipientsRefused, e:  | 
| 
2535.2.4
by Adeodato Simó
 Don't use BzrCommandError in non-UI code; create and use an SMTPError  | 
169  | 
raise SMTPError('server refused recipient: %d %s' %  | 
170  | 
e.recipients.values()[0])  | 
|
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
171  | 
except smtplib.SMTPResponseException, e:  | 
| 
2535.2.4
by Adeodato Simó
 Don't use BzrCommandError in non-UI code; create and use an SMTPError  | 
172  | 
raise SMTPError('%d %s' % (e.smtp_code, e.smtp_error))  | 
| 
2535.2.1
by Adeodato Simó
 New SMTPConnection class, a reduced version of that in bzr-email.  | 
173  | 
except smtplib.SMTPException, e:  | 
| 
2535.2.4
by Adeodato Simó
 Don't use BzrCommandError in non-UI code; create and use an SMTPError  | 
174  | 
raise SMTPError(str(e))  |