mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #24623 -- Fixed EmailMessage.attach_file() with text files on Python 3.
Thanks tkrapp for the report and Tim Graham for the review.
This commit is contained in:
		| @@ -308,10 +308,36 @@ class EmailMessage(object): | |||||||
|             self.attachments.append((filename, content, mimetype)) |             self.attachments.append((filename, content, mimetype)) | ||||||
|  |  | ||||||
|     def attach_file(self, path, mimetype=None): |     def attach_file(self, path, mimetype=None): | ||||||
|         """Attaches a file from the filesystem.""" |         """ | ||||||
|  |         Attaches a file from the filesystem. | ||||||
|  |  | ||||||
|  |         The mimetype will be set to the DEFAULT_ATTACHMENT_MIME_TYPE if it is | ||||||
|  |         not specified and cannot be guessed or (PY3 only) if it suggests | ||||||
|  |         text/* for a binary file. | ||||||
|  |         """ | ||||||
|         filename = os.path.basename(path) |         filename = os.path.basename(path) | ||||||
|         with open(path, 'rb') as f: |         if not mimetype: | ||||||
|             content = f.read() |             mimetype, _ = mimetypes.guess_type(filename) | ||||||
|  |             if not mimetype: | ||||||
|  |                 mimetype = DEFAULT_ATTACHMENT_MIME_TYPE | ||||||
|  |         basetype, subtype = mimetype.split('/', 1) | ||||||
|  |         read_mode = 'r' if basetype == 'text' else 'rb' | ||||||
|  |         content = None | ||||||
|  |  | ||||||
|  |         with open(path, read_mode) as f: | ||||||
|  |             try: | ||||||
|  |                 content = f.read() | ||||||
|  |             except UnicodeDecodeError: | ||||||
|  |                 # If mimetype suggests the file is text but it's actually | ||||||
|  |                 # binary, read() will raise a UnicodeDecodeError on Python 3. | ||||||
|  |                 pass | ||||||
|  |  | ||||||
|  |         # If the previous read in text mode failed, try binary mode. | ||||||
|  |         if content is None: | ||||||
|  |             with open(path, 'rb') as f: | ||||||
|  |                 content = f.read() | ||||||
|  |                 mimetype = DEFAULT_ATTACHMENT_MIME_TYPE | ||||||
|  |  | ||||||
|         self.attach(filename, content, mimetype) |         self.attach(filename, content, mimetype) | ||||||
|  |  | ||||||
|     def _create_message(self, msg): |     def _create_message(self, msg): | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								tests/mail/attachments/file.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								tests/mail/attachments/file.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 144 B | 
							
								
								
									
										1
									
								
								tests/mail/attachments/file.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/mail/attachments/file.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | django/django | ||||||
							
								
								
									
										
											BIN
										
									
								
								tests/mail/attachments/file_png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								tests/mail/attachments/file_png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 144 B | 
							
								
								
									
										
											BIN
										
									
								
								tests/mail/attachments/file_png.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								tests/mail/attachments/file_png.txt
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 144 B | 
							
								
								
									
										1
									
								
								tests/mail/attachments/file_txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/mail/attachments/file_txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | django/django | ||||||
							
								
								
									
										1
									
								
								tests/mail/attachments/file_txt.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tests/mail/attachments/file_txt.png
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | django/django | ||||||
| @@ -2,6 +2,7 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  |  | ||||||
| import asyncore | import asyncore | ||||||
|  | import mimetypes | ||||||
| import os | import os | ||||||
| import shutil | import shutil | ||||||
| import smtpd | import smtpd | ||||||
| @@ -20,6 +21,7 @@ from django.core.mail import ( | |||||||
| from django.core.mail.backends import console, dummy, filebased, locmem, smtp | from django.core.mail.backends import console, dummy, filebased, locmem, smtp | ||||||
| from django.core.mail.message import BadHeaderError | from django.core.mail.message import BadHeaderError | ||||||
| from django.test import SimpleTestCase, override_settings | from django.test import SimpleTestCase, override_settings | ||||||
|  | from django.utils._os import upath | ||||||
| from django.utils.encoding import force_bytes, force_text | from django.utils.encoding import force_bytes, force_text | ||||||
| from django.utils.six import PY3, StringIO, binary_type | from django.utils.six import PY3, StringIO, binary_type | ||||||
| from django.utils.translation import ugettext_lazy | from django.utils.translation import ugettext_lazy | ||||||
| @@ -305,6 +307,35 @@ class MailTests(HeadersCheckMixin, SimpleTestCase): | |||||||
|         payload = message.get_payload() |         payload = message.get_payload() | ||||||
|         self.assertEqual(payload[1].get_filename(), 'une pièce jointe.pdf') |         self.assertEqual(payload[1].get_filename(), 'une pièce jointe.pdf') | ||||||
|  |  | ||||||
|  |     def test_attach_file(self): | ||||||
|  |         """ | ||||||
|  |         Test attaching a file against different mimetypes and make sure that | ||||||
|  |         a file will be attached and sent properly even if an invalid mimetype | ||||||
|  |         is specified. | ||||||
|  |         """ | ||||||
|  |         files = ( | ||||||
|  |             # filename, actual mimetype | ||||||
|  |             ('file.txt', 'text/plain'), | ||||||
|  |             ('file.png', 'image/png'), | ||||||
|  |             ('file_txt', None), | ||||||
|  |             ('file_png', None), | ||||||
|  |             ('file_txt.png', 'image/png'), | ||||||
|  |             ('file_png.txt', 'text/plain'), | ||||||
|  |         ) | ||||||
|  |         test_mimetypes = ['text/plain', 'image/png', None] | ||||||
|  |  | ||||||
|  |         for basename, real_mimetype in files: | ||||||
|  |             for mimetype in test_mimetypes: | ||||||
|  |                 email = EmailMessage('subject', 'body', 'from@example.com', ['to@example.com']) | ||||||
|  |                 self.assertEqual(mimetypes.guess_type(basename)[0], real_mimetype) | ||||||
|  |                 self.assertEqual(email.attachments, []) | ||||||
|  |                 file_path = os.path.join(os.path.dirname(upath(__file__)), 'attachments', basename) | ||||||
|  |                 email.attach_file(file_path, mimetype=mimetype) | ||||||
|  |                 self.assertEqual(len(email.attachments), 1) | ||||||
|  |                 self.assertIn(basename, email.attachments[0]) | ||||||
|  |                 msgs_sent_num = email.send() | ||||||
|  |                 self.assertEqual(msgs_sent_num, 1) | ||||||
|  |  | ||||||
|     def test_dummy_backend(self): |     def test_dummy_backend(self): | ||||||
|         """ |         """ | ||||||
|         Make sure that dummy backends returns correct number of sent messages |         Make sure that dummy backends returns correct number of sent messages | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user