~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/mail_client.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2007-11-03 01:53:30 UTC
  • mfrom: (2955.1.1 trunk)
  • Revision ID: pqm@pqm.ubuntu.com-20071103015330-pt1tec7wyxwwcey8
Fix #158972 don't use timeout for HttpServer

Show diffs side-by-side

added added

removed removed

Lines of Context:
20
20
import sys
21
21
import tempfile
22
22
 
23
 
import bzrlib
24
23
from bzrlib import (
25
24
    email_message,
26
25
    errors,
27
26
    msgeditor,
28
27
    osutils,
29
28
    urlutils,
30
 
    registry
31
29
    )
32
30
 
33
 
mail_client_registry = registry.Registry()
34
 
 
35
31
 
36
32
class MailClient(object):
37
33
    """A mail client that can send messages with attachements."""
40
36
        self.config = config
41
37
 
42
38
    def compose(self, prompt, to, subject, attachment, mime_subtype,
43
 
                extension, basename=None):
 
39
                extension):
44
40
        """Compose (and possibly send) an email message
45
41
 
46
42
        Must be implemented by subclasses.
55
51
            "plain", "x-patch", etc.
56
52
        :param extension: The file extension associated with the attachment
57
53
            type, e.g. ".patch"
58
 
        :param basename: The name to use for the attachment, e.g.
59
 
            "send-nick-3252"
60
54
        """
61
55
        raise NotImplementedError
62
56
 
63
 
    def compose_merge_request(self, to, subject, directive, basename=None):
 
57
    def compose_merge_request(self, to, subject, directive):
64
58
        """Compose (and possibly send) a merge request
65
59
 
66
60
        :param to: The address to send the request to
67
61
        :param subject: The subject line to use for the request
68
62
        :param directive: A merge directive representing the merge request, as
69
63
            a bytestring.
70
 
        :param basename: The name to use for the attachment, e.g.
71
 
            "send-nick-3252"
72
64
        """
73
65
        prompt = self._get_merge_prompt("Please describe these changes:", to,
74
66
                                        subject, directive)
75
67
        self.compose(prompt, to, subject, directive,
76
 
            'x-patch', '.patch', basename)
 
68
            'x-patch', '.patch')
77
69
 
78
70
    def _get_merge_prompt(self, prompt, to, subject, attachment):
79
71
        """Generate a prompt string.  Overridden by Editor.
98
90
                         attachment.decode('utf-8', 'replace')))
99
91
 
100
92
    def compose(self, prompt, to, subject, attachment, mime_subtype,
101
 
                extension, basename=None):
 
93
                extension):
102
94
        """See MailClient.compose"""
103
 
        if not to:
104
 
            raise errors.NoMailAddressSpecified()
105
95
        body = msgeditor.edit_commit_message(prompt)
106
96
        if body == '':
107
97
            raise errors.NoMessageSupplied()
112
102
                                        body,
113
103
                                        attachment,
114
104
                                        attachment_mime_subtype=mime_subtype)
115
 
mail_client_registry.register('editor', Editor,
116
 
                              help=Editor.__doc__)
117
105
 
118
106
 
119
107
class ExternalMailClient(MailClient):
128
116
            return self._client_commands
129
117
 
130
118
    def compose(self, prompt, to, subject, attachment, mime_subtype,
131
 
                extension, basename=None):
 
119
                extension):
132
120
        """See MailClient.compose.
133
121
 
134
122
        Writes the attachment to a temporary file, invokes _compose.
135
123
        """
136
 
        if basename is None:
137
 
            basename = 'attachment'
138
 
        pathname = osutils.mkdtemp(prefix='bzr-mail-')
139
 
        attach_path = osutils.pathjoin(pathname, basename + extension)
140
 
        outfile = open(attach_path, 'wb')
 
124
        fd, pathname = tempfile.mkstemp(extension, 'bzr-mail-')
141
125
        try:
142
 
            outfile.write(attachment)
 
126
            os.write(fd, attachment)
143
127
        finally:
144
 
            outfile.close()
145
 
        self._compose(prompt, to, subject, attach_path, mime_subtype,
146
 
                      extension)
 
128
            os.close(fd)
 
129
        self._compose(prompt, to, subject, pathname, mime_subtype, extension)
147
130
 
148
131
    def _compose(self, prompt, to, subject, attach_path, mime_subtype,
149
132
                extension):
159
142
            the attachment type.
160
143
        """
161
144
        for name in self._get_client_commands():
162
 
            cmdline = [self._encode_path(name, 'executable')]
 
145
            cmdline = [name]
163
146
            cmdline.extend(self._get_compose_commandline(to, subject,
164
147
                                                         attach_path))
165
148
            try:
182
165
        """
183
166
        raise NotImplementedError
184
167
 
185
 
    def _encode_safe(self, u):
186
 
        """Encode possible unicode string argument to 8-bit string
187
 
        in user_encoding. Unencodable characters will be replaced
188
 
        with '?'.
189
 
 
190
 
        :param  u:  possible unicode string.
191
 
        :return:    encoded string if u is unicode, u itself otherwise.
192
 
        """
193
 
        if isinstance(u, unicode):
194
 
            return u.encode(bzrlib.user_encoding, 'replace')
195
 
        return u
196
 
 
197
 
    def _encode_path(self, path, kind):
198
 
        """Encode unicode path in user encoding.
199
 
 
200
 
        :param  path:   possible unicode path.
201
 
        :param  kind:   path kind ('executable' or 'attachment').
202
 
        :return:        encoded path if path is unicode,
203
 
                        path itself otherwise.
204
 
        :raise:         UnableEncodePath.
205
 
        """
206
 
        if isinstance(path, unicode):
207
 
            try:
208
 
                return path.encode(bzrlib.user_encoding)
209
 
            except UnicodeEncodeError:
210
 
                raise errors.UnableEncodePath(path, kind)
211
 
        return path
212
 
 
213
168
 
214
169
class Evolution(ExternalMailClient):
215
170
    """Evolution mail client."""
224
179
        if attach_path is not None:
225
180
            message_options['attach'] = attach_path
226
181
        options_list = ['%s=%s' % (k, urlutils.escape(v)) for (k, v) in
227
 
                        sorted(message_options.iteritems())]
228
 
        return ['mailto:%s?%s' % (self._encode_safe(to or ''),
229
 
            '&'.join(options_list))]
230
 
mail_client_registry.register('evolution', Evolution,
231
 
                              help=Evolution.__doc__)
 
182
                        message_options.iteritems()]
 
183
        return ['mailto:%s?%s' % (to or '', '&'.join(options_list))]
232
184
 
233
185
 
234
186
class Mutt(ExternalMailClient):
240
192
        """See ExternalMailClient._get_compose_commandline"""
241
193
        message_options = []
242
194
        if subject is not None:
243
 
            message_options.extend(['-s', self._encode_safe(subject)])
 
195
            message_options.extend(['-s', subject ])
244
196
        if attach_path is not None:
245
 
            message_options.extend(['-a',
246
 
                self._encode_path(attach_path, 'attachment')])
 
197
            message_options.extend(['-a', attach_path])
247
198
        if to is not None:
248
 
            message_options.append(self._encode_safe(to))
 
199
            message_options.append(to)
249
200
        return message_options
250
 
mail_client_registry.register('mutt', Mutt,
251
 
                              help=Mutt.__doc__)
252
201
 
253
202
 
254
203
class Thunderbird(ExternalMailClient):
262
211
    """
263
212
 
264
213
    _client_commands = ['thunderbird', 'mozilla-thunderbird', 'icedove',
265
 
        '/Applications/Mozilla/Thunderbird.app/Contents/MacOS/thunderbird-bin',
266
 
        '/Applications/Thunderbird.app/Contents/MacOS/thunderbird-bin']
 
214
        '/Applications/Mozilla/Thunderbird.app/Contents/MacOS/thunderbird-bin']
267
215
 
268
216
    def _get_compose_commandline(self, to, subject, attach_path):
269
217
        """See ExternalMailClient._get_compose_commandline"""
270
218
        message_options = {}
271
219
        if to is not None:
272
 
            message_options['to'] = self._encode_safe(to)
 
220
            message_options['to'] = to
273
221
        if subject is not None:
274
 
            message_options['subject'] = self._encode_safe(subject)
 
222
            message_options['subject'] = subject
275
223
        if attach_path is not None:
276
224
            message_options['attachment'] = urlutils.local_path_to_url(
277
225
                attach_path)
278
226
        options_list = ["%s='%s'" % (k, v) for k, v in
279
227
                        sorted(message_options.iteritems())]
280
228
        return ['-compose', ','.join(options_list)]
281
 
mail_client_registry.register('thunderbird', Thunderbird,
282
 
                              help=Thunderbird.__doc__)
283
229
 
284
230
 
285
231
class KMail(ExternalMailClient):
291
237
        """See ExternalMailClient._get_compose_commandline"""
292
238
        message_options = []
293
239
        if subject is not None:
294
 
            message_options.extend(['-s', self._encode_safe(subject)])
 
240
            message_options.extend( ['-s', subject ] )
295
241
        if attach_path is not None:
296
 
            message_options.extend(['--attach',
297
 
                self._encode_path(attach_path, 'attachment')])
 
242
            message_options.extend( ['--attach', attach_path] )
298
243
        if to is not None:
299
 
            message_options.extend([self._encode_safe(to)])
 
244
            message_options.extend( [ to ] )
 
245
 
300
246
        return message_options
301
 
mail_client_registry.register('kmail', KMail,
302
 
                              help=KMail.__doc__)
303
247
 
304
248
 
305
249
class XDGEmail(ExternalMailClient):
309
253
 
310
254
    def _get_compose_commandline(self, to, subject, attach_path):
311
255
        """See ExternalMailClient._get_compose_commandline"""
312
 
        if not to:
313
 
            raise errors.NoMailAddressSpecified()
314
 
        commandline = [self._encode_safe(to)]
315
 
        if subject is not None:
316
 
            commandline.extend(['--subject', self._encode_safe(subject)])
317
 
        if attach_path is not None:
318
 
            commandline.extend(['--attach',
319
 
                self._encode_path(attach_path, 'attachment')])
320
 
        return commandline
321
 
mail_client_registry.register('xdg-email', XDGEmail,
322
 
                              help=XDGEmail.__doc__)
323
 
 
324
 
 
325
 
class EmacsMail(ExternalMailClient):
326
 
    """Call emacsclient to have a mail buffer.
327
 
 
328
 
    This only work for emacs >= 22.1 due to recent -e/--eval support.
329
 
 
330
 
    The good news is that this implementation will work with all mail
331
 
    agents registered against ``mail-user-agent``. So there is no need
332
 
    to instantiate ExternalMailClient for each and every GNU Emacs
333
 
    MUA.
334
 
 
335
 
    Users just have to ensure that ``mail-user-agent`` is set according
336
 
    to their tastes.
337
 
    """
338
 
 
339
 
    _client_commands = ['emacsclient']
340
 
 
341
 
    def _prepare_send_function(self):
342
 
        """Write our wrapper function into a temporary file.
343
 
 
344
 
        This temporary file will be loaded at runtime in
345
 
        _get_compose_commandline function.
346
 
 
347
 
        This function does not remove the file.  That's a wanted
348
 
        behaviour since _get_compose_commandline won't run the send
349
 
        mail function directly but return the eligible command line.
350
 
        Removing our temporary file here would prevent our sendmail
351
 
        function to work.  (The file is deleted by some elisp code
352
 
        after being read by Emacs.)
353
 
        """
354
 
 
355
 
        _defun = r"""(defun bzr-add-mime-att (file)
356
 
  "Attach FILE to a mail buffer as a MIME attachment."
357
 
  (let ((agent mail-user-agent))
358
 
    (if (and file (file-exists-p file))
359
 
        (cond
360
 
         ((eq agent 'sendmail-user-agent)
361
 
          (progn
362
 
            (mail-text)
363
 
            (newline)
364
 
            (if (functionp 'etach-attach)
365
 
              (etach-attach file)
366
 
              (mail-attach-file file))))
367
 
         ((or (eq agent 'message-user-agent)(eq agent 'gnus-user-agent))
368
 
          (progn
369
 
            (mml-attach-file file "text/x-patch" "BZR merge" "inline")))
370
 
         ((eq agent 'mew-user-agent)
371
 
          (progn
372
 
            (mew-draft-prepare-attachments)
373
 
            (mew-attach-link file (file-name-nondirectory file))
374
 
            (let* ((nums (mew-syntax-nums))
375
 
                   (syntax (mew-syntax-get-entry mew-encode-syntax nums)))
376
 
              (mew-syntax-set-cd syntax "BZR merge")
377
 
              (mew-encode-syntax-print mew-encode-syntax))
378
 
            (mew-header-goto-body)))
379
 
         (t
380
 
          (message "Unhandled MUA, report it on bazaar@lists.canonical.com")))
381
 
      (error "File %s does not exist." file))))
382
 
"""
383
 
 
384
 
        fd, temp_file = tempfile.mkstemp(prefix="emacs-bzr-send-",
385
 
                                         suffix=".el")
386
 
        try:
387
 
            os.write(fd, _defun)
388
 
        finally:
389
 
            os.close(fd) # Just close the handle but do not remove the file.
390
 
        return temp_file
391
 
 
392
 
    def _get_compose_commandline(self, to, subject, attach_path):
393
 
        commandline = ["--eval"]
394
 
 
395
 
        _to = "nil"
396
 
        _subject = "nil"
397
 
 
398
 
        if to is not None:
399
 
            _to = ("\"%s\"" % self._encode_safe(to).replace('"', '\\"'))
400
 
        if subject is not None:
401
 
            _subject = ("\"%s\"" %
402
 
                        self._encode_safe(subject).replace('"', '\\"'))
403
 
 
404
 
        # Funcall the default mail composition function
405
 
        # This will work with any mail mode including default mail-mode
406
 
        # User must tweak mail-user-agent variable to tell what function
407
 
        # will be called inside compose-mail.
408
 
        mail_cmd = "(compose-mail %s %s)" % (_to, _subject)
409
 
        commandline.append(mail_cmd)
410
 
 
411
 
        # Try to attach a MIME attachment using our wrapper function
412
 
        if attach_path is not None:
413
 
            # Do not create a file if there is no attachment
414
 
            elisp = self._prepare_send_function()
415
 
            lmmform = '(load "%s")' % elisp
416
 
            mmform  = '(bzr-add-mime-att "%s")' % \
417
 
                self._encode_path(attach_path, 'attachment')
418
 
            rmform = '(delete-file "%s")' % elisp
419
 
            commandline.append(lmmform)
420
 
            commandline.append(mmform)
421
 
            commandline.append(rmform)
422
 
 
423
 
        return commandline
424
 
mail_client_registry.register('emacsclient', EmacsMail,
425
 
                              help=EmacsMail.__doc__)
 
256
        commandline = [to]
 
257
        if subject is not None:
 
258
            commandline.extend(['--subject', subject])
 
259
        if attach_path is not None:
 
260
            commandline.extend(['--attach', attach_path])
 
261
        return commandline
426
262
 
427
263
 
428
264
class MAPIClient(ExternalMailClient):
441
277
            if e.code != simplemapi.MAPI_USER_ABORT:
442
278
                raise errors.MailClientNotFound(['MAPI supported mail client'
443
279
                                                 ' (error %d)' % (e.code,)])
444
 
mail_client_registry.register('mapi', MAPIClient,
445
 
                              help=MAPIClient.__doc__)
446
280
 
447
281
 
448
282
class DefaultMail(MailClient):
457
291
            return XDGEmail(self.config)
458
292
 
459
293
    def compose(self, prompt, to, subject, attachment, mime_subtype,
460
 
                extension, basename=None):
 
294
                extension):
461
295
        """See MailClient.compose"""
462
296
        try:
463
297
            return self._mail_client().compose(prompt, to, subject,
464
298
                                               attachment, mimie_subtype,
465
 
                                               extension, basename)
 
299
                                               extension)
466
300
        except errors.MailClientNotFound:
467
301
            return Editor(self.config).compose(prompt, to, subject,
468
302
                          attachment, mimie_subtype, extension)
469
303
 
470
 
    def compose_merge_request(self, to, subject, directive, basename=None):
 
304
    def compose_merge_request(self, to, subject, directive):
471
305
        """See MailClient.compose_merge_request"""
472
306
        try:
473
307
            return self._mail_client().compose_merge_request(to, subject,
474
 
                    directive, basename=basename)
 
308
                                                             directive)
475
309
        except errors.MailClientNotFound:
476
310
            return Editor(self.config).compose_merge_request(to, subject,
477
 
                          directive, basename=basename)
478
 
mail_client_registry.register('default', DefaultMail,
479
 
                              help=DefaultMail.__doc__)
480
 
mail_client_registry.default_key = 'default'
 
311
                          directive)