mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	Refs #23919 -- Removed Python 2 notes in docs.
This commit is contained in:
		| @@ -134,7 +134,6 @@ pygments_style = 'trac' | |||||||
| intersphinx_mapping = { | intersphinx_mapping = { | ||||||
|     'python': ('https://docs.python.org/3/', None), |     'python': ('https://docs.python.org/3/', None), | ||||||
|     'sphinx': ('http://sphinx-doc.org/', None), |     'sphinx': ('http://sphinx-doc.org/', None), | ||||||
|     'six': ('https://pythonhosted.org/six/', None), |  | ||||||
|     'psycopg2': ('http://initd.org/psycopg/docs/', None), |     'psycopg2': ('http://initd.org/psycopg/docs/', None), | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -65,18 +65,11 @@ is the last version to support Python 3.3. | |||||||
| What Python version should I use with Django? | What Python version should I use with Django? | ||||||
| ============================================= | ============================================= | ||||||
|  |  | ||||||
| As of Django 1.6, Python 3 support is considered stable and you can safely use | Python 3 is recommended. Django 1.11 is the last version to support Python 2.7. | ||||||
| it in production. See also :doc:`/topics/python3`. However, the community is | Support for Python 2.7 and Django 1.11 ends in 2020. | ||||||
| still in the process of migrating third-party packages and applications to |  | ||||||
| Python 3. |  | ||||||
|  |  | ||||||
| If you're starting a new project, and the dependencies you plan to use work on |  | ||||||
| Python 3, you should use Python 3. If they don't, consider contributing to the |  | ||||||
| porting efforts, or stick to Python 2. |  | ||||||
|  |  | ||||||
| Since newer versions of Python are often faster, have more features, and are | Since newer versions of Python are often faster, have more features, and are | ||||||
| better supported, all else being equal, we recommend that you use the latest | better supported, the latest version of Python 3 is recommended. | ||||||
| 2.x.y or 3.x.y release. |  | ||||||
|  |  | ||||||
| You don't lose anything in Django by using an older release, but you don't take | You don't lose anything in Django by using an older release, but you don't take | ||||||
| advantage of the improvements and optimizations in newer Python releases. | advantage of the improvements and optimizations in newer Python releases. | ||||||
|   | |||||||
| @@ -55,15 +55,6 @@ pitfalls producing this error: | |||||||
|   case, please refer to your system documentation to learn how you can change |   case, please refer to your system documentation to learn how you can change | ||||||
|   this to a UTF-8 locale. |   this to a UTF-8 locale. | ||||||
|  |  | ||||||
| * You created raw bytestrings, which is easy to do on Python 2:: |  | ||||||
|  |  | ||||||
|       my_string = 'café' |  | ||||||
|  |  | ||||||
|   Either use the ``u''`` prefix or even better, add the |  | ||||||
|   ``from __future__ import unicode_literals`` line at the top of your file |  | ||||||
|   so that your code will be compatible with Python 3.2 which doesn't support |  | ||||||
|   the ``u''`` prefix. |  | ||||||
|  |  | ||||||
| Related resources: | Related resources: | ||||||
|  |  | ||||||
| * :doc:`Unicode in Django </ref/unicode>` | * :doc:`Unicode in Django </ref/unicode>` | ||||||
|   | |||||||
| @@ -26,10 +26,6 @@ directory whose name doesn't begin with an underscore. For example:: | |||||||
|         tests.py |         tests.py | ||||||
|         views.py |         views.py | ||||||
|  |  | ||||||
| On Python 2, be sure to include ``__init__.py`` files in both the |  | ||||||
| ``management`` and ``management/commands`` directories as done above or your |  | ||||||
| command will not be detected. |  | ||||||
|  |  | ||||||
| In this example, the ``closepoll`` command will be made available to any project | In this example, the ``closepoll`` command will be made available to any project | ||||||
| that includes the ``polls`` application in :setting:`INSTALLED_APPS`. | that includes the ``polls`` application in :setting:`INSTALLED_APPS`. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -704,14 +704,12 @@ smoothly: | |||||||
|    a field that's similar to what you want and extend it a little bit, |    a field that's similar to what you want and extend it a little bit, | ||||||
|    instead of creating an entirely new field from scratch. |    instead of creating an entirely new field from scratch. | ||||||
|  |  | ||||||
| 2. Put a ``__str__()`` (``__unicode__()`` on Python 2) method on the class you're | 2. Put a ``__str__()`` method on the class you're wrapping up as a field. There | ||||||
|    wrapping up as a field. There are a lot of places where the default |    are a lot of places where the default behavior of the field code is to call | ||||||
|    behavior of the field code is to call |  | ||||||
|    :func:`~django.utils.encoding.force_text` on the value. (In our |    :func:`~django.utils.encoding.force_text` on the value. (In our | ||||||
|    examples in this document, ``value`` would be a ``Hand`` instance, not a |    examples in this document, ``value`` would be a ``Hand`` instance, not a | ||||||
|    ``HandField``). So if your ``__str__()`` method (``__unicode__()`` on |    ``HandField``). So if your ``__str__()`` method automatically converts to | ||||||
|    Python 2) automatically converts to the string form of your Python object, |    the string form of your Python object, you can save yourself a lot of work. | ||||||
|    you can save yourself a lot of work. |  | ||||||
|  |  | ||||||
| Writing a ``FileField`` subclass | Writing a ``FileField`` subclass | ||||||
| ================================ | ================================ | ||||||
|   | |||||||
| @@ -250,19 +250,3 @@ details about the default templates: | |||||||
| * :ref:`http_internal_server_error_view` | * :ref:`http_internal_server_error_view` | ||||||
| * :ref:`http_forbidden_view` | * :ref:`http_forbidden_view` | ||||||
| * :ref:`http_bad_request_view` | * :ref:`http_bad_request_view` | ||||||
|  |  | ||||||
| Python Options |  | ||||||
| ============== |  | ||||||
|  |  | ||||||
| It's strongly recommended that you invoke the Python process running your |  | ||||||
| Django application using the `-R`_ option or with the :envvar:`PYTHONHASHSEED` |  | ||||||
| environment variable set to ``random``. This option is enabled by default |  | ||||||
| starting with Python 3.3. |  | ||||||
|  |  | ||||||
| These options help protect your site from denial-of-service (DoS) |  | ||||||
| attacks triggered by carefully crafted inputs. Such an attack can |  | ||||||
| drastically increase CPU usage by causing worst-case performance when |  | ||||||
| creating ``dict`` instances. See `oCERT advisory #2011-003 |  | ||||||
| <http://www.ocert.org/advisories/ocert-2011-003.html>`_ for more information. |  | ||||||
|  |  | ||||||
| .. _-r: https://docs.python.org/2/using/cmdline.html#cmdoption-R |  | ||||||
|   | |||||||
| @@ -53,26 +53,6 @@ mention: | |||||||
|   about escaping strings with quotes or commas in them. Just pass |   about escaping strings with quotes or commas in them. Just pass | ||||||
|   ``writerow()`` your raw strings, and it'll do the right thing. |   ``writerow()`` your raw strings, and it'll do the right thing. | ||||||
|  |  | ||||||
| .. admonition:: Handling Unicode on Python 2 |  | ||||||
|  |  | ||||||
|     Python 2's :mod:`csv` module does not support Unicode input. Since Django |  | ||||||
|     uses Unicode internally this means strings read from sources such as |  | ||||||
|     :class:`~django.http.HttpRequest` are potentially problematic. There are a |  | ||||||
|     few options for handling this: |  | ||||||
|  |  | ||||||
|     * Manually encode all Unicode objects to a compatible encoding. |  | ||||||
|  |  | ||||||
|     * Use the ``UnicodeWriter`` class provided in the `csv module's examples |  | ||||||
|       section`_. |  | ||||||
|  |  | ||||||
|     * Use the `python-unicodecsv module`_, which aims to be a drop-in |  | ||||||
|       replacement for :mod:`csv` that gracefully handles Unicode. |  | ||||||
|  |  | ||||||
|     For more information, see the Python documentation of the :mod:`csv` module. |  | ||||||
|  |  | ||||||
|     .. _`csv module's examples section`: https://docs.python.org/2/library/csv.html#examples |  | ||||||
|     .. _`python-unicodecsv module`: https://github.com/jdunck/python-unicodecsv |  | ||||||
|  |  | ||||||
| .. _streaming-csv-files: | .. _streaming-csv-files: | ||||||
|  |  | ||||||
| Streaming large CSV files | Streaming large CSV files | ||||||
| @@ -89,7 +69,6 @@ the assembly and transmission of a large CSV file:: | |||||||
|  |  | ||||||
|     import csv |     import csv | ||||||
|  |  | ||||||
|     from django.utils.six.moves import range |  | ||||||
|     from django.http import StreamingHttpResponse |     from django.http import StreamingHttpResponse | ||||||
|  |  | ||||||
|     class Echo(object): |     class Echo(object): | ||||||
|   | |||||||
| @@ -282,7 +282,6 @@ Django aims to be compatible with multiple different flavors and versions of | |||||||
| Python: | Python: | ||||||
|  |  | ||||||
| * :doc:`Jython support <howto/jython>` | * :doc:`Jython support <howto/jython>` | ||||||
| * :doc:`Python 3 compatibility <topics/python3>` |  | ||||||
|  |  | ||||||
| Geographic framework | Geographic framework | ||||||
| ==================== | ==================== | ||||||
|   | |||||||
| @@ -122,8 +122,8 @@ browser. | |||||||
| To measure code coverage when running the tests, you need to view that file | To measure code coverage when running the tests, you need to view that file | ||||||
| over HTTP. To view code coverage: | over HTTP. To view code coverage: | ||||||
|  |  | ||||||
| * Execute ``python -m http.server`` (or ``python -m SimpleHTTPServer`` on | * Execute ``python -m http.server`` from the root directory (not from inside | ||||||
|   Python 2) from the root directory (not from inside ``js_tests``). |   ``js_tests``). | ||||||
| * Open http://localhost:8000/js_tests/tests.html in your web browser. | * Open http://localhost:8000/js_tests/tests.html in your web browser. | ||||||
|  |  | ||||||
| Testing from the command line | Testing from the command line | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ Next, clone your fork, install some requirements, and run the tests:: | |||||||
|    $ git clone git@github.com:YourGitHubName/django.git django-repo |    $ git clone git@github.com:YourGitHubName/django.git django-repo | ||||||
|    $ cd django-repo/tests |    $ cd django-repo/tests | ||||||
|    $ pip install -e .. |    $ pip install -e .. | ||||||
|    $ pip install -r requirements/py3.txt  # Python 2: py2.txt |    $ pip install -r requirements/py3.txt | ||||||
|    $ ./runtests.py |    $ ./runtests.py | ||||||
|  |  | ||||||
| Installing the requirements will likely require some operating system packages | Installing the requirements will likely require some operating system packages | ||||||
| @@ -39,8 +39,7 @@ that your computer doesn't have installed. You can usually figure out which | |||||||
| package to install by doing a Web search for the last line or so of the error | package to install by doing a Web search for the last line or so of the error | ||||||
| message. Try adding your operating system to the search query if needed. | message. Try adding your operating system to the search query if needed. | ||||||
|  |  | ||||||
| If you have trouble installing the requirements, you can skip that step, except | If you have trouble installing the requirements, you can skip that step. See | ||||||
| on Python 2, where you must ``pip install mock``. See |  | ||||||
| :ref:`running-unit-tests-dependencies` for details on installing the optional | :ref:`running-unit-tests-dependencies` for details on installing the optional | ||||||
| test dependencies. If you don't have an optional dependency installed, the | test dependencies. If you don't have an optional dependency installed, the | ||||||
| tests that require it will be skipped. | tests that require it will be skipped. | ||||||
| @@ -75,9 +74,8 @@ command from any place in the Django source tree:: | |||||||
| By default, ``tox`` runs the test suite with the bundled test settings file for | By default, ``tox`` runs the test suite with the bundled test settings file for | ||||||
| SQLite, ``flake8``, ``isort``, and the documentation spelling checker. In | SQLite, ``flake8``, ``isort``, and the documentation spelling checker. In | ||||||
| addition to the system dependencies noted elsewhere in this documentation, | addition to the system dependencies noted elsewhere in this documentation, | ||||||
| the commands ``python2`` and ``python3`` must be on your path and linked to | the command ``python3`` must be on your path and linked to the appropriate | ||||||
| the appropriate versions of Python. A list of default environments can be seen | version of Python. A list of default environments can be seen as follows:: | ||||||
| as follows:: |  | ||||||
|  |  | ||||||
|     $ tox -l |     $ tox -l | ||||||
|     py3 |     py3 | ||||||
| @@ -225,7 +223,6 @@ dependencies: | |||||||
| *  argon2-cffi_ 16.1.0+ | *  argon2-cffi_ 16.1.0+ | ||||||
| *  bcrypt_ | *  bcrypt_ | ||||||
| *  docutils_ | *  docutils_ | ||||||
| *  enum34_ (Python 2 only) |  | ||||||
| *  geoip2_ | *  geoip2_ | ||||||
| *  jinja2_ 2.7+ | *  jinja2_ 2.7+ | ||||||
| *  numpy_ | *  numpy_ | ||||||
| @@ -234,7 +231,6 @@ dependencies: | |||||||
| *  pytz_ (required) | *  pytz_ (required) | ||||||
| *  setuptools_ | *  setuptools_ | ||||||
| *  memcached_, plus a :ref:`supported Python binding <memcached>` | *  memcached_, plus a :ref:`supported Python binding <memcached>` | ||||||
| *  mock_ (for Python 2) |  | ||||||
| *  gettext_ (:ref:`gettext_on_windows`) | *  gettext_ (:ref:`gettext_on_windows`) | ||||||
| *  selenium_ | *  selenium_ | ||||||
| *  sqlparse_ | *  sqlparse_ | ||||||
| @@ -243,7 +239,7 @@ You can find these dependencies in `pip requirements files`_ inside the | |||||||
| ``tests/requirements`` directory of the Django source tree and install them | ``tests/requirements`` directory of the Django source tree and install them | ||||||
| like so:: | like so:: | ||||||
|  |  | ||||||
|    $ pip install -r tests/requirements/py3.txt  # Python 2: py2.txt |    $ pip install -r tests/requirements/py3.txt | ||||||
|  |  | ||||||
| If you encounter an error during the installation, your system might be missing | If you encounter an error during the installation, your system might be missing | ||||||
| a dependency for one or more of the Python packages. Consult the failing | a dependency for one or more of the Python packages. Consult the failing | ||||||
| @@ -265,7 +261,6 @@ associated tests will be skipped. | |||||||
| .. _argon2-cffi: https://pypi.python.org/pypi/argon2_cffi | .. _argon2-cffi: https://pypi.python.org/pypi/argon2_cffi | ||||||
| .. _bcrypt: https://pypi.python.org/pypi/bcrypt | .. _bcrypt: https://pypi.python.org/pypi/bcrypt | ||||||
| .. _docutils: https://pypi.python.org/pypi/docutils | .. _docutils: https://pypi.python.org/pypi/docutils | ||||||
| .. _enum34: https://pypi.python.org/pypi/enum34 |  | ||||||
| .. _geoip2: https://pypi.python.org/pypi/geoip2 | .. _geoip2: https://pypi.python.org/pypi/geoip2 | ||||||
| .. _jinja2: https://pypi.python.org/pypi/jinja2 | .. _jinja2: https://pypi.python.org/pypi/jinja2 | ||||||
| .. _numpy: https://pypi.python.org/pypi/numpy | .. _numpy: https://pypi.python.org/pypi/numpy | ||||||
| @@ -274,7 +269,6 @@ associated tests will be skipped. | |||||||
| .. _pytz: https://pypi.python.org/pypi/pytz/ | .. _pytz: https://pypi.python.org/pypi/pytz/ | ||||||
| .. _setuptools: https://pypi.python.org/pypi/setuptools/ | .. _setuptools: https://pypi.python.org/pypi/setuptools/ | ||||||
| .. _memcached: http://memcached.org/ | .. _memcached: http://memcached.org/ | ||||||
| .. _mock: https://pypi.python.org/pypi/mock |  | ||||||
| .. _gettext: https://www.gnu.org/software/gettext/manual/gettext.html | .. _gettext: https://www.gnu.org/software/gettext/manual/gettext.html | ||||||
| .. _selenium: https://pypi.python.org/pypi/selenium | .. _selenium: https://pypi.python.org/pypi/selenium | ||||||
| .. _sqlparse: https://pypi.python.org/pypi/sqlparse | .. _sqlparse: https://pypi.python.org/pypi/sqlparse | ||||||
|   | |||||||
| @@ -70,9 +70,9 @@ probably got the answers. | |||||||
|  |  | ||||||
| .. admonition:: Python 3 required! | .. admonition:: Python 3 required! | ||||||
|  |  | ||||||
|     This tutorial assumes you are using Python 3. Get the latest version at |     The current development version of Django doesn't support Python 2.7. Get | ||||||
|     `Python's download page <https://www.python.org/download/>`_ or with your |     Python 3 at `Python's download page <https://www.python.org/download/>`_ or | ||||||
|     operating system's package manager. |     with your operating system's package manager. | ||||||
|  |  | ||||||
| .. admonition:: For Windows users | .. admonition:: For Windows users | ||||||
|  |  | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ database-schema problems. Here's a quick example: | |||||||
|     class Reporter(models.Model): |     class Reporter(models.Model): | ||||||
|         full_name = models.CharField(max_length=70) |         full_name = models.CharField(max_length=70) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.full_name |             return self.full_name | ||||||
|  |  | ||||||
|     class Article(models.Model): |     class Article(models.Model): | ||||||
| @@ -42,7 +42,7 @@ database-schema problems. Here's a quick example: | |||||||
|         content = models.TextField() |         content = models.TextField() | ||||||
|         reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE) |         reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.headline |             return self.headline | ||||||
|  |  | ||||||
| Install it | Install it | ||||||
|   | |||||||
| @@ -220,10 +220,8 @@ this. For a small app like polls, this process isn't too difficult. | |||||||
|                'License :: OSI Approved :: BSD License',  # example license |                'License :: OSI Approved :: BSD License',  # example license | ||||||
|                'Operating System :: OS Independent', |                'Operating System :: OS Independent', | ||||||
|                'Programming Language :: Python', |                'Programming Language :: Python', | ||||||
|                # Replace these appropriately if you are stuck on Python 2. |  | ||||||
|                'Programming Language :: Python :: 3', |  | ||||||
|                'Programming Language :: Python :: 3.4', |  | ||||||
|                'Programming Language :: Python :: 3.5', |                'Programming Language :: Python :: 3.5', | ||||||
|  |                'Programming Language :: Python :: 3.6', | ||||||
|                'Topic :: Internet :: WWW/HTTP', |                'Topic :: Internet :: WWW/HTTP', | ||||||
|                'Topic :: Internet :: WWW/HTTP :: Dynamic Content', |                'Topic :: Internet :: WWW/HTTP :: Dynamic Content', | ||||||
|            ], |            ], | ||||||
|   | |||||||
| @@ -295,13 +295,12 @@ Methods | |||||||
|  |  | ||||||
| .. _namespace package: | .. _namespace package: | ||||||
|  |  | ||||||
| Namespace packages as apps (Python 3.3+) | Namespace packages as apps | ||||||
| ---------------------------------------- | -------------------------- | ||||||
|  |  | ||||||
| Python versions 3.3 and later support Python packages without an | Python packages without an ``__init__.py`` file are known as "namespace | ||||||
| ``__init__.py`` file. These packages are known as "namespace packages" and may | packages" and may be spread across multiple directories at different locations | ||||||
| be spread across multiple directories at different locations on ``sys.path`` | on ``sys.path`` (see :pep:`420`). | ||||||
| (see :pep:`420`). |  | ||||||
|  |  | ||||||
| Django applications require a single base filesystem path where Django | Django applications require a single base filesystem path where Django | ||||||
| (depending on configuration) will search for templates, static assets, | (depending on configuration) will search for templates, static assets, | ||||||
|   | |||||||
| @@ -45,9 +45,9 @@ Constructor arguments are: | |||||||
| ``obj`` | ``obj`` | ||||||
|     Optional. An object providing context for the message (for example, the |     Optional. An object providing context for the message (for example, the | ||||||
|     model where the problem was discovered). The object should be a model, |     model where the problem was discovered). The object should be a model, | ||||||
|     field, or manager or any other object that defines ``__str__`` method (on |     field, or manager or any other object that defines a ``__str__()`` method. | ||||||
|     Python 2 you need to define ``__unicode__`` method). The method is used |     The method is used while reporting all messages and its result precedes the | ||||||
|     while reporting all messages and its result precedes the message. |     message. | ||||||
|  |  | ||||||
| ``id`` | ``id`` | ||||||
|     Optional string. A unique identifier for the issue. Identifiers should |     Optional string. A unique identifier for the issue. Identifiers should | ||||||
|   | |||||||
| @@ -57,7 +57,7 @@ simple news application with an ``Article`` model:: | |||||||
|         body = models.TextField() |         body = models.TextField() | ||||||
|         status = models.CharField(max_length=1, choices=STATUS_CHOICES) |         status = models.CharField(max_length=1, choices=STATUS_CHOICES) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.title |             return self.title | ||||||
|  |  | ||||||
| A common task we might perform with a model like this is to update an | A common task we might perform with a model like this is to update an | ||||||
|   | |||||||
| @@ -137,10 +137,8 @@ The ``register`` decorator | |||||||
|  |  | ||||||
|     You can't use this decorator if you have to reference your model admin |     You can't use this decorator if you have to reference your model admin | ||||||
|     class in its ``__init__()`` method, e.g. |     class in its ``__init__()`` method, e.g. | ||||||
|     ``super(PersonAdmin, self).__init__(*args, **kwargs)``. If you are using |     ``super(PersonAdmin, self).__init__(*args, **kwargs)``. You can use | ||||||
|     Python 3 and don't have to worry about supporting Python 2, you can |     ``super().__init__(*args, **kwargs)``. | ||||||
|     use ``super().__init__(*args, **kwargs)`` . Otherwise, you'll have to use |  | ||||||
|     ``admin.site.register()`` instead of this decorator. |  | ||||||
|  |  | ||||||
| Discovery of admin files | Discovery of admin files | ||||||
| ------------------------ | ------------------------ | ||||||
| @@ -543,8 +541,7 @@ subclass:: | |||||||
|         list_display = ('first_name', 'last_name') |         list_display = ('first_name', 'last_name') | ||||||
|  |  | ||||||
|     If you don't set ``list_display``, the admin site will display a single |     If you don't set ``list_display``, the admin site will display a single | ||||||
|     column that displays the ``__str__()`` (``__unicode__()`` on Python 2) |     column that displays the ``__str__()`` representation of each object. | ||||||
|     representation of each object. |  | ||||||
|  |  | ||||||
|     You have four possible values that can be used in ``list_display``: |     You have four possible values that can be used in ``list_display``: | ||||||
|  |  | ||||||
| @@ -594,7 +591,7 @@ subclass:: | |||||||
|     A few special cases to note about ``list_display``: |     A few special cases to note about ``list_display``: | ||||||
|  |  | ||||||
|     * If the field is a ``ForeignKey``, Django will display the |     * If the field is a ``ForeignKey``, Django will display the | ||||||
|       ``__str__()`` (``__unicode__()`` on Python 2) of the related object. |       ``__str__()`` of the related object. | ||||||
|  |  | ||||||
|     * ``ManyToManyField`` fields aren't supported, because that would |     * ``ManyToManyField`` fields aren't supported, because that would | ||||||
|       entail executing a separate SQL statement for each row in the table. |       entail executing a separate SQL statement for each row in the table. | ||||||
| @@ -681,9 +678,8 @@ subclass:: | |||||||
|           class PersonAdmin(admin.ModelAdmin): |           class PersonAdmin(admin.ModelAdmin): | ||||||
|               list_display = ('name', 'born_in_fifties') |               list_display = ('name', 'born_in_fifties') | ||||||
|  |  | ||||||
|     * The ``__str__()`` (``__unicode__()`` on Python 2) method is just |     * The ``__str__()`` method is just as valid in ``list_display`` as any | ||||||
|       as valid in ``list_display`` as any other model method, so it's |       other model method, so it's perfectly OK to do this:: | ||||||
|       perfectly OK to do this:: |  | ||||||
|  |  | ||||||
|           list_display = ('__str__', 'some_other_field') |           list_display = ('__str__', 'some_other_field') | ||||||
|  |  | ||||||
|   | |||||||
| @@ -137,12 +137,11 @@ Attributes | |||||||
|     .. attribute:: username_validator |     .. attribute:: username_validator | ||||||
|  |  | ||||||
|         Points to a validator instance used to validate usernames. Defaults to |         Points to a validator instance used to validate usernames. Defaults to | ||||||
|         :class:`validators.UnicodeUsernameValidator` on Python 3 and |         :class:`validators.UnicodeUsernameValidator`. | ||||||
|         :class:`validators.ASCIIUsernameValidator` on Python 2. |  | ||||||
|  |  | ||||||
|         To change the default username validator, you can subclass the ``User`` |         To change the default username validator, you can subclass the ``User`` | ||||||
|         model and set this attribute to a different validator instance. For |         model and set this attribute to a different validator instance. For | ||||||
|         example, to use ASCII usernames on Python 3:: |         example, to use ASCII usernames:: | ||||||
|  |  | ||||||
|             from django.contrib.auth.models import User |             from django.contrib.auth.models import User | ||||||
|             from django.contrib.auth.validators import ASCIIUsernameValidator |             from django.contrib.auth.validators import ASCIIUsernameValidator | ||||||
| @@ -390,14 +389,12 @@ Validators | |||||||
| .. class:: validators.ASCIIUsernameValidator | .. class:: validators.ASCIIUsernameValidator | ||||||
|  |  | ||||||
|     A field validator allowing only ASCII letters, in addition to ``@``, ``.``, |     A field validator allowing only ASCII letters, in addition to ``@``, ``.``, | ||||||
|     ``+``, ``-``, and ``_``. The default validator for ``User.username`` on |     ``+``, ``-``, and ``_``. | ||||||
|     Python 2. |  | ||||||
|  |  | ||||||
| .. class:: validators.UnicodeUsernameValidator | .. class:: validators.UnicodeUsernameValidator | ||||||
|  |  | ||||||
|     A field validator allowing Unicode letters, in addition to ``@``, ``.``, |     A field validator allowing Unicode letters, in addition to ``@``, ``.``, | ||||||
|     ``+``, ``-``, and ``_``. The default validator for ``User.username`` on |     ``+``, ``-``, and ``_``. The default validator for ``User.username``. | ||||||
|     Python 3. |  | ||||||
|  |  | ||||||
| .. _topics-auth-signals: | .. _topics-auth-signals: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -251,7 +251,7 @@ A simple example is a tagging system, which might look like this:: | |||||||
|         object_id = models.PositiveIntegerField() |         object_id = models.PositiveIntegerField() | ||||||
|         content_object = GenericForeignKey('content_type', 'object_id') |         content_object = GenericForeignKey('content_type', 'object_id') | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.tag |             return self.tag | ||||||
|  |  | ||||||
| A normal :class:`~django.db.models.ForeignKey` can only "point | A normal :class:`~django.db.models.ForeignKey` can only "point | ||||||
|   | |||||||
| @@ -63,8 +63,8 @@ of using ``ogrinspect`` :ref:`in the tutorial <ogrinspect-intro>`. | |||||||
|  |  | ||||||
| .. django-admin-option:: --name-field NAME_FIELD | .. django-admin-option:: --name-field NAME_FIELD | ||||||
|  |  | ||||||
|     Generates a ``__str__`` routine (``__unicode__`` on Python 2) on the model |     Generates a ``__str__()`` method on the model that returns the given field | ||||||
|     that will return the given field name. |     name. | ||||||
|  |  | ||||||
| .. django-admin-option:: --no-imports | .. django-admin-option:: --no-imports | ||||||
|  |  | ||||||
|   | |||||||
| @@ -58,7 +58,7 @@ Example | |||||||
|         name = models.CharField(max_length=25) # corresponds to the 'str' field |         name = models.CharField(max_length=25) # corresponds to the 'str' field | ||||||
|         poly = models.PolygonField(srid=4269) # we want our model in a different SRID |         poly = models.PolygonField(srid=4269) # we want our model in a different SRID | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return 'Name: %s' % self.name |             return 'Name: %s' % self.name | ||||||
|  |  | ||||||
| 3. Use :class:`LayerMapping` to extract all the features and place them in the | 3. Use :class:`LayerMapping` to extract all the features and place them in the | ||||||
|   | |||||||
| @@ -216,7 +216,7 @@ model to represent this data:: | |||||||
|         mpoly = models.MultiPolygonField() |         mpoly = models.MultiPolygonField() | ||||||
|  |  | ||||||
|         # Returns the string representation of the model. |         # Returns the string representation of the model. | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
| Note that the ``models`` module is imported from ``django.contrib.gis.db``. | Note that the ``models`` module is imported from ``django.contrib.gis.db``. | ||||||
|   | |||||||
| @@ -104,7 +104,7 @@ We will use the following example model:: | |||||||
|         name = models.CharField(max_length=200) |         name = models.CharField(max_length=200) | ||||||
|         tags = ArrayField(models.CharField(max_length=200), blank=True) |         tags = ArrayField(models.CharField(max_length=200), blank=True) | ||||||
|  |  | ||||||
|         def __str__(self):  # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
| .. fieldlookup:: arrayfield.contains | .. fieldlookup:: arrayfield.contains | ||||||
| @@ -313,7 +313,7 @@ We will use the following example model:: | |||||||
|         name = models.CharField(max_length=200) |         name = models.CharField(max_length=200) | ||||||
|         data = HStoreField() |         data = HStoreField() | ||||||
|  |  | ||||||
|         def __str__(self):  # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
| .. fieldlookup:: hstorefield.key | .. fieldlookup:: hstorefield.key | ||||||
| @@ -521,7 +521,7 @@ We will use the following example model:: | |||||||
|         name = models.CharField(max_length=200) |         name = models.CharField(max_length=200) | ||||||
|         data = JSONField() |         data = JSONField() | ||||||
|  |  | ||||||
|         def __str__(self):  # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
| .. fieldlookup:: jsonfield.key | .. fieldlookup:: jsonfield.key | ||||||
| @@ -680,7 +680,7 @@ model:: | |||||||
|         ages = IntegerRangeField() |         ages = IntegerRangeField() | ||||||
|         start = models.DateTimeField() |         start = models.DateTimeField() | ||||||
|  |  | ||||||
|         def __str__(self):  # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
| We will also use the following example objects:: | We will also use the following example objects:: | ||||||
|   | |||||||
| @@ -273,24 +273,18 @@ running ``migrate``:: | |||||||
| MySQL DB API Drivers | MySQL DB API Drivers | ||||||
| -------------------- | -------------------- | ||||||
|  |  | ||||||
| The Python Database API is described in :pep:`249`. MySQL has three prominent | MySQL has a couple drivers that implement the Python Database API described in | ||||||
| drivers that implement this API: | :pep:`249`: | ||||||
|  |  | ||||||
| - `MySQLdb`_ is a native driver that has been developed and supported for over | - `mysqlclient`_ is a native driver. It's **the recommended choice**. | ||||||
|   a decade by Andy Dustman. |  | ||||||
| - `mysqlclient`_ is a fork of ``MySQLdb`` which notably supports Python 3 and |  | ||||||
|   can be used as a drop-in replacement for MySQLdb. At the time of this writing, |  | ||||||
|   this is **the recommended choice** for using MySQL with Django. |  | ||||||
| - `MySQL Connector/Python`_ is a pure Python driver from Oracle that does not | - `MySQL Connector/Python`_ is a pure Python driver from Oracle that does not | ||||||
|   require the MySQL client library or any Python modules outside the standard |   require the MySQL client library or any Python modules outside the standard | ||||||
|   library. |   library. | ||||||
|  |  | ||||||
| .. _MySQLdb: https://pypi.python.org/pypi/MySQL-python/1.2.4 |  | ||||||
| .. _mysqlclient: https://pypi.python.org/pypi/mysqlclient | .. _mysqlclient: https://pypi.python.org/pypi/mysqlclient | ||||||
| .. _MySQL Connector/Python: https://dev.mysql.com/downloads/connector/python | .. _MySQL Connector/Python: https://dev.mysql.com/downloads/connector/python | ||||||
|  |  | ||||||
| All these drivers are thread-safe and provide connection pooling. ``MySQLdb`` | All these drivers are thread-safe and provide connection pooling. | ||||||
| is the only one not supporting Python 3 currently. |  | ||||||
|  |  | ||||||
| In addition to a DB API driver, Django needs an adapter to access the database | In addition to a DB API driver, Django needs an adapter to access the database | ||||||
| drivers from its ORM. Django provides an adapter for MySQLdb/mysqlclient while | drivers from its ORM. Django provides an adapter for MySQLdb/mysqlclient while | ||||||
| @@ -298,30 +292,10 @@ MySQL Connector/Python includes `its own`_. | |||||||
|  |  | ||||||
| .. _its own: https://dev.mysql.com/doc/connector-python/en/connector-python-django-backend.html | .. _its own: https://dev.mysql.com/doc/connector-python/en/connector-python-django-backend.html | ||||||
|  |  | ||||||
| MySQLdb |  | ||||||
| ~~~~~~~ |  | ||||||
|  |  | ||||||
| Django requires MySQLdb version 1.2.1p2 or later. |  | ||||||
|  |  | ||||||
| At the time of writing, the latest release of MySQLdb (1.2.5) doesn't support |  | ||||||
| Python 3. In order to use MySQLdb under Python 3, you'll have to install |  | ||||||
| ``mysqlclient`` instead. |  | ||||||
|  |  | ||||||
| .. note:: |  | ||||||
|     There are known issues with the way MySQLdb converts date strings into |  | ||||||
|     datetime objects. Specifically, date strings with value ``0000-00-00`` are |  | ||||||
|     valid for MySQL but will be converted into ``None`` by MySQLdb. |  | ||||||
|  |  | ||||||
|     This means you should be careful while using :djadmin:`loaddata` and |  | ||||||
|     :djadmin:`dumpdata` with rows that may have ``0000-00-00`` values, as they |  | ||||||
|     will be converted to ``None``. |  | ||||||
|  |  | ||||||
| mysqlclient | mysqlclient | ||||||
| ~~~~~~~~~~~ | ~~~~~~~~~~~ | ||||||
|  |  | ||||||
| Django requires `mysqlclient`_ 1.3.3 or later. Note that Python 3.2 is not | Django requires `mysqlclient`_ 1.3.3 or later. | ||||||
| supported. Except for the Python 3.3+ support, mysqlclient should mostly behave |  | ||||||
| the same as MySQLDB. |  | ||||||
|  |  | ||||||
| MySQL Connector/Python | MySQL Connector/Python | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~ | ||||||
| @@ -689,23 +663,6 @@ substring filtering. | |||||||
|  |  | ||||||
| .. _documented at sqlite.org: https://www.sqlite.org/faq.html#q18 | .. _documented at sqlite.org: https://www.sqlite.org/faq.html#q18 | ||||||
|  |  | ||||||
| Old SQLite and ``CASE`` expressions |  | ||||||
| ----------------------------------- |  | ||||||
|  |  | ||||||
| SQLite 3.6.23.1 and older contains a bug when `handling query parameters`_ in |  | ||||||
| a ``CASE`` expression that contains an ``ELSE`` and arithmetic. |  | ||||||
|  |  | ||||||
| SQLite 3.6.23.1 was released in March 2010, and most current binary |  | ||||||
| distributions for different platforms include a newer version of SQLite, with |  | ||||||
| the notable exception of the Python 2.7 installers for Windows. |  | ||||||
|  |  | ||||||
| As of this writing, the latest release for Windows - Python 2.7.10 - includes |  | ||||||
| SQLite 3.6.21. You can install ``pysqlite2`` or replace ``sqlite3.dll`` (by |  | ||||||
| default installed in ``C:\Python27\DLLs``) with a newer version from |  | ||||||
| https://www.sqlite.org/ to remedy this issue. |  | ||||||
|  |  | ||||||
| .. _handling query parameters: https://code.djangoproject.com/ticket/24148 |  | ||||||
|  |  | ||||||
| .. _using-newer-versions-of-pysqlite: | .. _using-newer-versions-of-pysqlite: | ||||||
|  |  | ||||||
| Using newer versions of the SQLite DB-API 2.0 driver | Using newer versions of the SQLite DB-API 2.0 driver | ||||||
|   | |||||||
| @@ -217,11 +217,7 @@ Specification v2.0, for further information. | |||||||
|  |  | ||||||
| As per :pep:`3134`, a ``__cause__`` attribute is set with the original | As per :pep:`3134`, a ``__cause__`` attribute is set with the original | ||||||
| (underlying) database exception, allowing access to any additional | (underlying) database exception, allowing access to any additional | ||||||
| information provided. (Note that this attribute is available under | information provided. | ||||||
| both Python 2 and Python 3, although :pep:`3134` normally only applies |  | ||||||
| to Python 3. To avoid unexpected differences with Python 3, Django will also |  | ||||||
| ensure that the exception made available via ``__cause__`` has a usable |  | ||||||
| ``__traceback__`` attribute.) |  | ||||||
|  |  | ||||||
| .. exception:: models.ProtectedError | .. exception:: models.ProtectedError | ||||||
|  |  | ||||||
|   | |||||||
| @@ -97,8 +97,7 @@ The ``File`` class | |||||||
|  |  | ||||||
|     .. versionchanged:: 1.11 |     .. versionchanged:: 1.11 | ||||||
|  |  | ||||||
|         The ``readable()`` and ``writable()`` methods were added and the |         The ``readable()`` and ``writable()`` methods were added. | ||||||
|         ``seekable()`` method was made available on Python 2. |  | ||||||
|  |  | ||||||
| .. currentmodule:: django.core.files.base | .. currentmodule:: django.core.files.base | ||||||
|  |  | ||||||
|   | |||||||
| @@ -810,12 +810,11 @@ Customizing the error list format | |||||||
|  |  | ||||||
| By default, forms use ``django.forms.utils.ErrorList`` to format validation | By default, forms use ``django.forms.utils.ErrorList`` to format validation | ||||||
| errors. If you'd like to use an alternate class for displaying errors, you can | errors. If you'd like to use an alternate class for displaying errors, you can | ||||||
| pass that in at construction time (replace ``__str__`` by ``__unicode__`` on | pass that in at construction time:: | ||||||
| Python 2):: |  | ||||||
|  |  | ||||||
|     >>> from django.forms.utils import ErrorList |     >>> from django.forms.utils import ErrorList | ||||||
|     >>> class DivErrorList(ErrorList): |     >>> class DivErrorList(ErrorList): | ||||||
|     ...     def __str__(self):              # __unicode__ on Python 2 |     ...     def __str__(self): | ||||||
|     ...         return self.as_divs() |     ...         return self.as_divs() | ||||||
|     ...     def as_divs(self): |     ...     def as_divs(self): | ||||||
|     ...         if not self: return '' |     ...         if not self: return '' | ||||||
| @@ -840,8 +839,7 @@ they're not the only way a form object can be displayed. | |||||||
|    Used to display HTML or access attributes for a single field of a |    Used to display HTML or access attributes for a single field of a | ||||||
|    :class:`Form` instance. |    :class:`Form` instance. | ||||||
|  |  | ||||||
|    The ``__str__()`` (``__unicode__`` on Python 2) method of this |    The ``__str__()`` method of this object displays the HTML for this field. | ||||||
|    object displays the HTML for this field. |  | ||||||
|  |  | ||||||
| To retrieve a single ``BoundField``, use dictionary lookup syntax on your form | To retrieve a single ``BoundField``, use dictionary lookup syntax on your form | ||||||
| using the field's name as the key:: | using the field's name as the key:: | ||||||
|   | |||||||
| @@ -1183,12 +1183,11 @@ method:: | |||||||
|             ... |             ... | ||||||
|             </select> |             </select> | ||||||
|  |  | ||||||
|     The ``__str__`` (``__unicode__`` on Python 2) method of the model will be |     The ``__str__()`` method of the model will be called to generate string | ||||||
|     called to generate string representations of the objects for use in the |     representations of the objects for use in the field's choices. To provide | ||||||
|     field's choices; to provide customized representations, subclass |     customized representations, subclass ``ModelChoiceField`` and override | ||||||
|     ``ModelChoiceField`` and override ``label_from_instance``. This method will |     ``label_from_instance``. This method will receive a model object and should | ||||||
|     receive a model object, and should return a string suitable for representing |     return a string suitable for representing it. For example:: | ||||||
|     it. For example:: |  | ||||||
|  |  | ||||||
|         from django.forms import ModelChoiceField |         from django.forms import ModelChoiceField | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1002,7 +1002,7 @@ For example, suppose you have these models:: | |||||||
|         name = models.CharField(max_length=50) |         name = models.CharField(max_length=50) | ||||||
|         toppings = models.ManyToManyField(Topping) |         toppings = models.ManyToManyField(Topping) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return "%s (%s)" % ( |             return "%s (%s)" % ( | ||||||
|                 self.name, |                 self.name, | ||||||
|                 ", ".join(topping.name for topping in self.toppings.all()), |                 ", ".join(topping.name for topping in self.toppings.all()), | ||||||
|   | |||||||
| @@ -480,21 +480,6 @@ a subclass of dictionary. Exceptions are outlined here: | |||||||
|         >>> q.items() |         >>> q.items() | ||||||
|         [('a', '3')] |         [('a', '3')] | ||||||
|  |  | ||||||
| .. method:: QueryDict.iteritems() |  | ||||||
|  |  | ||||||
|     Just like the standard dictionary ``iteritems()`` method. Like |  | ||||||
|     :meth:`QueryDict.items()` this uses the same last-value logic as |  | ||||||
|     :meth:`QueryDict.__getitem__()`. |  | ||||||
|  |  | ||||||
|     Available only on Python 2. |  | ||||||
|  |  | ||||||
| .. method:: QueryDict.iterlists() |  | ||||||
|  |  | ||||||
|     Like :meth:`QueryDict.iteritems()` except it includes all values, as a list, |  | ||||||
|     for each member of the dictionary. |  | ||||||
|  |  | ||||||
|     Available only on Python 2. |  | ||||||
|  |  | ||||||
| .. method:: QueryDict.values() | .. method:: QueryDict.values() | ||||||
|  |  | ||||||
|     Just like the standard dictionary ``values()`` method, except this uses the |     Just like the standard dictionary ``values()`` method, except this uses the | ||||||
| @@ -504,12 +489,6 @@ a subclass of dictionary. Exceptions are outlined here: | |||||||
|         >>> q.values() |         >>> q.values() | ||||||
|         ['3'] |         ['3'] | ||||||
|  |  | ||||||
| .. method:: QueryDict.itervalues() |  | ||||||
|  |  | ||||||
|     Just like :meth:`QueryDict.values()`, except an iterator. |  | ||||||
|  |  | ||||||
|     Available only on Python 2. |  | ||||||
|  |  | ||||||
| In addition, ``QueryDict`` has the following methods: | In addition, ``QueryDict`` has the following methods: | ||||||
|  |  | ||||||
| .. method:: QueryDict.copy() | .. method:: QueryDict.copy() | ||||||
|   | |||||||
| @@ -105,13 +105,13 @@ Use a dot (``.``) to access attributes of a variable. | |||||||
|     override dictionary lookup. For example, consider the following code snippet |     override dictionary lookup. For example, consider the following code snippet | ||||||
|     that attempts to loop over a ``collections.defaultdict``:: |     that attempts to loop over a ``collections.defaultdict``:: | ||||||
|  |  | ||||||
|         {% for k, v in defaultdict.iteritems %} |         {% for k, v in defaultdict.items %} | ||||||
|             Do something with k and v here... |             Do something with k and v here... | ||||||
|         {% endfor %} |         {% endfor %} | ||||||
|  |  | ||||||
|     Because dictionary lookup happens first, that behavior kicks in and provides |     Because dictionary lookup happens first, that behavior kicks in and provides | ||||||
|     a default value instead of using the intended ``.iteritems()`` |     a default value instead of using the intended ``.items()`` method. In this | ||||||
|     method. In this case, consider converting to a dictionary first. |     case, consider converting to a dictionary first. | ||||||
|  |  | ||||||
| In the above example, ``{{ section.title }}`` will be replaced with the | In the above example, ``{{ section.title }}`` will be replaced with the | ||||||
| ``title`` attribute of the ``section`` object. | ``title`` attribute of the ``section`` object. | ||||||
|   | |||||||
| @@ -48,29 +48,7 @@ General string handling | |||||||
|  |  | ||||||
| Whenever you use strings with Django -- e.g., in database lookups, template | Whenever you use strings with Django -- e.g., in database lookups, template | ||||||
| rendering or anywhere else -- you have two choices for encoding those strings. | rendering or anywhere else -- you have two choices for encoding those strings. | ||||||
| You can use Unicode strings, or you can use normal strings (sometimes called | You can use normal Unicode strings or bytestrings (starting with a 'b'). | ||||||
| "bytestrings") that are encoded using UTF-8. |  | ||||||
|  |  | ||||||
| In Python 3, the logic is reversed, that is normal strings are Unicode, and |  | ||||||
| when you want to specifically create a bytestring, you have to prefix the |  | ||||||
| string with a 'b'. As we are doing in Django code from version 1.5, |  | ||||||
| we recommend that you import ``unicode_literals`` from the __future__ library |  | ||||||
| in your code. Then, when you specifically want to create a bytestring literal, |  | ||||||
| prefix the string with 'b'. |  | ||||||
|  |  | ||||||
| Python 2 legacy:: |  | ||||||
|  |  | ||||||
|     my_string = "This is a bytestring" |  | ||||||
|     my_unicode = u"This is an Unicode string" |  | ||||||
|  |  | ||||||
| Python 2 with unicode literals or Python 3:: |  | ||||||
|  |  | ||||||
|     from __future__ import unicode_literals |  | ||||||
|  |  | ||||||
|     my_string = b"This is a bytestring" |  | ||||||
|     my_unicode = "This is an Unicode string" |  | ||||||
|  |  | ||||||
| See also :doc:`Python 3 compatibility </topics/python3>`. |  | ||||||
|  |  | ||||||
| .. warning:: | .. warning:: | ||||||
|  |  | ||||||
| @@ -114,7 +92,7 @@ imported. | |||||||
| Normally, you won't have to worry about lazy translations. Just be aware that | Normally, you won't have to worry about lazy translations. Just be aware that | ||||||
| if you examine an object and it claims to be a | if you examine an object and it claims to be a | ||||||
| ``django.utils.functional.__proxy__`` object, it is a lazy translation. | ``django.utils.functional.__proxy__`` object, it is a lazy translation. | ||||||
| Calling ``unicode()`` with the lazy translation as the argument will generate a | Calling ``str()`` with the lazy translation as the argument will generate a | ||||||
| Unicode string in the current locale. | Unicode string in the current locale. | ||||||
|  |  | ||||||
| For more details about lazy translation objects, refer to the | For more details about lazy translation objects, refer to the | ||||||
| @@ -140,12 +118,9 @@ for converting back and forth between Unicode and bytestrings. | |||||||
|   ``strings_only`` parameter, if set to True, will result in Python |   ``strings_only`` parameter, if set to True, will result in Python | ||||||
|   numbers, booleans and ``None`` not being converted to a string (they keep |   numbers, booleans and ``None`` not being converted to a string (they keep | ||||||
|   their original types). The ``errors`` parameter takes any of the values |   their original types). The ``errors`` parameter takes any of the values | ||||||
|   that are accepted by Python's ``unicode()`` function for its error |   that are accepted by Python's ``str()`` function for its error | ||||||
|   handling. |   handling. | ||||||
|  |  | ||||||
|   If you pass ``smart_text()`` an object that has a ``__unicode__`` |  | ||||||
|   method, it will use that method to do the conversion. |  | ||||||
|  |  | ||||||
| * ``force_text(s, encoding='utf-8', strings_only=False, | * ``force_text(s, encoding='utf-8', strings_only=False, | ||||||
|   errors='strict')`` is identical to ``smart_text()`` in almost all |   errors='strict')`` is identical to ``smart_text()`` in almost all | ||||||
|   cases. The difference is when the first argument is a :ref:`lazy |   cases. The difference is when the first argument is a :ref:`lazy | ||||||
| @@ -292,8 +267,6 @@ You can pass either Unicode strings or UTF-8 bytestrings as arguments to | |||||||
| ``filter()`` methods and the like in the database API. The following two | ``filter()`` methods and the like in the database API. The following two | ||||||
| querysets are identical:: | querysets are identical:: | ||||||
|  |  | ||||||
|     from __future__ import unicode_literals |  | ||||||
|  |  | ||||||
|     qs = People.objects.filter(name__contains='Å') |     qs = People.objects.filter(name__contains='Å') | ||||||
|     qs = People.objects.filter(name__contains=b'\xc3\x85') # UTF-8 encoding of Å |     qs = People.objects.filter(name__contains=b'\xc3\x85') # UTF-8 encoding of Å | ||||||
|  |  | ||||||
| @@ -302,7 +275,6 @@ Templates | |||||||
|  |  | ||||||
| You can use either Unicode or bytestrings when creating templates manually:: | You can use either Unicode or bytestrings when creating templates manually:: | ||||||
|  |  | ||||||
|     from __future__ import unicode_literals |  | ||||||
|     from django.template import Template |     from django.template import Template | ||||||
|     t1 = Template(b'This is a bytestring template.') |     t1 = Template(b'This is a bytestring template.') | ||||||
|     t2 = Template('This is a Unicode template.') |     t2 = Template('This is a Unicode template.') | ||||||
| @@ -373,7 +345,6 @@ characters. | |||||||
| The following code example demonstrates that everything except email addresses | The following code example demonstrates that everything except email addresses | ||||||
| can be non-ASCII:: | can be non-ASCII:: | ||||||
|  |  | ||||||
|     from __future__ import unicode_literals |  | ||||||
|     from django.core.mail import EmailMessage |     from django.core.mail import EmailMessage | ||||||
|  |  | ||||||
|     subject = 'My visit to Sør-Trøndelag' |     subject = 'My visit to Sør-Trøndelag' | ||||||
|   | |||||||
| @@ -175,9 +175,9 @@ A :class:`ResolverMatch` object can also be assigned to a triple:: | |||||||
| One possible use of :func:`~django.urls.resolve` would be to test whether a | One possible use of :func:`~django.urls.resolve` would be to test whether a | ||||||
| view would raise a ``Http404`` error before redirecting to it:: | view would raise a ``Http404`` error before redirecting to it:: | ||||||
|  |  | ||||||
|  |     from urllib.parse import urlparse | ||||||
|     from django.urls import resolve |     from django.urls import resolve | ||||||
|     from django.http import HttpResponseRedirect, Http404 |     from django.http import HttpResponseRedirect, Http404 | ||||||
|     from django.utils.six.moves.urllib.parse import urlparse |  | ||||||
|  |  | ||||||
|     def myview(request): |     def myview(request): | ||||||
|         next = request.META.get('HTTP_REFERER', None) or '/' |         next = request.META.get('HTTP_REFERER', None) or '/' | ||||||
|   | |||||||
| @@ -203,16 +203,12 @@ The functions defined in this module share the following properties: | |||||||
|  |  | ||||||
| .. function:: smart_text(s, encoding='utf-8', strings_only=False, errors='strict') | .. function:: smart_text(s, encoding='utf-8', strings_only=False, errors='strict') | ||||||
|  |  | ||||||
|     Returns a text object representing ``s`` -- ``unicode`` on Python 2 and |     Returns a ``str`` object representing ``s``. Treats bytestrings using the | ||||||
|     ``str`` on Python 3. Treats bytestrings using the ``encoding`` codec. |     ``encoding`` codec. | ||||||
|  |  | ||||||
|     If ``strings_only`` is ``True``, don't convert (some) non-string-like |     If ``strings_only`` is ``True``, don't convert (some) non-string-like | ||||||
|     objects. |     objects. | ||||||
|  |  | ||||||
| .. function:: smart_unicode(s, encoding='utf-8', strings_only=False, errors='strict') |  | ||||||
|  |  | ||||||
|     Historical name of :func:`smart_text`. Only available under Python 2. |  | ||||||
|  |  | ||||||
| .. function:: is_protected_type(obj) | .. function:: is_protected_type(obj) | ||||||
|  |  | ||||||
|     Determine if the object instance is of a protected type. |     Determine if the object instance is of a protected type. | ||||||
| @@ -228,10 +224,6 @@ The functions defined in this module share the following properties: | |||||||
|     If ``strings_only`` is ``True``, don't convert (some) non-string-like |     If ``strings_only`` is ``True``, don't convert (some) non-string-like | ||||||
|     objects. |     objects. | ||||||
|  |  | ||||||
| .. function:: force_unicode(s, encoding='utf-8', strings_only=False, errors='strict') |  | ||||||
|  |  | ||||||
|     Historical name of :func:`force_text`. Only available under Python 2. |  | ||||||
|  |  | ||||||
| .. function:: smart_bytes(s, encoding='utf-8', strings_only=False, errors='strict') | .. function:: smart_bytes(s, encoding='utf-8', strings_only=False, errors='strict') | ||||||
|  |  | ||||||
|     Returns a bytestring version of ``s``, encoded as specified in |     Returns a bytestring version of ``s``, encoded as specified in | ||||||
| @@ -250,16 +242,20 @@ The functions defined in this module share the following properties: | |||||||
|  |  | ||||||
| .. function:: smart_str(s, encoding='utf-8', strings_only=False, errors='strict') | .. function:: smart_str(s, encoding='utf-8', strings_only=False, errors='strict') | ||||||
|  |  | ||||||
|     Alias of :func:`smart_bytes` on Python 2 and :func:`smart_text` on Python |     Alias of :func:`smart_text`. This function returns a ``str`` or a lazy | ||||||
|     3. This function returns a ``str`` or a lazy string. |     string. | ||||||
|  |  | ||||||
|     For instance, this is  suitable for writing to :data:`sys.stdout` on |     For instance, this is suitable for writing to :data:`sys.stdout`. | ||||||
|     Python 2 and 3. |  | ||||||
|  |     Alias of :func:`smart_bytes` on Python 2 (in older versions of Django that | ||||||
|  |     support it). | ||||||
|  |  | ||||||
| .. function:: force_str(s, encoding='utf-8', strings_only=False, errors='strict') | .. function:: force_str(s, encoding='utf-8', strings_only=False, errors='strict') | ||||||
|  |  | ||||||
|     Alias of :func:`force_bytes` on Python 2 and :func:`force_text` on Python |     Alias of :func:`force_text`. This function always returns a ``str``. | ||||||
|     3. This function always returns a ``str``. |  | ||||||
|  |     Alias of :func:`force_bytes` on Python 2 (in older versions of Django that | ||||||
|  |     support it). | ||||||
|  |  | ||||||
| .. function:: iri_to_uri(iri) | .. function:: iri_to_uri(iri) | ||||||
|  |  | ||||||
| @@ -538,23 +534,22 @@ https://web.archive.org/web/20110718035220/http://diveintomark.org/archives/2004 | |||||||
|  |  | ||||||
|     For example:: |     For example:: | ||||||
|  |  | ||||||
|         from django.utils import six |  | ||||||
|         from django.utils.functional import keep_lazy, keep_lazy_text |         from django.utils.functional import keep_lazy, keep_lazy_text | ||||||
|  |  | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, ...): | ||||||
|             # Do some conversion on string 's' |             # Do some conversion on string 's' | ||||||
|             ... |             ... | ||||||
|         fancy_utility_function = keep_lazy(six.text_type)(fancy_utility_function) |         fancy_utility_function = keep_lazy(str)(fancy_utility_function) | ||||||
|  |  | ||||||
|         # Or more succinctly: |         # Or more succinctly: | ||||||
|         @keep_lazy(six.text_type) |         @keep_lazy(str) | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, ...): | ||||||
|             ... |             ... | ||||||
|  |  | ||||||
|     The ``keep_lazy()`` decorator takes a number of extra arguments (``*args``) |     The ``keep_lazy()`` decorator takes a number of extra arguments (``*args``) | ||||||
|     specifying the type(s) that the original function can return. A common |     specifying the type(s) that the original function can return. A common | ||||||
|     use case is to have functions that return text. For these, you can just |     use case is to have functions that return text. For these, you can just | ||||||
|     pass the ``six.text_type`` type to ``keep_lazy`` (or even simpler, use the |     pass the ``str`` type to ``keep_lazy`` (or even simpler, use the | ||||||
|     :func:`keep_lazy_text` decorator described in the next section). |     :func:`keep_lazy_text` decorator described in the next section). | ||||||
|  |  | ||||||
|     Using this decorator means you can write your function and assume that the |     Using this decorator means you can write your function and assume that the | ||||||
| @@ -563,16 +558,15 @@ https://web.archive.org/web/20110718035220/http://diveintomark.org/archives/2004 | |||||||
|  |  | ||||||
| .. function:: keep_lazy_text(func) | .. function:: keep_lazy_text(func) | ||||||
|  |  | ||||||
|     A shortcut for ``keep_lazy(six.text_type)(func)``. |     A shortcut for ``keep_lazy(str)(func)``. | ||||||
|  |  | ||||||
|     If you have a function that returns text and you want to be able to take |     If you have a function that returns text and you want to be able to take | ||||||
|     lazy arguments while delaying their evaluation, simply use this decorator:: |     lazy arguments while delaying their evaluation, simply use this decorator:: | ||||||
|  |  | ||||||
|         from django.utils import six |  | ||||||
|         from django.utils.functional import keep_lazy, keep_lazy_text |         from django.utils.functional import keep_lazy, keep_lazy_text | ||||||
|  |  | ||||||
|         # Our previous example was: |         # Our previous example was: | ||||||
|         @keep_lazy(six.text_type) |         @keep_lazy(str) | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, ...): | ||||||
|             ... |             ... | ||||||
|  |  | ||||||
| @@ -680,11 +674,9 @@ escaping HTML. | |||||||
|     classes whose output doesn't require HTML escaping. |     classes whose output doesn't require HTML escaping. | ||||||
|  |  | ||||||
|     This decorator defines the ``__html__()`` method on the decorated class |     This decorator defines the ``__html__()`` method on the decorated class | ||||||
|     by wrapping the ``__unicode__()`` (Python 2) or ``__str__()`` (Python 3) |     by wrapping ``__str__()`` in :meth:`~django.utils.safestring.mark_safe`. | ||||||
|     in :meth:`~django.utils.safestring.mark_safe`. Ensure the ``__unicode__()`` |     Ensure the ``__str__()`` method does indeed return text that doesn't | ||||||
|     or ``__str__()`` method does indeed return text that doesn't require HTML |     require HTML escaping. | ||||||
|     escaping. |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ``django.utils.http`` | ``django.utils.http`` | ||||||
| ===================== | ===================== | ||||||
| @@ -737,13 +729,11 @@ escaping HTML. | |||||||
|  |  | ||||||
| .. function:: base36_to_int(s) | .. function:: base36_to_int(s) | ||||||
|  |  | ||||||
|     Converts a base 36 string to an integer. On Python 2 the output is |     Converts a base 36 string to an integer. | ||||||
|     guaranteed to be an ``int`` and not a ``long``. |  | ||||||
|  |  | ||||||
| .. function:: int_to_base36(i) | .. function:: int_to_base36(i) | ||||||
|  |  | ||||||
|     Converts a positive integer to a base 36 string. On Python 2 ``i`` must be |     Converts a positive integer to a base 36 string. | ||||||
|     smaller than `sys.maxint`_. |  | ||||||
|  |  | ||||||
|     .. _sys.maxint: https://docs.python.org/2/library/sys.html#sys.maxint |     .. _sys.maxint: https://docs.python.org/2/library/sys.html#sys.maxint | ||||||
|  |  | ||||||
| @@ -798,17 +788,16 @@ appropriate entities. | |||||||
| .. class:: SafeString | .. class:: SafeString | ||||||
|  |  | ||||||
|     A ``str`` subclass that has been specifically marked as "safe" |     A ``str`` subclass that has been specifically marked as "safe" | ||||||
|     (requires no further escaping) for HTML output purposes. This is |     (requires no further escaping) for HTML output purposes. Alias of | ||||||
|     :class:`SafeBytes` on Python 2 and :class:`SafeText` on Python 3. |     :class:`SafeText`. | ||||||
|  |  | ||||||
|  |     Alias of :class:`SafeBytes` on Python 2 (in older versions of Django that | ||||||
|  |     support it). | ||||||
|  |  | ||||||
| .. class:: SafeText | .. class:: SafeText | ||||||
|  |  | ||||||
|     A ``str`` (in Python 3) or ``unicode`` (in Python 2) subclass |     A ``str`` subclass that has been specifically marked as "safe" for HTML | ||||||
|     that has been specifically marked as "safe" for HTML output purposes. |     output purposes. | ||||||
|  |  | ||||||
| .. class:: SafeUnicode |  | ||||||
|  |  | ||||||
|     Historical name of :class:`SafeText`. Only available under Python 2. |  | ||||||
|  |  | ||||||
| .. function:: mark_safe(s) | .. function:: mark_safe(s) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ Django 1.4.11 release notes | |||||||
| *April 21, 2014* | *April 21, 2014* | ||||||
|  |  | ||||||
| Django 1.4.11 fixes three security issues in 1.4.10. Additionally, | Django 1.4.11 fixes three security issues in 1.4.10. Additionally, | ||||||
| Django's vendored version of six, :mod:`django.utils.six`, has been | Django's vendored version of six, ``django.utils.six``, has been | ||||||
| upgraded to the latest release (1.6.1). | upgraded to the latest release (1.6.1). | ||||||
|  |  | ||||||
| Unexpected code execution using ``reverse()`` | Unexpected code execution using ``reverse()`` | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ Django 1.4.17 release notes | |||||||
|  |  | ||||||
| Django 1.4.17 fixes a regression in the 1.4.14 security release. | Django 1.4.17 fixes a regression in the 1.4.14 security release. | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six`, has | Additionally, Django's vendored version of six, ``django.utils.six``, has | ||||||
| been upgraded to the latest release (1.9.0). | been upgraded to the latest release (1.9.0). | ||||||
|  |  | ||||||
| Bugfixes | Bugfixes | ||||||
|   | |||||||
| @@ -64,5 +64,5 @@ Bugfixes | |||||||
| ======== | ======== | ||||||
|  |  | ||||||
| * To maintain compatibility with Python 2.5, Django's vendored version of six, | * To maintain compatibility with Python 2.5, Django's vendored version of six, | ||||||
|   :mod:`django.utils.six`, has been downgraded to 1.8.0 which is the last |   ``django.utils.six``, has been downgraded to 1.8.0 which is the last | ||||||
|   version to support Python 2.5. |   version to support Python 2.5. | ||||||
|   | |||||||
| @@ -113,5 +113,5 @@ Bugfixes | |||||||
|   ``UnboundLocalError`` if :func:`~django.contrib.auth.get_user_model` |   ``UnboundLocalError`` if :func:`~django.contrib.auth.get_user_model` | ||||||
|   raised an error (#21439). |   raised an error (#21439). | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six`, | Additionally, Django's vendored version of six, ``django.utils.six``, | ||||||
| has been upgraded to the latest release (1.6.1). | has been upgraded to the latest release (1.6.1). | ||||||
|   | |||||||
| @@ -107,9 +107,9 @@ to Python 3, so it's unlikely that a real-world application will have all its | |||||||
| dependencies satisfied under Python 3. | dependencies satisfied under Python 3. | ||||||
|  |  | ||||||
| Thus, we're recommending that Django 1.5 not be used in production under Python | Thus, we're recommending that Django 1.5 not be used in production under Python | ||||||
| 3. Instead, use this opportunity to begin :doc:`porting applications to Python 3 | 3. Instead, use this opportunity to begin porting applications to Python 3. If | ||||||
| </topics/python3>`. If you're an author of a pluggable component, we encourage you | you're an author of a pluggable component, we encourage you to start porting | ||||||
| to start porting now. | now. | ||||||
|  |  | ||||||
| We plan to offer first-class, production-ready support for Python 3 in our next | We plan to offer first-class, production-ready support for Python 3 in our next | ||||||
| release, Django 1.6. | release, Django 1.6. | ||||||
|   | |||||||
| @@ -48,5 +48,5 @@ several bugs in 1.6.1: | |||||||
|   through :func:`~django.utils.safestring.mark_safe` and could end up being |   through :func:`~django.utils.safestring.mark_safe` and could end up being | ||||||
|   double-escaped (:ticket:`21882`). |   double-escaped (:ticket:`21882`). | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six` has been | Additionally, Django's vendored version of six, ``django.utils.six`` has been | ||||||
| upgraded to the latest release (1.5.2). | upgraded to the latest release (1.5.2). | ||||||
|   | |||||||
| @@ -179,5 +179,5 @@ Other bugfixes and changes | |||||||
|   query to include an unnecessary join |   query to include an unnecessary join | ||||||
|   (:ticket:`21760`). |   (:ticket:`21760`). | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six` has been | Additionally, Django's vendored version of six, ``django.utils.six`` has been | ||||||
| upgraded to the latest release (1.6.1). | upgraded to the latest release (1.6.1). | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ Django 1.6.9 release notes | |||||||
|  |  | ||||||
| Django 1.6.9 fixes a regression in the 1.6.6 security release. | Django 1.6.9 fixes a regression in the 1.6.6 security release. | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six`, has | Additionally, Django's vendored version of six, ``django.utils.six``, has | ||||||
| been upgraded to the latest release (1.9.0). | been upgraded to the latest release (1.9.0). | ||||||
|  |  | ||||||
| Bugfixes | Bugfixes | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ Django 1.7.2 release notes | |||||||
|  |  | ||||||
| Django 1.7.2 fixes several bugs in 1.7.1. | Django 1.7.2 fixes several bugs in 1.7.1. | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six`, has | Additionally, Django's vendored version of six, ``django.utils.six``, has | ||||||
| been upgraded to the latest release (1.9.0). | been upgraded to the latest release (1.9.0). | ||||||
|  |  | ||||||
| Bugfixes | Bugfixes | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ Django 1.8.7 release notes | |||||||
|  |  | ||||||
| Django 1.8.7 fixes a security issue and several bugs in 1.8.6. | Django 1.8.7 fixes a security issue and several bugs in 1.8.6. | ||||||
|  |  | ||||||
| Additionally, Django's vendored version of six, :mod:`django.utils.six`, has | Additionally, Django's vendored version of six, ``django.utils.six``, has | ||||||
| been upgraded to the latest release (1.10.0). | been upgraded to the latest release (1.10.0). | ||||||
|  |  | ||||||
| Fixed settings leak possibility in ``date`` template filter | Fixed settings leak possibility in ``date`` template filter | ||||||
|   | |||||||
| @@ -1060,7 +1060,7 @@ authentication app:: | |||||||
|             # The user is identified by their email address |             # The user is identified by their email address | ||||||
|             return self.email |             return self.email | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.email |             return self.email | ||||||
|  |  | ||||||
|         def has_perm(self, perm, obj=None): |         def has_perm(self, perm, obj=None): | ||||||
|   | |||||||
| @@ -805,8 +805,7 @@ The basic interface is ``set(key, value, timeout)`` and ``get(key)``:: | |||||||
|     >>> cache.get('my_key') |     >>> cache.get('my_key') | ||||||
|     'hello, world!' |     'hello, world!' | ||||||
|  |  | ||||||
| ``key`` should be a ``str`` (or ``unicode`` on Python 2), and ``value`` can be | ``key`` should be a ``str``, and ``value`` can be any picklable Python object. | ||||||
| any picklable Python object. |  | ||||||
|  |  | ||||||
| The ``timeout`` argument is optional and defaults to the ``timeout`` argument | The ``timeout`` argument is optional and defaults to the ``timeout`` argument | ||||||
| of the appropriate backend in the :setting:`CACHES` setting (explained above). | of the appropriate backend in the :setting:`CACHES` setting (explained above). | ||||||
|   | |||||||
| @@ -87,7 +87,7 @@ We'll be using these models:: | |||||||
|         class Meta: |         class Meta: | ||||||
|             ordering = ["-name"] |             ordering = ["-name"] | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Author(models.Model): |     class Author(models.Model): | ||||||
| @@ -96,7 +96,7 @@ We'll be using these models:: | |||||||
|         email = models.EmailField() |         email = models.EmailField() | ||||||
|         headshot = models.ImageField(upload_to='author_headshots') |         headshot = models.ImageField(upload_to='author_headshots') | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Book(models.Model): |     class Book(models.Model): | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ objects, and a ``Publication`` has multiple ``Article`` objects: | |||||||
|     class Publication(models.Model): |     class Publication(models.Model): | ||||||
|         title = models.CharField(max_length=30) |         title = models.CharField(max_length=30) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.title |             return self.title | ||||||
|  |  | ||||||
|         class Meta: |         class Meta: | ||||||
| @@ -27,7 +27,7 @@ objects, and a ``Publication`` has multiple ``Article`` objects: | |||||||
|         headline = models.CharField(max_length=100) |         headline = models.CharField(max_length=100) | ||||||
|         publications = models.ManyToManyField(Publication) |         publications = models.ManyToManyField(Publication) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.headline |             return self.headline | ||||||
|  |  | ||||||
|         class Meta: |         class Meta: | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ To define a many-to-one relationship, use :class:`~django.db.models.ForeignKey`: | |||||||
|         last_name = models.CharField(max_length=30) |         last_name = models.CharField(max_length=30) | ||||||
|         email = models.EmailField() |         email = models.EmailField() | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return "%s %s" % (self.first_name, self.last_name) |             return "%s %s" % (self.first_name, self.last_name) | ||||||
|  |  | ||||||
|     class Article(models.Model): |     class Article(models.Model): | ||||||
| @@ -19,7 +19,7 @@ To define a many-to-one relationship, use :class:`~django.db.models.ForeignKey`: | |||||||
|         pub_date = models.DateField() |         pub_date = models.DateField() | ||||||
|         reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE) |         reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.headline |             return self.headline | ||||||
|  |  | ||||||
|         class Meta: |         class Meta: | ||||||
| @@ -64,13 +64,6 @@ Article objects have access to their related Reporter objects:: | |||||||
|  |  | ||||||
|     >>> r = a.reporter |     >>> r = a.reporter | ||||||
|  |  | ||||||
| On Python 2, these are strings of type ``str`` instead of unicode strings |  | ||||||
| because that's what was used in the creation of this reporter (and we haven't |  | ||||||
| refreshed the data from the database, which always returns unicode strings):: |  | ||||||
|  |  | ||||||
|     >>> r.first_name, r.last_name |  | ||||||
|     ('John', 'Smith') |  | ||||||
|  |  | ||||||
| Create an Article via the Reporter object:: | Create an Article via the Reporter object:: | ||||||
|  |  | ||||||
|     >>> new_article = r.article_set.create(headline="John's second story", pub_date=date(2005, 7, 29)) |     >>> new_article = r.article_set.create(headline="John's second story", pub_date=date(2005, 7, 29)) | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ In this example, a ``Place`` optionally can be a ``Restaurant``:: | |||||||
|         name = models.CharField(max_length=50) |         name = models.CharField(max_length=50) | ||||||
|         address = models.CharField(max_length=80) |         address = models.CharField(max_length=80) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return "%s the place" % self.name |             return "%s the place" % self.name | ||||||
|  |  | ||||||
|     class Restaurant(models.Model): |     class Restaurant(models.Model): | ||||||
| @@ -25,14 +25,14 @@ In this example, a ``Place`` optionally can be a ``Restaurant``:: | |||||||
|         serves_hot_dogs = models.BooleanField(default=False) |         serves_hot_dogs = models.BooleanField(default=False) | ||||||
|         serves_pizza = models.BooleanField(default=False) |         serves_pizza = models.BooleanField(default=False) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return "%s the restaurant" % self.place.name |             return "%s the restaurant" % self.place.name | ||||||
|  |  | ||||||
|     class Waiter(models.Model): |     class Waiter(models.Model): | ||||||
|         restaurant = models.ForeignKey(Restaurant, on_delete=models.CASCADE) |         restaurant = models.ForeignKey(Restaurant, on_delete=models.CASCADE) | ||||||
|         name = models.CharField(max_length=50) |         name = models.CharField(max_length=50) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return "%s the waiter at %s" % (self.name, self.restaurant) |             return "%s the waiter at %s" % (self.name, self.restaurant) | ||||||
|  |  | ||||||
| What follows are examples of operations that can be performed using the Python | What follows are examples of operations that can be performed using the Python | ||||||
|   | |||||||
| @@ -445,14 +445,14 @@ something like this:: | |||||||
|     class Person(models.Model): |     class Person(models.Model): | ||||||
|         name = models.CharField(max_length=128) |         name = models.CharField(max_length=128) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Group(models.Model): |     class Group(models.Model): | ||||||
|         name = models.CharField(max_length=128) |         name = models.CharField(max_length=128) | ||||||
|         members = models.ManyToManyField(Person, through='Membership') |         members = models.ManyToManyField(Person, through='Membership') | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Membership(models.Model): |     class Membership(models.Model): | ||||||
| @@ -761,7 +761,7 @@ of :ref:`methods automatically given to each model <model-instance-methods>`. | |||||||
| You can override most of these -- see `overriding predefined model methods`_, | You can override most of these -- see `overriding predefined model methods`_, | ||||||
| below -- but there are a couple that you'll almost always want to define: | below -- but there are a couple that you'll almost always want to define: | ||||||
|  |  | ||||||
| :meth:`~Model.__str__` (Python 3) | :meth:`~Model.__str__` | ||||||
|     A Python "magic method" that returns a unicode "representation" of any |     A Python "magic method" that returns a unicode "representation" of any | ||||||
|     object. This is what Python and Django will use whenever a model |     object. This is what Python and Django will use whenever a model | ||||||
|     instance needs to be coerced and displayed as a plain string. Most |     instance needs to be coerced and displayed as a plain string. Most | ||||||
| @@ -771,9 +771,6 @@ below -- but there are a couple that you'll almost always want to define: | |||||||
|     You'll always want to define this method; the default isn't very helpful |     You'll always want to define this method; the default isn't very helpful | ||||||
|     at all. |     at all. | ||||||
|  |  | ||||||
| ``__unicode__()`` (Python 2) |  | ||||||
|     Python 2 equivalent of ``__str__()``. |  | ||||||
|  |  | ||||||
| :meth:`~Model.get_absolute_url` | :meth:`~Model.get_absolute_url` | ||||||
|     This tells Django how to calculate the URL for an object. Django uses |     This tells Django how to calculate the URL for an object. Django uses | ||||||
|     this in its admin interface, and any time it needs to figure out a URL |     this in its admin interface, and any time it needs to figure out a URL | ||||||
|   | |||||||
| @@ -23,14 +23,14 @@ models, which comprise a Weblog application: | |||||||
|         name = models.CharField(max_length=100) |         name = models.CharField(max_length=100) | ||||||
|         tagline = models.TextField() |         tagline = models.TextField() | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Author(models.Model): |     class Author(models.Model): | ||||||
|         name = models.CharField(max_length=200) |         name = models.CharField(max_length=200) | ||||||
|         email = models.EmailField() |         email = models.EmailField() | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Entry(models.Model): |     class Entry(models.Model): | ||||||
| @@ -44,7 +44,7 @@ models, which comprise a Weblog application: | |||||||
|         n_pingbacks = models.IntegerField() |         n_pingbacks = models.IntegerField() | ||||||
|         rating = models.IntegerField() |         rating = models.IntegerField() | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.headline |             return self.headline | ||||||
|  |  | ||||||
| Creating objects | Creating objects | ||||||
|   | |||||||
| @@ -169,7 +169,7 @@ Consider this set of models:: | |||||||
|         title = models.CharField(max_length=3, choices=TITLE_CHOICES) |         title = models.CharField(max_length=3, choices=TITLE_CHOICES) | ||||||
|         birth_date = models.DateField(blank=True, null=True) |         birth_date = models.DateField(blank=True, null=True) | ||||||
|  |  | ||||||
|         def __str__(self):              # __unicode__ on Python 2 |         def __str__(self): | ||||||
|             return self.name |             return self.name | ||||||
|  |  | ||||||
|     class Book(models.Model): |     class Book(models.Model): | ||||||
|   | |||||||
| @@ -431,9 +431,6 @@ strings before passing them to non-Django code:: | |||||||
|  |  | ||||||
|     requests.post('https://example.com/send', data={'body': str(body)}) |     requests.post('https://example.com/send', data={'body': str(body)}) | ||||||
|  |  | ||||||
| Use ``unicode`` in place of ``str`` on Python 2, or :data:`six.text_type` to |  | ||||||
| support Python 2 and 3. |  | ||||||
|  |  | ||||||
| If you try to use a ``ugettext_lazy()`` result where a bytestring (a | If you try to use a ``ugettext_lazy()`` result where a bytestring (a | ||||||
| :class:`bytes` object) is expected, things won't work as expected since a | :class:`bytes` object) is expected, things won't work as expected since a | ||||||
| ``ugettext_lazy()`` object doesn't know how to convert itself to a bytestring. | ``ugettext_lazy()`` object doesn't know how to convert itself to a bytestring. | ||||||
| @@ -534,12 +531,11 @@ For any other case where you would like to delay the translation, but have to | |||||||
| pass the translatable string as argument to another function, you can wrap | pass the translatable string as argument to another function, you can wrap | ||||||
| this function inside a lazy call yourself. For example:: | this function inside a lazy call yourself. For example:: | ||||||
|  |  | ||||||
|     from django.utils import six  # Python 3 compatibility |  | ||||||
|     from django.utils.functional import lazy |     from django.utils.functional import lazy | ||||||
|     from django.utils.safestring import mark_safe |     from django.utils.safestring import mark_safe | ||||||
|     from django.utils.translation import ugettext_lazy as _ |     from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
|     mark_safe_lazy = lazy(mark_safe, six.text_type) |     mark_safe_lazy = lazy(mark_safe, str) | ||||||
|  |  | ||||||
| And then later:: | And then later:: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,7 +24,6 @@ Introductions to all the key parts of Django you'll need to know: | |||||||
|    i18n/index |    i18n/index | ||||||
|    logging |    logging | ||||||
|    pagination |    pagination | ||||||
|    python3 |  | ||||||
|    security |    security | ||||||
|    performance |    performance | ||||||
|    serialization |    serialization | ||||||
|   | |||||||
| @@ -171,7 +171,7 @@ This is the recommended way to install Django. | |||||||
|    isolated Python environments, which are more practical than installing |    isolated Python environments, which are more practical than installing | ||||||
|    packages systemwide. They also allow installing packages without |    packages systemwide. They also allow installing packages without | ||||||
|    administrator privileges. The :doc:`contributing tutorial |    administrator privileges. The :doc:`contributing tutorial | ||||||
|    </intro/contributing>` walks through how to create a virtualenv on Python 3. |    </intro/contributing>` walks through how to create a virtualenv. | ||||||
|  |  | ||||||
| 3. After you've created and activated a virtual environment, enter the command | 3. After you've created and activated a virtual environment, enter the command | ||||||
|    ``pip install Django`` at the shell prompt. |    ``pip install Django`` at the shell prompt. | ||||||
| @@ -227,7 +227,7 @@ latest bug fixes and improvements, follow these instructions: | |||||||
| 3. Make sure that the Python interpreter can load Django's code. The most | 3. Make sure that the Python interpreter can load Django's code. The most | ||||||
|    convenient way to do this is to use virtualenv_, virtualenvwrapper_, and |    convenient way to do this is to use virtualenv_, virtualenvwrapper_, and | ||||||
|    pip_. The :doc:`contributing tutorial </intro/contributing>` walks through |    pip_. The :doc:`contributing tutorial </intro/contributing>` walks through | ||||||
|    how to create a virtualenv on Python 3. |    how to create a virtualenv. | ||||||
|  |  | ||||||
| 4. After setting up and activating the virtualenv, run the following command: | 4. After setting up and activating the virtualenv, run the following command: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -665,6 +665,7 @@ Django can serialize the following: | |||||||
| - ``LazyObject`` instances which wrap a serializable value. | - ``LazyObject`` instances which wrap a serializable value. | ||||||
| - Any Django field | - Any Django field | ||||||
| - Any function or method reference (e.g. ``datetime.datetime.today``) (must be in module's top-level scope) | - Any function or method reference (e.g. ``datetime.datetime.today``) (must be in module's top-level scope) | ||||||
|  | - Unbound methods used from within the class body | ||||||
| - Any class reference (must be in module's top-level scope) | - Any class reference (must be in module's top-level scope) | ||||||
| - Anything with a custom ``deconstruct()`` method (:ref:`see below <custom-deconstruct-method>`) | - Anything with a custom ``deconstruct()`` method (:ref:`see below <custom-deconstruct-method>`) | ||||||
|  |  | ||||||
| @@ -672,31 +673,12 @@ Django can serialize the following: | |||||||
|  |  | ||||||
|     Serialization support for ``uuid.UUID`` was added. |     Serialization support for ``uuid.UUID`` was added. | ||||||
|  |  | ||||||
| Django can serialize the following on Python 3 only: |  | ||||||
|  |  | ||||||
| - Unbound methods used from within the class body (see below) |  | ||||||
|  |  | ||||||
| Django cannot serialize: | Django cannot serialize: | ||||||
|  |  | ||||||
| - Nested classes | - Nested classes | ||||||
| - Arbitrary class instances (e.g. ``MyClass(4.3, 5.7)``) | - Arbitrary class instances (e.g. ``MyClass(4.3, 5.7)``) | ||||||
| - Lambdas | - Lambdas | ||||||
|  |  | ||||||
| Due to the fact ``__qualname__`` was only introduced in Python 3, Django can only |  | ||||||
| serialize the following pattern (an unbound method used within the class body) |  | ||||||
| on Python 3, and will fail to serialize a reference to it on Python 2:: |  | ||||||
|  |  | ||||||
|     class MyModel(models.Model): |  | ||||||
|  |  | ||||||
|         def upload_to(self): |  | ||||||
|             return "something dynamic" |  | ||||||
|  |  | ||||||
|         my_file = models.FileField(upload_to=upload_to) |  | ||||||
|  |  | ||||||
| If you are using Python 2, we recommend you move your methods for upload_to |  | ||||||
| and similar arguments that accept callables (e.g. ``default``) to live in |  | ||||||
| the main module body, rather than the class body. |  | ||||||
|  |  | ||||||
| .. _custom-deconstruct-method: | .. _custom-deconstruct-method: | ||||||
|  |  | ||||||
| Adding a ``deconstruct()`` method | Adding a ``deconstruct()`` method | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ page:: | |||||||
|     4 |     4 | ||||||
|     >>> p.num_pages |     >>> p.num_pages | ||||||
|     2 |     2 | ||||||
|     >>> type(p.page_range)  # `<type 'rangeiterator'>` in Python 2. |     >>> type(p.page_range) | ||||||
|     <class 'range_iterator'> |     <class 'range_iterator'> | ||||||
|     >>> p.page_range |     >>> p.page_range | ||||||
|     range(1, 3) |     range(1, 3) | ||||||
|   | |||||||
| @@ -1,369 +0,0 @@ | |||||||
| =================== |  | ||||||
| Porting to Python 3 |  | ||||||
| =================== |  | ||||||
|  |  | ||||||
| Django 1.5 is the first version of Django to support Python 3. The same code |  | ||||||
| runs both on Python 2 (≥ 2.6.5) and Python 3 (≥ 3.2), thanks to the six_ |  | ||||||
| compatibility layer. |  | ||||||
|  |  | ||||||
| .. _six: https://pythonhosted.org/six/ |  | ||||||
|  |  | ||||||
| This document is primarily targeted at authors of pluggable applications |  | ||||||
| who want to support both Python 2 and 3. It also describes guidelines that |  | ||||||
| apply to Django's code. |  | ||||||
|  |  | ||||||
| Philosophy |  | ||||||
| ========== |  | ||||||
|  |  | ||||||
| This document assumes that you are familiar with the changes between Python 2 |  | ||||||
| and Python 3. If you aren't, read :ref:`Python's official porting guide |  | ||||||
| <pyporting-howto>` first. Refreshing your knowledge of unicode handling on |  | ||||||
| Python 2 and 3 will help; the `Pragmatic Unicode`_ presentation is a good |  | ||||||
| resource. |  | ||||||
|  |  | ||||||
| Django uses the *Python 2/3 Compatible Source* strategy. Of course, you're |  | ||||||
| free to chose another strategy for your own code, especially if you don't need |  | ||||||
| to stay compatible with Python 2. But authors of pluggable applications are |  | ||||||
| encouraged to use the same porting strategy as Django itself. |  | ||||||
|  |  | ||||||
| Writing compatible code is much easier if you target Python ≥ 2.6. Django 1.5 |  | ||||||
| introduces compatibility tools such as :mod:`django.utils.six`, which is a |  | ||||||
| customized version of the :mod:`six module <six>`. For convenience, |  | ||||||
| forwards-compatible aliases were introduced in Django 1.4.2. If your |  | ||||||
| application takes advantage of these tools, it will require Django ≥ 1.4.2. |  | ||||||
|  |  | ||||||
| Obviously, writing compatible source code adds some overhead, and that can |  | ||||||
| cause frustration. Django's developers have found that attempting to write |  | ||||||
| Python 3 code that's compatible with Python 2 is much more rewarding than the |  | ||||||
| opposite. Not only does that make your code more future-proof, but Python 3's |  | ||||||
| advantages (like the saner string handling) start shining quickly. Dealing |  | ||||||
| with Python 2 becomes a backwards compatibility requirement, and we as |  | ||||||
| developers are used to dealing with such constraints. |  | ||||||
|  |  | ||||||
| Porting tools provided by Django are inspired by this philosophy, and it's |  | ||||||
| reflected throughout this guide. |  | ||||||
|  |  | ||||||
| .. _Pragmatic Unicode: http://nedbatchelder.com/text/unipain.html |  | ||||||
|  |  | ||||||
| Porting tips |  | ||||||
| ============ |  | ||||||
|  |  | ||||||
| Unicode literals |  | ||||||
| ---------------- |  | ||||||
|  |  | ||||||
| This step consists in: |  | ||||||
|  |  | ||||||
| - Adding ``from __future__ import unicode_literals`` at the top of your Python |  | ||||||
|   modules -- it's best to put it in each and every module, otherwise you'll |  | ||||||
|   keep checking the top of your files to see which mode is in effect; |  | ||||||
| - Removing the ``u`` prefix before unicode strings; |  | ||||||
| - Adding a ``b`` prefix before bytestrings. |  | ||||||
|  |  | ||||||
| Performing these changes systematically guarantees backwards compatibility. |  | ||||||
|  |  | ||||||
| However, Django applications generally don't need bytestrings, since Django |  | ||||||
| only exposes unicode interfaces to the programmer. Python 3 discourages using |  | ||||||
| bytestrings, except for binary data or byte-oriented interfaces. Python 2 |  | ||||||
| makes bytestrings and unicode strings effectively interchangeable, as long as |  | ||||||
| they only contain ASCII data. Take advantage of this to use unicode strings |  | ||||||
| wherever possible and avoid the ``b`` prefixes. |  | ||||||
|  |  | ||||||
| .. note:: |  | ||||||
|  |  | ||||||
|     Python 2's ``u`` prefix is a syntax error in Python 3.2 but it will be |  | ||||||
|     allowed again in Python 3.3 thanks to :pep:`414`. Thus, this |  | ||||||
|     transformation is optional if you target Python ≥ 3.3. It's still |  | ||||||
|     recommended, per the "write Python 3 code" philosophy. |  | ||||||
|  |  | ||||||
| String handling |  | ||||||
| --------------- |  | ||||||
|  |  | ||||||
| Python 2's `unicode`_ type was renamed :class:`str` in Python 3, |  | ||||||
| ``str()`` was renamed :class:`bytes`, and `basestring`_ disappeared. |  | ||||||
| six_ provides :ref:`tools <string-handling-with-six>` to deal with these |  | ||||||
| changes. |  | ||||||
|  |  | ||||||
| Django also contains several string related classes and functions in the |  | ||||||
| :mod:`django.utils.encoding` and :mod:`django.utils.safestring` modules. Their |  | ||||||
| names used the words ``str``, which doesn't mean the same thing in Python 2 |  | ||||||
| and Python 3, and ``unicode``, which doesn't exist in Python 3. In order to |  | ||||||
| avoid ambiguity and confusion these concepts were renamed ``bytes`` and |  | ||||||
| ``text``. |  | ||||||
|  |  | ||||||
| Here are the name changes in :mod:`django.utils.encoding`: |  | ||||||
|  |  | ||||||
| ==================  ================== |  | ||||||
| Old name            New name |  | ||||||
| ==================  ================== |  | ||||||
| ``smart_str``       ``smart_bytes`` |  | ||||||
| ``smart_unicode``   ``smart_text`` |  | ||||||
| ``force_unicode``   ``force_text`` |  | ||||||
| ==================  ================== |  | ||||||
|  |  | ||||||
| For backwards compatibility, the old names still work on Python 2. Under |  | ||||||
| Python 3, ``smart_str`` is an alias for ``smart_text``. |  | ||||||
|  |  | ||||||
| For forwards compatibility, the new names work as of Django 1.4.2. |  | ||||||
|  |  | ||||||
| .. note:: |  | ||||||
|  |  | ||||||
|     :mod:`django.utils.encoding` was deeply refactored in Django 1.5 to |  | ||||||
|     provide a more consistent API. Check its documentation for more |  | ||||||
|     information. |  | ||||||
|  |  | ||||||
| :mod:`django.utils.safestring` is mostly used via the |  | ||||||
| :func:`~django.utils.safestring.mark_safe` function, which didn't change. In |  | ||||||
| case you're using the internals, here are the name changes: |  | ||||||
|  |  | ||||||
| ==================  ================== |  | ||||||
| Old name            New name |  | ||||||
| ==================  ================== |  | ||||||
| ``SafeString``      ``SafeBytes`` |  | ||||||
| ``SafeUnicode``     ``SafeText`` |  | ||||||
| ==================  ================== |  | ||||||
|  |  | ||||||
| For backwards compatibility, the old names still work on Python 2. On Python 3, |  | ||||||
| ``SafeString`` is an alias for ``SafeText``. |  | ||||||
|  |  | ||||||
| For forwards compatibility, the new names work as of Django 1.4.2. |  | ||||||
|  |  | ||||||
| ``__str__()`` and ``__unicode__()`` methods |  | ||||||
| ------------------------------------------- |  | ||||||
|  |  | ||||||
| In Python 2, the object model specifies :meth:`~object.__str__` and |  | ||||||
| ` __unicode__()`_ methods. If these methods exist, they must return |  | ||||||
| ``str`` (bytes) and ``unicode`` (text) respectively. |  | ||||||
|  |  | ||||||
| The ``print`` statement and the :class:`str` built-in call |  | ||||||
| :meth:`~object.__str__` to determine the human-readable representation of an |  | ||||||
| object. The ``unicode`` built-in calls ` __unicode__()`_ if it |  | ||||||
| exists, and otherwise falls back to :meth:`~object.__str__` and decodes the |  | ||||||
| result with the system encoding. Conversely, the |  | ||||||
| :class:`~django.db.models.Model` base class automatically derives |  | ||||||
| :meth:`~object.__str__` from ` __unicode__()`_ by encoding to UTF-8. |  | ||||||
|  |  | ||||||
| In Python 3, there's simply :meth:`~object.__str__`, which must return ``str`` |  | ||||||
| (text). |  | ||||||
|  |  | ||||||
| (It is also possible to define :meth:`~object.__bytes__`, but Django applications |  | ||||||
| have little use for that method, because they hardly ever deal with ``bytes``.) |  | ||||||
|  |  | ||||||
| Finally, note that :meth:`~object.__repr__` must return a ``str`` on all |  | ||||||
| versions of Python. |  | ||||||
|  |  | ||||||
| :class:`dict` and :class:`dict`-like classes |  | ||||||
| -------------------------------------------- |  | ||||||
|  |  | ||||||
| :meth:`dict.keys`, :meth:`dict.items` and :meth:`dict.values` return lists in |  | ||||||
| Python 2 and iterators in Python 3. :class:`~django.http.QueryDict` and the |  | ||||||
| :class:`dict`-like classes defined in ``django.utils.datastructures`` |  | ||||||
| behave likewise in Python 3. |  | ||||||
|  |  | ||||||
| six_ provides compatibility functions to work around this change: |  | ||||||
| :func:`~six.iterkeys`, :func:`~six.iteritems`, and :func:`~six.itervalues`. |  | ||||||
| It also contains an undocumented ``iterlists`` function that works well for |  | ||||||
| ``django.utils.datastructures.MultiValueDict`` and its subclasses. |  | ||||||
|  |  | ||||||
| :class:`~django.http.HttpRequest` and :class:`~django.http.HttpResponse` objects |  | ||||||
| -------------------------------------------------------------------------------- |  | ||||||
|  |  | ||||||
| According to :pep:`3333`: |  | ||||||
|  |  | ||||||
| - headers are always ``str`` objects, |  | ||||||
| - input and output streams are always ``bytes`` objects. |  | ||||||
|  |  | ||||||
| Specifically, :attr:`HttpResponse.content <django.http.HttpResponse.content>` |  | ||||||
| contains ``bytes``, which may become an issue if you compare it with a |  | ||||||
| ``str`` in your tests. The preferred solution is to rely on |  | ||||||
| :meth:`~django.test.SimpleTestCase.assertContains` and |  | ||||||
| :meth:`~django.test.SimpleTestCase.assertNotContains`. These methods accept a |  | ||||||
| response and a unicode string as arguments. |  | ||||||
|  |  | ||||||
| Coding guidelines |  | ||||||
| ================= |  | ||||||
|  |  | ||||||
| The following guidelines are enforced in Django's source code. They're also |  | ||||||
| recommended for third-party applications that follow the same porting strategy. |  | ||||||
|  |  | ||||||
| Syntax requirements |  | ||||||
| ------------------- |  | ||||||
|  |  | ||||||
| Unicode |  | ||||||
| ~~~~~~~ |  | ||||||
|  |  | ||||||
| In Python 3, all strings are considered Unicode by default. The ``unicode`` |  | ||||||
| type from Python 2 is called ``str`` in Python 3, and ``str`` becomes |  | ||||||
| ``bytes``. |  | ||||||
|  |  | ||||||
| You mustn't use the ``u`` prefix before a unicode string literal because it's |  | ||||||
| a syntax error in Python 3.2. You must prefix byte strings with ``b``. |  | ||||||
|  |  | ||||||
| In order to enable the same behavior in Python 2, every module must import |  | ||||||
| ``unicode_literals`` from ``__future__``:: |  | ||||||
|  |  | ||||||
|     from __future__ import unicode_literals |  | ||||||
|  |  | ||||||
|     my_string = "This is an unicode literal" |  | ||||||
|     my_bytestring = b"This is a bytestring" |  | ||||||
|  |  | ||||||
| If you need a byte string literal under Python 2 and a unicode string literal |  | ||||||
| under Python 3, use the :class:`str` builtin:: |  | ||||||
|  |  | ||||||
|     str('my string') |  | ||||||
|  |  | ||||||
| In Python 3, there aren't any automatic conversions between ``str`` and |  | ||||||
| ``bytes``, and the :mod:`codecs` module became more strict. :meth:`str.encode` |  | ||||||
| always returns ``bytes``, and ``bytes.decode`` always returns ``str``. As a |  | ||||||
| consequence, the following pattern is sometimes necessary:: |  | ||||||
|  |  | ||||||
|     value = value.encode('ascii', 'ignore').decode('ascii') |  | ||||||
|  |  | ||||||
| Be cautious if you have to `index bytestrings`_. |  | ||||||
|  |  | ||||||
| .. _index bytestrings: https://docs.python.org/3/howto/pyporting.html#text-versus-binary-data |  | ||||||
|  |  | ||||||
| Exceptions |  | ||||||
| ~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| When you capture exceptions, use the ``as`` keyword:: |  | ||||||
|  |  | ||||||
|     try: |  | ||||||
|         ... |  | ||||||
|     except MyException as exc: |  | ||||||
|         ... |  | ||||||
|  |  | ||||||
| This older syntax was removed in Python 3:: |  | ||||||
|  |  | ||||||
|     try: |  | ||||||
|         ... |  | ||||||
|     except MyException, exc:    # Don't do that! |  | ||||||
|         ... |  | ||||||
|  |  | ||||||
| The syntax to reraise an exception with a different traceback also changed. |  | ||||||
| Use :func:`six.reraise`. |  | ||||||
|  |  | ||||||
| Magic methods |  | ||||||
| ------------- |  | ||||||
|  |  | ||||||
| Use the patterns below to handle magic methods renamed in Python 3. |  | ||||||
|  |  | ||||||
| Iterators |  | ||||||
| ~~~~~~~~~ |  | ||||||
|  |  | ||||||
| :: |  | ||||||
|  |  | ||||||
|     class MyIterator(six.Iterator): |  | ||||||
|         def __iter__(self): |  | ||||||
|             return self             # implement some logic here |  | ||||||
|  |  | ||||||
|         def __next__(self): |  | ||||||
|             raise StopIteration     # implement some logic here |  | ||||||
|  |  | ||||||
| Boolean evaluation |  | ||||||
| ~~~~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| :: |  | ||||||
|  |  | ||||||
|     class MyBoolean(object): |  | ||||||
|  |  | ||||||
|         def __bool__(self): |  | ||||||
|             return True             # implement some logic here |  | ||||||
|  |  | ||||||
|         def __nonzero__(self):      # Python 2 compatibility |  | ||||||
|             return type(self).__bool__(self) |  | ||||||
|  |  | ||||||
| Division |  | ||||||
| ~~~~~~~~ |  | ||||||
|  |  | ||||||
| :: |  | ||||||
|  |  | ||||||
|     class MyDivisible(object): |  | ||||||
|  |  | ||||||
|         def __truediv__(self, other): |  | ||||||
|             return self / other     # implement some logic here |  | ||||||
|  |  | ||||||
|         def __div__(self, other):   # Python 2 compatibility |  | ||||||
|             return type(self).__truediv__(self, other) |  | ||||||
|  |  | ||||||
|         def __itruediv__(self, other): |  | ||||||
|             return self // other    # implement some logic here |  | ||||||
|  |  | ||||||
|         def __idiv__(self, other):  # Python 2 compatibility |  | ||||||
|             return type(self).__itruediv__(self, other) |  | ||||||
|  |  | ||||||
| Special methods are looked up on the class and not on the instance to reflect |  | ||||||
| the behavior of the Python interpreter. |  | ||||||
|  |  | ||||||
| .. module: django.utils.six |  | ||||||
|  |  | ||||||
| Writing compatible code with six |  | ||||||
| -------------------------------- |  | ||||||
|  |  | ||||||
| six_ is the canonical compatibility library for supporting Python 2 and 3 in |  | ||||||
| a single codebase. Read its documentation! |  | ||||||
|  |  | ||||||
| A :mod:`customized version of six <django.utils.six>` is bundled with Django |  | ||||||
| as of version 1.4.2. You can import it as ``django.utils.six``. |  | ||||||
|  |  | ||||||
| Here are the most common changes required to write compatible code. |  | ||||||
|  |  | ||||||
| .. _string-handling-with-six: |  | ||||||
|  |  | ||||||
| String handling |  | ||||||
| ~~~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| The ``basestring`` and ``unicode`` types were removed in Python 3, and the |  | ||||||
| meaning of ``str`` changed. To test these types, use the following idioms:: |  | ||||||
|  |  | ||||||
|     isinstance(myvalue, six.string_types)       # replacement for basestring |  | ||||||
|     isinstance(myvalue, six.text_type)          # replacement for unicode |  | ||||||
|     isinstance(myvalue, bytes)                  # replacement for str |  | ||||||
|  |  | ||||||
| Python ≥ 2.6 provides ``bytes`` as an alias for ``str``, so you don't need |  | ||||||
| :data:`six.binary_type`. |  | ||||||
|  |  | ||||||
| ``long`` |  | ||||||
| ~~~~~~~~ |  | ||||||
|  |  | ||||||
| The ``long`` type no longer exists in Python 3. ``1L`` is a syntax error. Use |  | ||||||
| :data:`six.integer_types` check if a value is an integer or a long:: |  | ||||||
|  |  | ||||||
|     isinstance(myvalue, six.integer_types)      # replacement for (int, long) |  | ||||||
|  |  | ||||||
| ``xrange`` |  | ||||||
| ~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| If you use ``xrange`` on Python 2, import ``six.moves.range`` and use that |  | ||||||
| instead. You can also import ``six.moves.xrange`` (it's equivalent to |  | ||||||
| ``six.moves.range``) but the first technique allows you to simply drop the |  | ||||||
| import when dropping support for Python 2. |  | ||||||
|  |  | ||||||
| Moved modules |  | ||||||
| ~~~~~~~~~~~~~ |  | ||||||
|  |  | ||||||
| Some modules were renamed in Python 3. The ``django.utils.six.moves`` |  | ||||||
| module (based on the :mod:`six.moves module <six.moves>`) provides a |  | ||||||
| compatible location to import them. |  | ||||||
|  |  | ||||||
| ``PY2`` |  | ||||||
| ~~~~~~~ |  | ||||||
|  |  | ||||||
| If you need different code in Python 2 and Python 3, check :data:`six.PY2`:: |  | ||||||
|  |  | ||||||
|     if six.PY2: |  | ||||||
|         # compatibility code for Python 2 |  | ||||||
|  |  | ||||||
| This is a last resort solution when :mod:`six` doesn't provide an appropriate |  | ||||||
| function. |  | ||||||
|  |  | ||||||
| .. module:: django.utils.six |  | ||||||
|  |  | ||||||
| Django customized version of ``six`` |  | ||||||
| ------------------------------------ |  | ||||||
|  |  | ||||||
| The version of six bundled with Django (``django.utils.six``) includes a few |  | ||||||
| customizations for internal use only. |  | ||||||
|  |  | ||||||
| .. _unicode: https://docs.python.org/2/library/functions.html#unicode |  | ||||||
| .. _ __unicode__(): https://docs.python.org/2/reference/datamodel.html#object.__unicode__ |  | ||||||
| .. _basestring: https://docs.python.org/2/library/functions.html#basestring |  | ||||||
| @@ -177,9 +177,9 @@ control the particular collation used by the test database. See the | |||||||
| :doc:`settings documentation </ref/settings>` for details of these | :doc:`settings documentation </ref/settings>` for details of these | ||||||
| and other advanced settings. | and other advanced settings. | ||||||
|  |  | ||||||
| If using an SQLite in-memory database with Python 3.4+ and SQLite 3.7.13+, | If using an SQLite in-memory database with SQLite 3.7.13+, `shared cache | ||||||
| `shared cache <https://www.sqlite.org/sharedcache.html>`_ will be enabled, so | <https://www.sqlite.org/sharedcache.html>`_ is enabled, so you can write tests | ||||||
| you can write tests with ability to share the database between threads. | with ability to share the database between threads. | ||||||
|  |  | ||||||
| .. admonition:: Finding data from your production database when running tests? | .. admonition:: Finding data from your production database when running tests? | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1708,9 +1708,9 @@ Management commands can be tested with the | |||||||
| :func:`~django.core.management.call_command` function. The output can be | :func:`~django.core.management.call_command` function. The output can be | ||||||
| redirected into a ``StringIO`` instance:: | redirected into a ``StringIO`` instance:: | ||||||
|  |  | ||||||
|  |     from io import StringIO | ||||||
|     from django.core.management import call_command |     from django.core.management import call_command | ||||||
|     from django.test import TestCase |     from django.test import TestCase | ||||||
|     from django.utils.six import StringIO |  | ||||||
|  |  | ||||||
|     class ClosepollTest(TestCase): |     class ClosepollTest(TestCase): | ||||||
|         def test_command_output(self): |         def test_command_output(self): | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user