mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #19934 - Use of Pillow is now preferred over PIL.
This starts the deprecation period for PIL (support to end in 1.8).
This commit is contained in:
		| @@ -1,7 +1,7 @@ | |||||||
| """ | """ | ||||||
| Utility functions for handling images. | Utility functions for handling images. | ||||||
|  |  | ||||||
| Requires PIL, as you might imagine. | Requires Pillow (or PIL), as you might imagine. | ||||||
| """ | """ | ||||||
| import zlib | import zlib | ||||||
|  |  | ||||||
| @@ -35,11 +35,7 @@ def get_image_dimensions(file_or_path, close=False): | |||||||
|     'close' to True to close the file at the end if it is initially in an open |     'close' to True to close the file at the end if it is initially in an open | ||||||
|     state. |     state. | ||||||
|     """ |     """ | ||||||
|     # Try to import PIL in either of the two ways it can end up installed. |     from django.utils.image import ImageFile as PILImageFile | ||||||
|     try: |  | ||||||
|         from PIL import ImageFile as PILImageFile |  | ||||||
|     except ImportError: |  | ||||||
|         import ImageFile as PILImageFile |  | ||||||
|  |  | ||||||
|     p = PILImageFile.Parser() |     p = PILImageFile.Parser() | ||||||
|     if hasattr(file_or_path, 'read'): |     if hasattr(file_or_path, 'read'): | ||||||
|   | |||||||
| @@ -105,14 +105,10 @@ def get_validation_errors(outfile, app=None): | |||||||
|             if isinstance(f, models.FileField) and not f.upload_to: |             if isinstance(f, models.FileField) and not f.upload_to: | ||||||
|                 e.add(opts, '"%s": FileFields require an "upload_to" attribute.' % f.name) |                 e.add(opts, '"%s": FileFields require an "upload_to" attribute.' % f.name) | ||||||
|             if isinstance(f, models.ImageField): |             if isinstance(f, models.ImageField): | ||||||
|                 # Try to import PIL in either of the two ways it can end up installed. |  | ||||||
|                 try: |                 try: | ||||||
|                     from PIL import Image |                     from django.utils.image import Image | ||||||
|                 except ImportError: |                 except ImportError: | ||||||
|                     try: |                     e.add(opts, '"%s": To use ImageFields, you need to install Pillow. Get it at https://pypi.python.org/pypi/Pillow.' % f.name) | ||||||
|                         import Image |  | ||||||
|                     except ImportError: |  | ||||||
|                         e.add(opts, '"%s": To use ImageFields, you need to install the Python Imaging Library. Get it at http://www.pythonware.com/products/pil/ .' % f.name) |  | ||||||
|             if isinstance(f, models.BooleanField) and getattr(f, 'null', False): |             if isinstance(f, models.BooleanField) and getattr(f, 'null', False): | ||||||
|                 e.add(opts, '"%s": BooleanFields do not accept null values. Use a NullBooleanField instead.' % f.name) |                 e.add(opts, '"%s": BooleanFields do not accept null values. Use a NullBooleanField instead.' % f.name) | ||||||
|             if isinstance(f, models.FilePathField) and not (f.allow_files or f.allow_folders): |             if isinstance(f, models.FilePathField) and not (f.allow_files or f.allow_folders): | ||||||
|   | |||||||
| @@ -602,13 +602,9 @@ class ImageField(FileField): | |||||||
|         if f is None: |         if f is None: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         # Try to import PIL in either of the two ways it can end up installed. |         from django.utils.image import Image | ||||||
|         try: |  | ||||||
|             from PIL import Image |  | ||||||
|         except ImportError: |  | ||||||
|             import Image |  | ||||||
|  |  | ||||||
|         # We need to get a file object for PIL. We might have a path or we might |         # We need to get a file object for Pillow. We might have a path or we might | ||||||
|         # have to read the data into memory. |         # have to read the data into memory. | ||||||
|         if hasattr(data, 'temporary_file_path'): |         if hasattr(data, 'temporary_file_path'): | ||||||
|             file = data.temporary_file_path() |             file = data.temporary_file_path() | ||||||
| @@ -623,12 +619,8 @@ class ImageField(FileField): | |||||||
|             # image in memory, which is a DoS vector. See #3848 and #18520. |             # image in memory, which is a DoS vector. See #3848 and #18520. | ||||||
|             # verify() must be called immediately after the constructor. |             # verify() must be called immediately after the constructor. | ||||||
|             Image.open(file).verify() |             Image.open(file).verify() | ||||||
|         except ImportError: |         except Exception: | ||||||
|             # Under PyPy, it is possible to import PIL. However, the underlying |             # Pillow (or PIL) doesn't recognize it as an image. | ||||||
|             # _imaging C module isn't available, so an ImportError will be |  | ||||||
|             # raised. Catch and re-raise. |  | ||||||
|             raise |  | ||||||
|         except Exception: # Python Imaging Library doesn't recognize it as an image |  | ||||||
|             six.reraise(ValidationError, ValidationError(self.error_messages['invalid_image']), sys.exc_info()[2]) |             six.reraise(ValidationError, ValidationError(self.error_messages['invalid_image']), sys.exc_info()[2]) | ||||||
|         if hasattr(f, 'seek') and callable(f.seek): |         if hasattr(f, 'seek') and callable(f.seek): | ||||||
|             f.seek(0) |             f.seek(0) | ||||||
|   | |||||||
							
								
								
									
										146
									
								
								django/utils/image.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								django/utils/image.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,146 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | """ | ||||||
|  | To provide a shim layer over Pillow/PIL situation until the PIL support is | ||||||
|  | removed. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Combinations To Account For | ||||||
|  | =========================== | ||||||
|  |  | ||||||
|  | * Pillow: | ||||||
|  |  | ||||||
|  |     * never has ``_imaging`` under any Python | ||||||
|  |     * has the ``Image.alpha_composite``, which may aid in detection | ||||||
|  |  | ||||||
|  | * PIL | ||||||
|  |  | ||||||
|  |     * CPython 2.x may have _imaging (& work) | ||||||
|  |     * CPython 2.x may *NOT* have _imaging (broken & needs a error message) | ||||||
|  |     * CPython 3.x doesn't work | ||||||
|  |     * PyPy will *NOT* have _imaging (but works?) | ||||||
|  |  | ||||||
|  | Restated, that looks like: | ||||||
|  |  | ||||||
|  | * If we're on Python 2.x, it could be either Pillow or PIL: | ||||||
|  |  | ||||||
|  |     * If ``import _imaging`` results in ``ImportError``, either they have a | ||||||
|  |       working Pillow installation or a broken PIL installation, so we need to | ||||||
|  |       detect further: | ||||||
|  |  | ||||||
|  |         * To detect, we first ``import Image``. | ||||||
|  |         * If ``Image`` has a ``alpha_composite`` attribute present, only Pillow | ||||||
|  |           has this, so we assume it's working. | ||||||
|  |         * If ``Image`` DOES NOT have a ``alpha_composite``attribute, it must be | ||||||
|  |           PIL & is a broken (likely C compiler-less) install, which we need to | ||||||
|  |           warn the user about. | ||||||
|  |  | ||||||
|  |     * If ``import _imaging`` works, it must be PIL & is a working install. | ||||||
|  |  | ||||||
|  | * Python 3.x | ||||||
|  |  | ||||||
|  |     * If ``import Image`` works, it must be Pillow, since PIL isn't Python 3.x | ||||||
|  |       compatible. | ||||||
|  |  | ||||||
|  | * PyPy | ||||||
|  |  | ||||||
|  |     * If ``import _imaging`` results in ``ImportError``, it could be either | ||||||
|  |       Pillow or PIL, both of which work without it on PyPy, so we're fine. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Approach | ||||||
|  | ======== | ||||||
|  |  | ||||||
|  | * Attempt to import ``Image`` | ||||||
|  |  | ||||||
|  |     * ``ImportError`` - nothing is installed, toss an exception | ||||||
|  |     * Either Pillow or the PIL is installed, so continue detecting | ||||||
|  |  | ||||||
|  | * Attempt to ``hasattr(Image, 'alpha_composite')`` | ||||||
|  |  | ||||||
|  |     * If it works, it's Pillow & working | ||||||
|  |     * If it fails, we've got a PIL install, continue detecting | ||||||
|  |  | ||||||
|  |         * The only option here is that we're on Python 2.x or PyPy, of which | ||||||
|  |           we only care about if we're on CPython. | ||||||
|  |         * If we're on CPython, attempt to ``import _imaging`` | ||||||
|  |  | ||||||
|  |             * ``ImportError`` - Bad install, toss an exception | ||||||
|  |  | ||||||
|  | """ | ||||||
|  | import warnings | ||||||
|  |  | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
|  | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Image = None | ||||||
|  | _imaging = None | ||||||
|  | ImageFile = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _detect_image_library(): | ||||||
|  |     global Image | ||||||
|  |     global _imaging | ||||||
|  |     global ImageFile | ||||||
|  |  | ||||||
|  |     # Skip re-attempting to import if we've already run detection. | ||||||
|  |     if Image is not None: | ||||||
|  |         return Image, _imaging, ImageFile | ||||||
|  |  | ||||||
|  |     # Assume it's not there. | ||||||
|  |     PIL_imaging = False | ||||||
|  |  | ||||||
|  |     try: | ||||||
|  |         # Try from the Pillow (or one variant of PIL) install location first. | ||||||
|  |         from PIL import Image as PILImage | ||||||
|  |     except ImportError as err: | ||||||
|  |         try: | ||||||
|  |             # If that failed, try the alternate import syntax for PIL. | ||||||
|  |             import Image as PILImage | ||||||
|  |         except ImportError as err: | ||||||
|  |             # Neither worked, so it's likely not installed. | ||||||
|  |             raise ImproperlyConfigured( | ||||||
|  |                 _(u"Neither Pillow nor PIL could be imported: %s" % err) | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     # ``Image.alpha_composite`` was added to Pillow in SHA: e414c6 & is not | ||||||
|  |     # available in any version of the PIL. | ||||||
|  |     if hasattr(PILImage, u'alpha_composite'): | ||||||
|  |         PIL_imaging = False | ||||||
|  |     else: | ||||||
|  |         # We're dealing with the PIL. Determine if we're on CPython & if | ||||||
|  |         # ``_imaging`` is available. | ||||||
|  |         import platform | ||||||
|  |  | ||||||
|  |         # This is the Alex Approved™ way. | ||||||
|  |         # See http://mail.python.org/pipermail//pypy-dev/2011-November/008739.html | ||||||
|  |         if platform.python_implementation().lower() == u'cpython': | ||||||
|  |             # We're on CPython (likely 2.x). Since a C compiler is needed to | ||||||
|  |             # produce a fully-working PIL & will create a ``_imaging`` module, | ||||||
|  |             # we'll attempt to import it to verify their kit works. | ||||||
|  |             try: | ||||||
|  |                 import _imaging as PIL_imaging | ||||||
|  |             except ImportError as err: | ||||||
|  |                 raise ImproperlyConfigured( | ||||||
|  |                     _(u"The '_imaging' module for the PIL could not be " + | ||||||
|  |                       u"imported: %s" % err) | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |     # Try to import ImageFile as well. | ||||||
|  |     try: | ||||||
|  |         from PIL import ImageFile as PILImageFile | ||||||
|  |     except ImportError: | ||||||
|  |         import ImageFile as PILImageFile | ||||||
|  |  | ||||||
|  |     # Finally, warn about deprecation... | ||||||
|  |     if PIL_imaging is not False: | ||||||
|  |         warnings.warn( | ||||||
|  |             "Support for the PIL will be removed in Django 1.8. Please " + | ||||||
|  |             "uninstall it & install Pillow instead.", | ||||||
|  |             PendingDeprecationWarning | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     return PILImage, PIL_imaging, PILImageFile | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Image, _imaging, ImageFile = _detect_image_library() | ||||||
| @@ -27,7 +27,7 @@ to make it dead easy, even for someone who may not be intimately familiar with | |||||||
| that area of the code, to understand the problem and verify the fix: | that area of the code, to understand the problem and verify the fix: | ||||||
|  |  | ||||||
| * Are there clear instructions on how to reproduce the bug? If this | * Are there clear instructions on how to reproduce the bug? If this | ||||||
|   touches a dependency (such as PIL), a contrib module, or a specific |   touches a dependency (such as Pillow/PIL), a contrib module, or a specific | ||||||
|   database, are those instructions clear enough even for someone not |   database, are those instructions clear enough even for someone not | ||||||
|   familiar with it? |   familiar with it? | ||||||
|  |  | ||||||
|   | |||||||
| @@ -365,6 +365,12 @@ these changes. | |||||||
| * ``django.conf.urls.shortcut`` and ``django.views.defaults.shortcut`` will be | * ``django.conf.urls.shortcut`` and ``django.views.defaults.shortcut`` will be | ||||||
|   removed. |   removed. | ||||||
|  |  | ||||||
|  | * Support for the Python Imaging Library (PIL) module will be removed, as it | ||||||
|  |   no longer appears to be actively maintained & does not work on Python 3. | ||||||
|  |   You are advised to install `Pillow`_, which should be used instead. | ||||||
|  |  | ||||||
|  | .. _`Pillow`: https://pypi.python.org/pypi/Pillow | ||||||
|  |  | ||||||
| * The following private APIs will be removed: | * The following private APIs will be removed: | ||||||
|  |  | ||||||
|   - ``django.db.close_connection()`` |   - ``django.db.close_connection()`` | ||||||
|   | |||||||
| @@ -608,19 +608,21 @@ For each field, we describe the default widget used if you don't specify | |||||||
|     * Normalizes to: An ``UploadedFile`` object that wraps the file content |     * Normalizes to: An ``UploadedFile`` object that wraps the file content | ||||||
|       and file name into a single object. |       and file name into a single object. | ||||||
|     * Validates that file data has been bound to the form, and that the |     * Validates that file data has been bound to the form, and that the | ||||||
|       file is of an image format understood by PIL. |       file is of an image format understood by Pillow/PIL. | ||||||
|     * Error message keys: ``required``, ``invalid``, ``missing``, ``empty``, |     * Error message keys: ``required``, ``invalid``, ``missing``, ``empty``, | ||||||
|       ``invalid_image`` |       ``invalid_image`` | ||||||
|  |  | ||||||
|     Using an ``ImageField`` requires that the `Python Imaging Library`_ (PIL) |     Using an ``ImageField`` requires that either `Pillow`_ (recommended) or the | ||||||
|     is installed and supports the image formats you use. If you encounter a |     `Python Imaging Library`_ (PIL) are installed and supports the image | ||||||
|     ``corrupt image`` error when you upload an image, it usually means PIL |     formats you use. If you encounter a ``corrupt image`` error when you | ||||||
|  |     upload an image, it usually means either Pillow or PIL | ||||||
|     doesn't understand its format. To fix this, install the appropriate |     doesn't understand its format. To fix this, install the appropriate | ||||||
|     library and reinstall PIL. |     library and reinstall Pillow or PIL. | ||||||
|  |  | ||||||
|     When you use an ``ImageField`` on a form, you must also remember to |     When you use an ``ImageField`` on a form, you must also remember to | ||||||
|     :ref:`bind the file data to the form <binding-uploaded-files>`. |     :ref:`bind the file data to the form <binding-uploaded-files>`. | ||||||
|  |  | ||||||
|  | .. _Pillow: http://python-imaging.github.io/Pillow/ | ||||||
| .. _Python Imaging Library: http://www.pythonware.com/products/pil/ | .. _Python Imaging Library: http://www.pythonware.com/products/pil/ | ||||||
|  |  | ||||||
| ``IntegerField`` | ``IntegerField`` | ||||||
|   | |||||||
| @@ -220,6 +220,13 @@ Minor features | |||||||
| * Added ``BCryptSHA256PasswordHasher`` to resolve the password truncation issue | * Added ``BCryptSHA256PasswordHasher`` to resolve the password truncation issue | ||||||
|   with bcrypt. |   with bcrypt. | ||||||
|  |  | ||||||
|  | * `Pillow`_ is now the preferred image manipulation library to use with Django. | ||||||
|  |   `PIL`_ is pending deprecation (support to be removed in Django 1.8). | ||||||
|  |   To upgrade, you should **first** uninstall PIL, **then** install Pillow. | ||||||
|  |  | ||||||
|  | .. _`Pillow`: https://pypi.python.org/pypi/Pillow | ||||||
|  | .. _`PIL`: https://pypi.python.org/pypi/PIL | ||||||
|  |  | ||||||
| Backwards incompatible changes in 1.6 | Backwards incompatible changes in 1.6 | ||||||
| ===================================== | ===================================== | ||||||
|  |  | ||||||
|   | |||||||
| @@ -29,15 +29,9 @@ from django.utils._os import upath | |||||||
| from django.test.utils import override_settings | from django.test.utils import override_settings | ||||||
| from servers.tests import LiveServerBase | from servers.tests import LiveServerBase | ||||||
|  |  | ||||||
| # Try to import PIL in either of the two ways it can end up installed. |  | ||||||
| # Checking for the existence of Image is enough for CPython, but |  | ||||||
| # for PyPy, you need to check for the underlying modules |  | ||||||
| try: | try: | ||||||
|     from PIL import Image, _imaging |     from django.utils.image import Image | ||||||
| except ImportError: | except ImproperlyConfigured: | ||||||
|     try: |  | ||||||
|         import Image, _imaging |  | ||||||
|     except ImportError: |  | ||||||
|     Image = None |     Image = None | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -494,7 +488,7 @@ class DimensionClosingBug(unittest.TestCase): | |||||||
|     """ |     """ | ||||||
|     Test that get_image_dimensions() properly closes files (#8817) |     Test that get_image_dimensions() properly closes files (#8817) | ||||||
|     """ |     """ | ||||||
|     @unittest.skipUnless(Image, "PIL not installed") |     @unittest.skipUnless(Image, "Pillow/PIL not installed") | ||||||
|     def test_not_closing_of_files(self): |     def test_not_closing_of_files(self): | ||||||
|         """ |         """ | ||||||
|         Open files passed into get_image_dimensions() should stay opened. |         Open files passed into get_image_dimensions() should stay opened. | ||||||
| @@ -505,7 +499,7 @@ class DimensionClosingBug(unittest.TestCase): | |||||||
|         finally: |         finally: | ||||||
|             self.assertTrue(not empty_io.closed) |             self.assertTrue(not empty_io.closed) | ||||||
|  |  | ||||||
|     @unittest.skipUnless(Image, "PIL not installed") |     @unittest.skipUnless(Image, "Pillow/PIL not installed") | ||||||
|     def test_closing_of_filenames(self): |     def test_closing_of_filenames(self): | ||||||
|         """ |         """ | ||||||
|         get_image_dimensions() called with a filename should closed the file. |         get_image_dimensions() called with a filename should closed the file. | ||||||
| @@ -542,7 +536,7 @@ class InconsistentGetImageDimensionsBug(unittest.TestCase): | |||||||
|     Test that get_image_dimensions() works properly after various calls |     Test that get_image_dimensions() works properly after various calls | ||||||
|     using a file handler (#11158) |     using a file handler (#11158) | ||||||
|     """ |     """ | ||||||
|     @unittest.skipUnless(Image, "PIL not installed") |     @unittest.skipUnless(Image, "Pillow/PIL not installed") | ||||||
|     def test_multiple_calls(self): |     def test_multiple_calls(self): | ||||||
|         """ |         """ | ||||||
|         Multiple calls of get_image_dimensions() should return the same size. |         Multiple calls of get_image_dimensions() should return the same size. | ||||||
| @@ -556,7 +550,7 @@ class InconsistentGetImageDimensionsBug(unittest.TestCase): | |||||||
|         self.assertEqual(image_pil.size, size_1) |         self.assertEqual(image_pil.size, size_1) | ||||||
|         self.assertEqual(size_1, size_2) |         self.assertEqual(size_1, size_2) | ||||||
|  |  | ||||||
|     @unittest.skipUnless(Image, "PIL not installed") |     @unittest.skipUnless(Image, "Pillow/PIL not installed") | ||||||
|     def test_bug_19457(self): |     def test_bug_19457(self): | ||||||
|         """ |         """ | ||||||
|         Regression test for #19457 |         Regression test for #19457 | ||||||
|   | |||||||
| @@ -1,16 +1,11 @@ | |||||||
| import os | import os | ||||||
| import tempfile | import tempfile | ||||||
|  |  | ||||||
| # Try to import PIL in either of the two ways it can end up installed. | from django.core.exceptions import ImproperlyConfigured | ||||||
| # Checking for the existence of Image is enough for CPython, but for PyPy, |  | ||||||
| # you need to check for the underlying modules. |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     from PIL import Image, _imaging |     from django.utils.image import Image | ||||||
| except ImportError: | except ImproperlyConfigured: | ||||||
|     try: |  | ||||||
|         import Image, _imaging |  | ||||||
|     except ImportError: |  | ||||||
|     Image = None |     Image = None | ||||||
|  |  | ||||||
| from django.core.files.storage import FileSystemStorage | from django.core.files.storage import FileSystemStorage | ||||||
| @@ -87,7 +82,7 @@ class VerboseNameField(models.Model): | |||||||
|     field9 = models.FileField("verbose field9", upload_to="unused") |     field9 = models.FileField("verbose field9", upload_to="unused") | ||||||
|     field10 = models.FilePathField("verbose field10") |     field10 = models.FilePathField("verbose field10") | ||||||
|     field11 = models.FloatField("verbose field11") |     field11 = models.FloatField("verbose field11") | ||||||
|     # Don't want to depend on PIL in this test |     # Don't want to depend on Pillow/PIL in this test | ||||||
|     #field_image = models.ImageField("verbose field") |     #field_image = models.ImageField("verbose field") | ||||||
|     field12 = models.IntegerField("verbose field12") |     field12 = models.IntegerField("verbose field12") | ||||||
|     field13 = models.IPAddressField("verbose field13") |     field13 = models.IPAddressField("verbose field13") | ||||||
| @@ -119,7 +114,7 @@ class Document(models.Model): | |||||||
| ############################################################################### | ############################################################################### | ||||||
| # ImageField | # ImageField | ||||||
|  |  | ||||||
| # If PIL available, do these tests. | # If Pillow/PIL available, do these tests. | ||||||
| if Image: | if Image: | ||||||
|     class TestImageFieldFile(ImageFieldFile): |     class TestImageFieldFile(ImageFieldFile): | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -3,20 +3,24 @@ from __future__ import absolute_import | |||||||
| import os | import os | ||||||
| import shutil | import shutil | ||||||
|  |  | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.core.files import File | from django.core.files import File | ||||||
| from django.core.files.images import ImageFile | from django.core.files.images import ImageFile | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| from django.utils._os import upath | from django.utils._os import upath | ||||||
| from django.utils.unittest import skipIf | from django.utils.unittest import skipIf | ||||||
|  |  | ||||||
| from .models import Image | try: | ||||||
|  |     from .models import Image | ||||||
|  | except ImproperlyConfigured: | ||||||
|  |     Image = None | ||||||
|  |  | ||||||
| if Image: | if Image: | ||||||
|     from .models import (Person, PersonWithHeight, PersonWithHeightAndWidth, |     from .models import (Person, PersonWithHeight, PersonWithHeightAndWidth, | ||||||
|         PersonDimensionsFirst, PersonTwoImages, TestImageFieldFile) |         PersonDimensionsFirst, PersonTwoImages, TestImageFieldFile) | ||||||
|     from .models import temp_storage_dir |     from .models import temp_storage_dir | ||||||
| else: | else: | ||||||
|     # PIL not available, create dummy classes (tests will be skipped anyway) |     # Pillow not available, create dummy classes (tests will be skipped anyway) | ||||||
|     class Person(): |     class Person(): | ||||||
|         pass |         pass | ||||||
|     PersonWithHeight = PersonWithHeightAndWidth = PersonDimensionsFirst = Person |     PersonWithHeight = PersonWithHeightAndWidth = PersonDimensionsFirst = Person | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ from __future__ import unicode_literals | |||||||
| import os | import os | ||||||
| import tempfile | import tempfile | ||||||
|  |  | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.core.files.storage import FileSystemStorage | from django.core.files.storage import FileSystemStorage | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils import six | from django.utils import six | ||||||
| @@ -91,14 +92,7 @@ class TextFile(models.Model): | |||||||
|         return self.description |         return self.description | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     # If PIL is available, try testing ImageFields. Checking for the existence |     from django.utils.image import Image | ||||||
|     # of Image is enough for CPython, but for PyPy, you need to check for the |  | ||||||
|     # underlying modules If PIL is not available, ImageField tests are omitted. |  | ||||||
|     # Try to import PIL in either of the two ways it can end up installed. |  | ||||||
|     try: |  | ||||||
|         from PIL import Image, _imaging |  | ||||||
|     except ImportError: |  | ||||||
|         import Image, _imaging |  | ||||||
|  |  | ||||||
|     test_images = True |     test_images = True | ||||||
|  |  | ||||||
| @@ -137,7 +131,7 @@ try: | |||||||
|  |  | ||||||
|         def __str__(self): |         def __str__(self): | ||||||
|             return self.description |             return self.description | ||||||
| except ImportError: | except ImproperlyConfigured: | ||||||
|     test_images = False |     test_images = False | ||||||
|  |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| A test spanning all the capabilities of all the serializers. | A test spanning all the capabilities of all the serializers. | ||||||
|  |  | ||||||
| This class sets up a model for each model field type | This class sets up a model for each model field type | ||||||
| (except for image types, because of the PIL dependency). | (except for image types, because of the Pillow/PIL dependency). | ||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.db import models | from django.db import models | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user