1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
# Copyright (C) 2007 Canonical Ltd
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
import os
import subprocess
import tempfile
from bzrlib import (
email_message,
errors,
msgeditor,
urlutils,
)
class MailClient(object):
"""A mail client that can send messages with attachements."""
def __init__(self, config):
self.config = config
def compose(self, prompt, to, subject, attachment, mime_subtype,
extension):
raise NotImplementedError
def compose_merge_request(self, to, subject, directive):
self.compose("Please describe these changes:", to, subject, directive,
'x-patch', '.patch')
class Editor(MailClient):
"""DIY mail client that uses commit message editor"""
def compose(self, prompt, to, subject, attachment, mime_subtype,
extension):
info = ("%s\n\nTo: %s\nSubject: %s\n\n%s" % (prompt, to, subject,
attachment))
body = msgeditor.edit_commit_message(info)
if body == '':
raise errors.NoMessageSupplied()
email_message.EmailMessage.send(self.config,
self.config.username(),
to,
subject,
body,
attachment,
attachment_mime_subtype=mime_subtype)
class Thunderbird(MailClient):
"""Mozilla Thunderbird (or Icedove)
Note that Thunderbird 1.5 is buggy and does not support setting
"to" simultaneously with including a attachment.
There is a workaround if no attachment is present, but we always need to
send attachments.
"""
def compose(self, prompt, to, subject, attachment, mime_subtype,
extension):
fd, pathname = tempfile.mkstemp(extension, 'bzr-mail-')
try:
os.write(fd, attachment)
finally:
os.close(fd)
cmdline = ['thunderbird']
cmdline.extend(self._get_compose_commandline(to, subject, pathname))
subprocess.call(cmdline)
def _get_compose_commandline(self, to, subject, attach_path):
message_options = {}
if to is not None:
message_options['to'] = to
if subject is not None:
message_options['subject'] = subject
if attach_path is not None:
message_options['attachment'] = urlutils.local_path_to_url(
attach_path)
options_list = ["%s='%s'" % (k, v) for k, v in
sorted(message_options.iteritems())]
return ['-compose', ','.join(options_list)]
class Evolution(MailClient):
"""Evolution mail client."""
def compose(self, prompt, to, subject, attachment, mime_subtype,
extension):
fd, pathname = tempfile.mkstemp(extension, 'bzr-mail-')
try:
os.write(fd, attachment)
finally:
os.close(fd)
cmdline = ['evolution']
cmdline.append(self._get_compose_commandline(to, subject, pathname))
subprocess.call(cmdline)
def _get_compose_commandline(self, to, subject, attach_path):
message_options = {}
if subject is not None:
message_options['subject'] = subject
if attach_path is not None:
message_options['attach'] = attach_path
options_list = ['%s=%s' % (k, urlutils.escape(v)) for (k, v) in
message_options.iteritems()]
return 'mailto:%s?%s' % (to or '', '&'.join(options_list))
|