Files @ 523b1011a625
Branch filter:

Location: kallithea/rhodecode/lib/rcmail/response.py - annotation

Marcin Kuzminski
- fixed issue with mark all read button for notifications and listeners on delete buttons
- hide mark all read if notification inbox is empty
  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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
7ff304d3028f
# The code in this module is entirely lifted from the Lamson project
# (http://lamsonproject.org/).  Its copyright is:

# Copyright (c) 2008, Zed A. Shaw
# All rights reserved.

# It is provided under this license:

# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:

# * Redistributions of source code must retain the above copyright notice, this
#   list of conditions and the following disclaimer.

# * Redistributions in binary form must reproduce the above copyright notice,
#   this list of conditions and the following disclaimer in the documentation
#   and/or other materials provided with the distribution.

# * Neither the name of the Zed A. Shaw nor the names of its contributors may
#   be used to endorse or promote products derived from this software without
#   specific prior written permission.

# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
# COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
# INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
# STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.

import os
import mimetypes
import string
from email import encoders
from email.charset import Charset
from email.utils import parseaddr
from email.mime.base import MIMEBase

ADDRESS_HEADERS_WHITELIST = ['From', 'To', 'Delivered-To', 'Cc', 'Bcc']
DEFAULT_ENCODING = "utf-8"
VALUE_IS_EMAIL_ADDRESS = lambda v: '@' in v

def normalize_header(header):
    return string.capwords(header.lower(), '-')

class EncodingError(Exception): 
    """Thrown when there is an encoding error."""
    pass

class MailBase(object):
    """MailBase is used as the basis of lamson.mail and contains the basics of
    encoding an email.  You actually can do all your email processing with this
    class, but it's more raw.
    """
    def __init__(self, items=()):
        self.headers = dict(items)
        self.parts = []
        self.body = None
        self.content_encoding = {'Content-Type': (None, {}), 
                                 'Content-Disposition': (None, {}),
                                 'Content-Transfer-Encoding': (None, {})}

    def __getitem__(self, key):
        return self.headers.get(normalize_header(key), None)

    def __len__(self):
        return len(self.headers)

    def __iter__(self):
        return iter(self.headers)

    def __contains__(self, key):
        return normalize_header(key) in self.headers

    def __setitem__(self, key, value):
        self.headers[normalize_header(key)] = value

    def __delitem__(self, key):
        del self.headers[normalize_header(key)]

    def __nonzero__(self):
        return self.body != None or len(self.headers) > 0 or len(self.parts) > 0

    def keys(self):
        """Returns the sorted keys."""
        return sorted(self.headers.keys())

    def attach_file(self, filename, data, ctype, disposition):
        """
        A file attachment is a raw attachment with a disposition that
        indicates the file name.
        """
        assert filename, "You can't attach a file without a filename."
        ctype = ctype.lower()

        part = MailBase()
        part.body = data
        part.content_encoding['Content-Type'] = (ctype, {'name': filename})
        part.content_encoding['Content-Disposition'] = (disposition,
                                                        {'filename': filename})
        self.parts.append(part)


    def attach_text(self, data, ctype):
        """
        This attaches a simpler text encoded part, which doesn't have a
        filename.
        """
        ctype = ctype.lower()

        part = MailBase()
        part.body = data
        part.content_encoding['Content-Type'] = (ctype, {})
        self.parts.append(part)

    def walk(self):
        for p in self.parts:
            yield p
            for x in p.walk():
                yield x

class MailResponse(object):
    """
    You are given MailResponse objects from the lamson.view methods, and
    whenever you want to generate an email to send to someone.  It has the
    same basic functionality as MailRequest, but it is designed to be written
    to, rather than read from (although you can do both).

    You can easily set a Body or Html during creation or after by passing it
    as __init__ parameters, or by setting those attributes.

    You can initially set the From, To, and Subject, but they are headers so
    use the dict notation to change them: msg['From'] = 'joe@test.com'.

    The message is not fully crafted until right when you convert it with
    MailResponse.to_message.  This lets you change it and work with it, then
    send it out when it's ready.
    """
    def __init__(self, To=None, From=None, Subject=None, Body=None, Html=None):
        self.Body = Body
        self.Html = Html
        self.base = MailBase([('To', To), ('From', From), ('Subject', Subject)])
        self.multipart = self.Body and self.Html
        self.attachments = []

    def __contains__(self, key):
        return self.base.__contains__(key)

    def __getitem__(self, key):
        return self.base.__getitem__(key)

    def __setitem__(self, key, val):
        return self.base.__setitem__(key, val)

    def __delitem__(self, name):
        del self.base[name]

    def attach(self, filename=None, content_type=None, data=None,
               disposition=None):
        """

        Simplifies attaching files from disk or data as files.  To attach
        simple text simple give data and a content_type.  To attach a file,
        give the data/content_type/filename/disposition combination.

        For convenience, if you don't give data and only a filename, then it
        will read that file's contents when you call to_message() later.  If
        you give data and filename then it will assume you've filled data
        with what the file's contents are and filename is just the name to
        use.
        """

        assert filename or data, ("You must give a filename or some data to "
                                  "attach.")
        assert data or os.path.exists(filename), ("File doesn't exist, and no "
                                                  "data given.")

        self.multipart = True

        if filename and not content_type:
            content_type, encoding = mimetypes.guess_type(filename)

        assert content_type, ("No content type given, and couldn't guess "
                              "from the filename: %r" % filename)

        self.attachments.append({'filename': filename,
                                 'content_type': content_type,
                                 'data': data,
                                 'disposition': disposition,})
    def attach_part(self, part):
        """
        Attaches a raw MailBase part from a MailRequest (or anywhere)
        so that you can copy it over.
        """
        self.multipart = True

        self.attachments.append({'filename': None,
                                 'content_type': None,
                                 'data': None,
                                 'disposition': None,
                                 'part': part,
                                 })

    def attach_all_parts(self, mail_request):
        """
        Used for copying the attachment parts of a mail.MailRequest
        object for mailing lists that need to maintain attachments.
        """
        for part in mail_request.all_parts():
            self.attach_part(part)

        self.base.content_encoding = mail_request.base.content_encoding.copy()

    def clear(self):
        """
        Clears out the attachments so you can redo them.  Use this to keep the
        headers for a series of different messages with different attachments.
        """
        del self.attachments[:]
        del self.base.parts[:]
        self.multipart = False


    def update(self, message):
        """
        Used to easily set a bunch of heading from another dict
        like object.
        """
        for k in message.keys():
            self.base[k] = message[k]

    def __str__(self):
        """
        Converts to a string.
        """
        return self.to_message().as_string()

    def _encode_attachment(self, filename=None, content_type=None, data=None,
                           disposition=None, part=None):
        """
        Used internally to take the attachments mentioned in self.attachments
        and do the actual encoding in a lazy way when you call to_message.
        """
        if part:
            self.base.parts.append(part)
        elif filename:
            if not data:
                data = open(filename).read()

            self.base.attach_file(filename, data, content_type,
                                  disposition or 'attachment')
        else:
            self.base.attach_text(data, content_type)

        ctype = self.base.content_encoding['Content-Type'][0]

        if ctype and not ctype.startswith('multipart'):
            self.base.content_encoding['Content-Type'] = ('multipart/mixed', {})

    def to_message(self):
        """
        Figures out all the required steps to finally craft the
        message you need and return it.  The resulting message
        is also available as a self.base attribute.

        What is returned is a Python email API message you can
        use with those APIs.  The self.base attribute is the raw
        lamson.encoding.MailBase.
        """
        del self.base.parts[:]

        if self.Body and self.Html:
            self.multipart = True
            self.base.content_encoding['Content-Type'] = (
                'multipart/alternative', {})

        if self.multipart:
            self.base.body = None
            if self.Body:
                self.base.attach_text(self.Body, 'text/plain')

            if self.Html:
                self.base.attach_text(self.Html, 'text/html')

            for args in self.attachments:
                self._encode_attachment(**args)

        elif self.Body:
            self.base.body = self.Body
            self.base.content_encoding['Content-Type'] = ('text/plain', {})

        elif self.Html:
            self.base.body = self.Html
            self.base.content_encoding['Content-Type'] = ('text/html', {})

        return to_message(self.base)

    def all_parts(self):
        """
        Returns all the encoded parts.  Only useful for debugging
        or inspecting after calling to_message().
        """
        return self.base.parts

    def keys(self):
        return self.base.keys()

def to_message(mail):
    """
    Given a MailBase message, this will construct a MIMEPart 
    that is canonicalized for use with the Python email API.
    """
    ctype, params = mail.content_encoding['Content-Type']

    if not ctype:
        if mail.parts:
            ctype = 'multipart/mixed'
        else:
            ctype = 'text/plain'
    else:
        if mail.parts:
            assert ctype.startswith(("multipart", "message")), \
                   "Content type should be multipart or message, not %r" % ctype

    # adjust the content type according to what it should be now
    mail.content_encoding['Content-Type'] = (ctype, params)

    try:
        out = MIMEPart(ctype, **params)
    except TypeError, exc: # pragma: no cover
        raise EncodingError("Content-Type malformed, not allowed: %r; "
                            "%r (Python ERROR: %s" %
                            (ctype, params, exc.message))

    for k in mail.keys():
        if k in ADDRESS_HEADERS_WHITELIST:
            out[k.encode('ascii')] = header_to_mime_encoding(mail[k])
        else:
            out[k.encode('ascii')] = header_to_mime_encoding(mail[k],
                                                             not_email=True)

    out.extract_payload(mail)

    # go through the children
    for part in mail.parts:
        out.attach(to_message(part))

    return out

class MIMEPart(MIMEBase):
    """
    A reimplementation of nearly everything in email.mime to be more useful
    for actually attaching things.  Rather than one class for every type of
    thing you'd encode, there's just this one, and it figures out how to
    encode what you ask it.
    """
    def __init__(self, type, **params):
        self.maintype, self.subtype = type.split('/')
        MIMEBase.__init__(self, self.maintype, self.subtype, **params)

    def add_text(self, content):
        # this is text, so encode it in canonical form
        try:
            encoded = content.encode('ascii')
            charset = 'ascii'
        except UnicodeError:
            encoded = content.encode('utf-8')
            charset = 'utf-8'

        self.set_payload(encoded, charset=charset)


    def extract_payload(self, mail):
        if mail.body == None: return  # only None, '' is still ok

        ctype, ctype_params = mail.content_encoding['Content-Type']
        cdisp, cdisp_params = mail.content_encoding['Content-Disposition']

        assert ctype, ("Extract payload requires that mail.content_encoding "
                       "have a valid Content-Type.")

        if ctype.startswith("text/"):
            self.add_text(mail.body)
        else:
            if cdisp:
                # replicate the content-disposition settings
                self.add_header('Content-Disposition', cdisp, **cdisp_params)

            self.set_payload(mail.body)
            encoders.encode_base64(self)

    def __repr__(self):
        return "<MIMEPart '%s/%s': %r, %r, multipart=%r>" % (
            self.subtype,
            self.maintype,
            self['Content-Type'],
            self['Content-Disposition'],
            self.is_multipart())


def header_to_mime_encoding(value, not_email=False):
    if not value: return ""

    encoder = Charset(DEFAULT_ENCODING)
    if type(value) == list:
        return "; ".join(properly_encode_header(
            v, encoder, not_email) for v in value)
    else:
        return properly_encode_header(value, encoder, not_email)

def properly_encode_header(value, encoder, not_email):
    """
    The only thing special (weird) about this function is that it tries
    to do a fast check to see if the header value has an email address in
    it.  Since random headers could have an email address, and email addresses
    have weird special formatting rules, we have to check for it.

    Normally this works fine, but in Librelist, we need to "obfuscate" email
    addresses by changing the '@' to '-AT-'.  This is where
    VALUE_IS_EMAIL_ADDRESS exists.  It's a simple lambda returning True/False
    to check if a header value has an email address.  If you need to make this
    check different, then change this.
    """
    try:
        return value.encode("ascii")
    except UnicodeEncodeError:
        if not_email is False and VALUE_IS_EMAIL_ADDRESS(value):
            # this could have an email address, make sure we don't screw it up
            name, address = parseaddr(value)
            return '"%s" <%s>' % (
                encoder.header_encode(name.encode("utf-8")), address)

        return encoder.header_encode(value.encode("utf-8"))