Кодировать MIMEText как цитируемые печатные формы

Python поддерживает довольно функциональную MIME-библиотеку email.mime,

Чего я хочу добиться, так это получить часть MIME, содержащую простой текст UTF-8, который будет закодирован как печатные материалы в кавычках, а не как base64. Хотя все функции доступны в библиотеке, мне не удалось использовать ее:

Пример:

import email.mime.text, email.encoders
m=email.mime.text.MIMEText(u'This is the text containing ünicöde', _charset='utf-8')
m.as_string()
# => Leads to a base64-encoded message, as base64 is the default.

email.encoders.encode_quopri(m)
m.as_string()
# => Leads to a strange message

Последняя команда приводит к странному сообщению:

Content-Type: text/plain; charset="utf-8"
MIME-Version: 1.0
Content-Transfer-Encoding: base64
Content-Transfer-Encoding: quoted-printable

GhpcyBpcyB0aGUgdGV4dCBjb250YWluaW5nIMO8bmljw7ZkZQ=3D=3D

Это явно не закодировано как цитируемые печатные формы, двойной transfer-encoding заголовок наконец-то странный (если не незаконный).

Как я могу закодировать текст в виде цитируемых сообщений в сообщении MIME?

2 ответа

Решение

Хорошо, у меня есть одно решение, которое очень хакерское, но, по крайней мере, оно ведет в некотором направлении: MIMEText предполагает base64, и я не знаю, как это изменить. По этой причине я использую MIMENonMultipart:

import email.mime, email.mime.nonmultipart, email.charset
m=email.mime.nonmultipart.MIMENonMultipart('text', 'plain', charset='utf-8')

#Construct a new charset which uses Quoted Printables (base64 is default)
cs=email.charset.Charset('utf-8')
cs.body_encoding = email.charset.QP

#Now set the content using the new charset
m.set_payload(u'This is the text containing ünicöde', charset=cs)

Теперь кажется, что сообщение правильно закодировано:

Content-Type: text/plain; charset="utf-8"
MIME-Version: 1.0
Content-Transfer-Encoding: quoted-printable

This is the text containing =C3=BCnic=C3=B6de

Можно даже построить новый класс, который скрывает сложность:

class MIMEUTF8QPText(email.mime.nonmultipart.MIMENonMultipart):
  def __init__(self, payload):
    email.mime.nonmultipart.MIMENonMultipart.__init__(self, 'text', 'plain',
                                                      charset='utf-8')

    utf8qp=email.charset.Charset('utf-8')
    utf8qp.body_encoding=email.charset.QP

    self.set_payload(payload, charset=utf8qp) 

И используйте это так:

m = MIMEUTF8QPText(u'This is the text containing ünicöde')
m.as_string()

Тебе не нужен твой взлом

import email

# Construct a new charset which uses Quoted Printables (base64 is default)
cs = email.charset.Charset('utf-8')
cs.body_encoding = email.charset.QP

m = email.mime.text.MIMEText(u'This is the text containing ünicöde', 'plain', _charset=cs)

print(m.as_string())

Адаптировано из выпуска 1525919 и протестировано на Python 2.7:

from email.Message import Message
from email.Charset import Charset, QP

text = "\xc3\xa1 = \xc3\xa9"
msg = Message()

charset = Charset('utf-8')
charset.header_encoding = QP
charset.body_encoding = QP

msg.set_charset(charset)
msg.set_payload(msg._charset.body_encode(text))

print msg.as_string()

дам тебе:

MIME-Version: 1.0
Content-Type: text/plain; charset="utf-8"
Content-Transfer-Encoding: quoted-printable

=C3=A1 =3D =C3=A9

Также посмотрите этот ответ от коммиттера Python.

Другие вопросы по тегам