From bb924f1f4d3981ec200ce8036a6a9ca33f399339 Mon Sep 17 00:00:00 2001 From: Pascal F Date: Wed, 3 Apr 2024 11:37:21 +0200 Subject: [PATCH 1/4] Add minimal type check --- django_mailbox/models.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/django_mailbox/models.py b/django_mailbox/models.py index d98ba1fd..e5027536 100644 --- a/django_mailbox/models.py +++ b/django_mailbox/models.py @@ -655,6 +655,9 @@ def reply(self, message): pre-set it. """ + if not isinstance(message, EmailMessage): + raise ValueError('Message must be an instance of EmailMessage') + if not message.from_email: if self.mailbox.from_email: message.from_email = self.mailbox.from_email From 7337d3130ee9dcc04236d4539219440c013d5ef8 Mon Sep 17 00:00:00 2001 From: Pascal F Date: Fri, 24 May 2024 07:36:53 +0200 Subject: [PATCH 2/4] Improve documentation --- django_mailbox/models.py | 9 ++++++++- docs/topics/mailbox_types.rst | 29 +++++++++++++++-------------- 2 files changed, 23 insertions(+), 15 deletions(-) diff --git a/django_mailbox/models.py b/django_mailbox/models.py index e5027536..95a9fccf 100644 --- a/django_mailbox/models.py +++ b/django_mailbox/models.py @@ -654,9 +654,16 @@ def reply(self, message): to retrieve that information through normal channels, so we must pre-set it. + For conveninence, you can use django.core.mail.EmailMessage to build a Message instance:: + + from django.core.mail import EmailMessage + + message.reply( + EmailMessage(subject="pong", body="pongpong") + ) """ if not isinstance(message, EmailMessage): - raise ValueError('Message must be an instance of EmailMessage') + raise ValueError('Message must be an instance of email.Message') if not message.from_email: if self.mailbox.from_email: diff --git a/docs/topics/mailbox_types.rst b/docs/topics/mailbox_types.rst index 6fabf768..349986ef 100644 --- a/docs/topics/mailbox_types.rst +++ b/docs/topics/mailbox_types.rst @@ -7,20 +7,20 @@ POP3 and IMAP as well as local file-based mailboxes. .. table:: 'Protocol' Options - ============ ================ ==================================================================================================================================================================== - Mailbox Type 'Protocol':// Notes - ============ ================ ==================================================================================================================================================================== - POP3 ``pop3://`` Can also specify SSL with ``pop3+ssl://`` - IMAP ``imap://`` Can also specify SSL with ``imap+ssl://`` or STARTTLS with ``imap+tls``; additional configuration is also possible: see :ref:`pop3-and-imap-mailboxes` for details. - Gmail IMAP ``gmail+ssl://`` Uses OAuth authentication for Gmail's IMAP transport. See :ref:`gmail-oauth` for details. - Office365 API``office365://`` Uses OAuth authentication for Office365 API transport. See :ref:`office365-oauth` for details. - Maildir ``maildir://`` - Mbox ``mbox://`` - Babyl ``babyl://`` - MH ``mh://`` - MMDF ``mmdf://`` - Piped Mail *empty* See :ref:`receiving-mail-from-exim4-or-postfix` - ============ ================ ==================================================================================================================================================================== + ================ ================ ==================================================================================================================================================================== + Mailbox Type 'Protocol':// Notes + ================ ================ ==================================================================================================================================================================== + POP3 ``pop3://`` Can also specify SSL with ``pop3+ssl://`` + IMAP ``imap://`` Can also specify SSL with ``imap+ssl://`` or STARTTLS with ``imap+tls``; additional configuration is also possible: see :ref:`pop3-and-imap-mailboxes` for details. + Gmail IMAP ``gmail+ssl://`` Uses OAuth authentication for Gmail's IMAP transport. See :ref:`gmail-oauth` for details. + Office365 API ``office365://`` Uses OAuth authentication for Office365 API transport. See :ref:`office365-oauth` for details. + Maildir ``maildir://`` *empty* + Mbox ``mbox://`` *empty* + Babyl ``babyl://`` *empty* + MH ``mh://`` *empty* + MMDF ``mmdf://`` *empty* + Piped Mail *empty* See :ref:`receiving-mail-from-exim4-or-postfix` + ================ ================ ==================================================================================================================================================================== .. warning:: @@ -117,6 +117,7 @@ Build your URI accordingly:: .. _office365-oauth: + Office 365 API ------------------------------------- From 88e17ea6842477c479b77ab1ecf727472bb2093d Mon Sep 17 00:00:00 2001 From: Pascal F Date: Fri, 24 May 2024 08:36:04 +0200 Subject: [PATCH 3/4] Fix typing --- django_mailbox/models.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/django_mailbox/models.py b/django_mailbox/models.py index 95a9fccf..86108eaf 100644 --- a/django_mailbox/models.py +++ b/django_mailbox/models.py @@ -662,8 +662,10 @@ def reply(self, message): EmailMessage(subject="pong", body="pongpong") ) """ - if not isinstance(message, EmailMessage): - raise ValueError('Message must be an instance of email.Message') + from django.core.mail import EmailMessage as DjangoEmailMessage + + if not isinstance(message, DjangoEmailMessage): + raise ValueError('Message must be an instance of django.core.mail.EmailMessage') if not message.from_email: if self.mailbox.from_email: From 6dcded0e7515ca1fbcdcdc03254450757065c512 Mon Sep 17 00:00:00 2001 From: Pascal F Date: Fri, 24 May 2024 09:25:54 +0200 Subject: [PATCH 4/4] Improve documentation and flake8 --- django_mailbox/models.py | 17 +++++++---------- django_mailbox/tests/test_process_email.py | 5 ++++- django_mailbox/transports/base.py | 2 +- django_mailbox/transports/imap.py | 2 +- 4 files changed, 13 insertions(+), 13 deletions(-) diff --git a/django_mailbox/models.py b/django_mailbox/models.py index 86108eaf..4c76dc55 100644 --- a/django_mailbox/models.py +++ b/django_mailbox/models.py @@ -300,7 +300,7 @@ def _get_dehydrated_message(self, msg, record): new = EmailMessage() if ( msg.is_multipart() - and not 'attachment' in msg.get('Content-Disposition', '') + and 'attachment' not in msg.get('Content-Disposition', '') ): for header, value in msg.items(): new[header] = value @@ -647,20 +647,18 @@ def to_addresses(self): return addresses def reply(self, message): - """Sends a message as a reply to this message instance. - - Although Django's e-mail processing will set both Message-ID - and Date upon generating the e-mail message, we will not be able - to retrieve that information through normal channels, so we must - pre-set it. - - For conveninence, you can use django.core.mail.EmailMessage to build a Message instance:: + """Sends an EmailMessage as a reply to this message instance:: from django.core.mail import EmailMessage message.reply( EmailMessage(subject="pong", body="pongpong") ) + + Although Django's e-mail processing will set both Message-ID + and Date upon generating the e-mail message, we will not be able + to retrieve that information through normal channels, so we must + pre-set it. """ from django.core.mail import EmailMessage as DjangoEmailMessage @@ -900,7 +898,6 @@ def __str__(self): return f'{self.get_filename()}: {self.document.url}' return self.get_filename() - class Meta: verbose_name = _('Message attachment') verbose_name_plural = _('Message attachments') diff --git a/django_mailbox/tests/test_process_email.py b/django_mailbox/tests/test_process_email.py index d3316e89..50d43ff6 100644 --- a/django_mailbox/tests/test_process_email.py +++ b/django_mailbox/tests/test_process_email.py @@ -345,6 +345,9 @@ def test_message_reply(self): ) msg = self.mailbox.record_outgoing_message(email_object.message()) + with self.assertRaises(ValueError): + msg.reply(Message(subject="ping", body="pong")) + self.assertTrue(msg.outgoing) actual_from = 'username@example.com' @@ -439,7 +442,7 @@ def test_message_compressed(self): msg = self.mailbox.process_incoming_message(message) - actual_email_object = msg.get_email_object() + _actual_email_object = msg.get_email_object() self.assertTrue(msg.eml.name.endswith('.eml.gz')) diff --git a/django_mailbox/transports/base.py b/django_mailbox/transports/base.py index 57679f8a..51c6b54e 100644 --- a/django_mailbox/transports/base.py +++ b/django_mailbox/transports/base.py @@ -11,4 +11,4 @@ def get_email_from_bytes(self, contents): return message def close(self): - pass \ No newline at end of file + pass diff --git a/django_mailbox/transports/imap.py b/django_mailbox/transports/imap.py index 5b0d2dc8..fb67a8dd 100644 --- a/django_mailbox/transports/imap.py +++ b/django_mailbox/transports/imap.py @@ -53,7 +53,7 @@ def close(self): try: self.server.close() self.server.logout() - except (imaplib.IMAP4.error, OSError) as e: + except (imaplib.IMAP4.error, OSError) as e: logger.warning(f'Failed to close IMAP connection, ignoring: {e}') pass