| 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
 | # -*- test-case-name: twisted.mail.test.test_bounce -*-
#
# Copyright (c) Twisted Matrix Laboratories.
# See LICENSE for details.
"""
Support for bounce message generation.
"""
import email.utils
import time
import os
from io import StringIO, SEEK_SET, SEEK_END
from twisted.mail import smtp
BOUNCE_FORMAT = u"""\
From: postmaster@{failedDomain}
To: {failedFrom}
Subject: Returned Mail: see transcript for details
Message-ID: {messageID}
Content-Type: multipart/report; report-type=delivery-status;
    boundary="{boundary}"
--{boundary}
{transcript}
--{boundary}
Content-Type: message/delivery-status
Arrival-Date: {ctime}
Final-Recipient: RFC822; {failedTo}
"""
def generateBounce(message, failedFrom, failedTo, transcript='',
                   encoding='utf-8'):
    """
    Generate a bounce message for an undeliverable email message.
    @type message: a file-like object
    @param message: The undeliverable message.
    @type failedFrom: L{bytes} or L{unicode}
    @param failedFrom: The originator of the undeliverable message.
    @type failedTo: L{bytes} or L{unicode}
    @param failedTo: The destination of the undeliverable message.
    @type transcript: L{bytes} or L{unicode}
    @param transcript: An error message to include in the bounce message.
    @type encoding: L{str} or L{unicode}
    @param encoding: Encoding to use, default: utf-8
    @rtype: 3-L{tuple} of (E{1}) L{bytes}, (E{2}) L{bytes}, (E{3}) L{bytes}
    @return: The originator, the destination and the contents of the bounce
        message.  The destination of the bounce message is the originator of
        the undeliverable message.
    """
    if isinstance(failedFrom, bytes):
        failedFrom = failedFrom.decode(encoding)
    if isinstance(failedTo, bytes):
        failedTo = failedTo.decode(encoding)
    if not transcript:
        transcript = u'''\
I'm sorry, the following address has permanent errors: {failedTo}.
I've given up, and I will not retry the message again.
'''.format(failedTo=failedTo)
    failedAddress = email.utils.parseaddr(failedTo)[1]
    data = {
        'boundary': "{}_{}_{}".format(time.time(), os.getpid(), 'XXXXX'),
        'ctime': time.ctime(time.time()),
        'failedAddress': failedAddress,
        'failedDomain': failedAddress.split('@', 1)[1],
        'failedFrom': failedFrom,
        'failedTo': failedTo,
        'messageID': smtp.messageid(uniq='bounce'),
        'message': message,
        'transcript': transcript,
        }
    fp = StringIO()
    fp.write(BOUNCE_FORMAT.format(**data))
    orig = message.tell()
    message.seek(0, SEEK_END)
    sz = message.tell()
    message.seek(orig, SEEK_SET)
    if sz > 10000:
        while 1:
            line = message.readline()
            if isinstance(line, bytes):
                line = line.decode(encoding)
            if len(line) <= 0:
                break
            fp.write(line)
    else:
        messageContent = message.read()
        if isinstance(messageContent, bytes):
            messageContent = messageContent.decode(encoding)
        fp.write(messageContent)
    return b'', failedFrom.encode(encoding), fp.getvalue().encode(encoding)
 |