mirror of
				https://github.com/django/django.git
				synced 2025-10-26 07:06:08 +00:00 
			
		
		
		
	[4.2.x] Refs #34140 -- Corrected rst code-block and various formatting issues in docs.
Backport of ba755ca131 from main
			
			
This commit is contained in:
		
				
					committed by
					
						 Mariusz Felisiak
						Mariusz Felisiak
					
				
			
			
				
	
			
			
			
						parent
						
							fae76b81ce
						
					
				
				
					commit
					5bdd6223a2
				
			| @@ -51,7 +51,7 @@ Lookup registration can also be done using a decorator pattern:: | |||||||
|  |  | ||||||
|     @Field.register_lookup |     @Field.register_lookup | ||||||
|     class NotEqualLookup(Lookup): |     class NotEqualLookup(Lookup): | ||||||
|         # ... |         ... | ||||||
|  |  | ||||||
| We can now use ``foo__ne`` for any field ``foo``. You will need to ensure that | We can now use ``foo__ne`` for any field ``foo``. You will need to ensure that | ||||||
| this registration happens before you try to create any querysets using it. You | this registration happens before you try to create any querysets using it. You | ||||||
|   | |||||||
| @@ -117,7 +117,7 @@ file ``general.log`` (at the project root): | |||||||
|     :emphasize-lines: 3-8 |     :emphasize-lines: 3-8 | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'handlers': { |         'handlers': { | ||||||
|             'file': { |             'file': { | ||||||
|                 'class': 'logging.FileHandler', |                 'class': 'logging.FileHandler', | ||||||
| @@ -156,7 +156,7 @@ example: | |||||||
|     :emphasize-lines: 3-8 |     :emphasize-lines: 3-8 | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'loggers': { |         'loggers': { | ||||||
|             '': { |             '': { | ||||||
|                 'level': 'DEBUG', |                 'level': 'DEBUG', | ||||||
| @@ -195,7 +195,7 @@ formatters named ``verbose`` and ``simple``: | |||||||
|     :emphasize-lines: 3-12 |     :emphasize-lines: 3-12 | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'formatters': { |         'formatters': { | ||||||
|             'verbose': { |             'verbose': { | ||||||
|                 'format': '{name} {levelname} {asctime} {module} {process:d} {thread:d} {message}', |                 'format': '{name} {levelname} {asctime} {module} {process:d} {thread:d} {message}', | ||||||
| @@ -226,7 +226,7 @@ dictionary referring to the formatter by name, for example: | |||||||
|             'filename': 'general.log', |             'filename': 'general.log', | ||||||
|             'formatter': 'verbose', |             'formatter': 'verbose', | ||||||
|         }, |         }, | ||||||
|     }, |     } | ||||||
|  |  | ||||||
| .. _naming-loggers: | .. _naming-loggers: | ||||||
|  |  | ||||||
| @@ -253,7 +253,7 @@ A logger mapping named ``my_app.views`` will capture records from this logger: | |||||||
|     :emphasize-lines: 4 |     :emphasize-lines: 4 | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'loggers': { |         'loggers': { | ||||||
|             'my_app.views': { |             'my_app.views': { | ||||||
|                 ... |                 ... | ||||||
| @@ -269,7 +269,7 @@ from loggers anywhere within the ``my_app`` namespace (including | |||||||
|     :emphasize-lines: 4 |     :emphasize-lines: 4 | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'loggers': { |         'loggers': { | ||||||
|             'my_app': { |             'my_app': { | ||||||
|                 ... |                 ... | ||||||
| @@ -297,16 +297,16 @@ by a mapping for both ``my_app`` and ``my_app.views``. | |||||||
| To manage this behavior, set the propagation key on the mappings you define:: | To manage this behavior, set the propagation key on the mappings you define:: | ||||||
|  |  | ||||||
|     LOGGING = { |     LOGGING = { | ||||||
|         [...] |         # ... | ||||||
|         'loggers': { |         'loggers': { | ||||||
|             'my_app': { |             'my_app': { | ||||||
|                 [...] |                 # ... | ||||||
|             }, |             }, | ||||||
|             'my_app.views': { |             'my_app.views': { | ||||||
|                 [...] |                 # ... | ||||||
|             }, |             }, | ||||||
|             'my_app.views.private': { |             'my_app.views.private': { | ||||||
|                 [...] |                 # ... | ||||||
|                 'propagate': False, |                 'propagate': False, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|   | |||||||
| @@ -42,7 +42,7 @@ called ``blog``, which provides the templates ``blog/post.html`` and | |||||||
|             'BACKEND': 'django.template.backends.django.DjangoTemplates', |             'BACKEND': 'django.template.backends.django.DjangoTemplates', | ||||||
|             'DIRS': [BASE_DIR / 'templates'], |             'DIRS': [BASE_DIR / 'templates'], | ||||||
|             'APP_DIRS': True, |             'APP_DIRS': True, | ||||||
|             ... |             # ... | ||||||
|         }, |         }, | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
| @@ -77,9 +77,9 @@ First, make sure your template settings are checking inside app directories:: | |||||||
|  |  | ||||||
|     TEMPLATES = [ |     TEMPLATES = [ | ||||||
|         { |         { | ||||||
|             ..., |             # ... | ||||||
|             'APP_DIRS': True, |             'APP_DIRS': True, | ||||||
|             ... |             # ... | ||||||
|         }, |         }, | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -199,9 +199,10 @@ Imports | |||||||
|  |  | ||||||
|  |  | ||||||
|       class Example: |       class Example: | ||||||
|           # ... |           ... | ||||||
|  |  | ||||||
| * Use convenience imports whenever available. For example, do this:: | * Use convenience imports whenever available. For example, do this | ||||||
|  |   :: | ||||||
|  |  | ||||||
|       from django.views import View |       from django.views import View | ||||||
|  |  | ||||||
| @@ -236,12 +237,12 @@ View style | |||||||
|   Do this:: |   Do this:: | ||||||
|  |  | ||||||
|       def my_view(request, foo): |       def my_view(request, foo): | ||||||
|           # ... |           ... | ||||||
|  |  | ||||||
|   Don't do this:: |   Don't do this:: | ||||||
|  |  | ||||||
|       def my_view(req, foo): |       def my_view(req, foo): | ||||||
|           # ... |           ... | ||||||
|  |  | ||||||
| Model style | Model style | ||||||
| =========== | =========== | ||||||
|   | |||||||
| @@ -207,6 +207,7 @@ You can also add a test for the deprecation warning:: | |||||||
|         msg = 'Expected deprecation message' |         msg = 'Expected deprecation message' | ||||||
|         with self.assertWarnsMessage(RemovedInDjangoXXWarning, msg): |         with self.assertWarnsMessage(RemovedInDjangoXXWarning, msg): | ||||||
|             # invoke deprecated behavior |             # invoke deprecated behavior | ||||||
|  |             ... | ||||||
|  |  | ||||||
| Finally, there are a couple of updates to Django's documentation to make: | Finally, there are a couple of updates to Django's documentation to make: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -163,7 +163,7 @@ this. For a small app like polls, this process isn't too difficult. | |||||||
|        1. Add "polls" to your INSTALLED_APPS setting like this:: |        1. Add "polls" to your INSTALLED_APPS setting like this:: | ||||||
|  |  | ||||||
|            INSTALLED_APPS = [ |            INSTALLED_APPS = [ | ||||||
|                ... |                ..., | ||||||
|                'polls', |                'polls', | ||||||
|            ] |            ] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -115,7 +115,9 @@ and traverses the patterns in order. After finding the match at ``'polls/'``, | |||||||
| it strips off the matching text (``"polls/"``) and sends the remaining text -- | it strips off the matching text (``"polls/"``) and sends the remaining text -- | ||||||
| ``"34/"`` -- to the 'polls.urls' URLconf for further processing. There it | ``"34/"`` -- to the 'polls.urls' URLconf for further processing. There it | ||||||
| matches ``'<int:question_id>/'``, resulting in a call to the ``detail()`` view | matches ``'<int:question_id>/'``, resulting in a call to the ``detail()`` view | ||||||
| like so:: | like so: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     detail(request=<HttpRequest object>, question_id=34) |     detail(request=<HttpRequest object>, question_id=34) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -171,7 +171,7 @@ tabular way of displaying inline related objects. To use it, change the | |||||||
|     :caption: ``polls/admin.py`` |     :caption: ``polls/admin.py`` | ||||||
|  |  | ||||||
|     class ChoiceInline(admin.TabularInline): |     class ChoiceInline(admin.TabularInline): | ||||||
|         #... |         ... | ||||||
|  |  | ||||||
| With that ``TabularInline`` (instead of ``StackedInline``), the | With that ``TabularInline`` (instead of ``StackedInline``), the | ||||||
| related objects are displayed in a more compact, table-based format: | related objects are displayed in a more compact, table-based format: | ||||||
|   | |||||||
| @@ -76,9 +76,9 @@ Alternatively, :setting:`INSTALLED_APPS` may contain the dotted path to a | |||||||
| configuration class to specify it explicitly:: | configuration class to specify it explicitly:: | ||||||
|  |  | ||||||
|     INSTALLED_APPS = [ |     INSTALLED_APPS = [ | ||||||
|         ... |         ..., | ||||||
|         'polls.apps.PollsAppConfig', |         'polls.apps.PollsAppConfig', | ||||||
|         ... |         ..., | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
| For application authors | For application authors | ||||||
|   | |||||||
| @@ -58,9 +58,9 @@ To set the same ``X-Frame-Options`` value for all responses in your site, put | |||||||
| :setting:`MIDDLEWARE`:: | :setting:`MIDDLEWARE`:: | ||||||
|  |  | ||||||
|     MIDDLEWARE = [ |     MIDDLEWARE = [ | ||||||
|         ... |         ..., | ||||||
|         'django.middleware.clickjacking.XFrameOptionsMiddleware', |         'django.middleware.clickjacking.XFrameOptionsMiddleware', | ||||||
|         ... |         ..., | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
| This middleware is enabled in the settings file generated by | This middleware is enabled in the settings file generated by | ||||||
|   | |||||||
| @@ -722,6 +722,8 @@ subclass:: | |||||||
|       like:: |       like:: | ||||||
|  |  | ||||||
|           @admin.display(ordering='-first_name') |           @admin.display(ordering='-first_name') | ||||||
|  |           def colored_first_name(self): | ||||||
|  |               ... | ||||||
|  |  | ||||||
|       The ``ordering`` argument supports query lookups to sort by values on |       The ``ordering`` argument supports query lookups to sort by values on | ||||||
|       related models. This example includes an "author first name" column in |       related models. This example includes an "author first name" column in | ||||||
| @@ -752,7 +754,8 @@ subclass:: | |||||||
|               def full_name(self): |               def full_name(self): | ||||||
|                   return self.first_name + ' ' + self.last_name |                   return self.first_name + ' ' + self.last_name | ||||||
|  |  | ||||||
|     * Elements of ``list_display`` can also be properties:: |     * Elements of ``list_display`` can also be properties | ||||||
|  |       :: | ||||||
|  |  | ||||||
|           class Person(models.Model): |           class Person(models.Model): | ||||||
|               first_name = models.CharField(max_length=50) |               first_name = models.CharField(max_length=50) | ||||||
| @@ -3028,9 +3031,9 @@ returns a site instance. | |||||||
|     :caption: ``myproject/settings.py`` |     :caption: ``myproject/settings.py`` | ||||||
|  |  | ||||||
|     INSTALLED_APPS = [ |     INSTALLED_APPS = [ | ||||||
|         ... |         # ... | ||||||
|         'myproject.apps.MyAdminConfig',  # replaces 'django.contrib.admin' |         'myproject.apps.MyAdminConfig',  # replaces 'django.contrib.admin' | ||||||
|         ... |         # ... | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
| .. _multiple-admin-sites: | .. _multiple-admin-sites: | ||||||
|   | |||||||
| @@ -139,7 +139,7 @@ Geometry Lookups | |||||||
| Geographic queries with geometries take the following general form (assuming | Geographic queries with geometries take the following general form (assuming | ||||||
| the ``Zipcode`` model used in the :doc:`model-api`): | the ``Zipcode`` model used in the :doc:`model-api`): | ||||||
|  |  | ||||||
| .. code-block:: pycon | .. code-block:: text | ||||||
|  |  | ||||||
|     >>> qs = Zipcode.objects.filter(<field>__<lookup_type>=<parameter>) |     >>> qs = Zipcode.objects.filter(<field>__<lookup_type>=<parameter>) | ||||||
|     >>> qs = Zipcode.objects.exclude(...) |     >>> qs = Zipcode.objects.exclude(...) | ||||||
| @@ -175,7 +175,7 @@ band index can be specified. | |||||||
| This results in the following general form for lookups involving rasters | This results in the following general form for lookups involving rasters | ||||||
| (assuming the ``Elevation`` model used in the :doc:`model-api`): | (assuming the ``Elevation`` model used in the :doc:`model-api`): | ||||||
|  |  | ||||||
| .. code-block:: pycon | .. code-block:: text | ||||||
|  |  | ||||||
|     >>> qs = Elevation.objects.filter(<field>__<lookup_type>=<parameter>) |     >>> qs = Elevation.objects.filter(<field>__<lookup_type>=<parameter>) | ||||||
|     >>> qs = Elevation.objects.filter(<field>__<band_index>__<lookup_type>=<parameter>) |     >>> qs = Elevation.objects.filter(<field>__<band_index>__<lookup_type>=<parameter>) | ||||||
|   | |||||||
| @@ -1007,15 +1007,15 @@ Coordinate System Objects | |||||||
|         >>> proj = '+proj=longlat +ellps=WGS84 +datum=WGS84 +no_defs ' |         >>> proj = '+proj=longlat +ellps=WGS84 +datum=WGS84 +no_defs ' | ||||||
|         >>> wgs84 = SpatialReference(proj) # PROJ string |         >>> wgs84 = SpatialReference(proj) # PROJ string | ||||||
|         >>> wgs84 = SpatialReference("""GEOGCS["WGS 84", |         >>> wgs84 = SpatialReference("""GEOGCS["WGS 84", | ||||||
|         DATUM["WGS_1984", |         ... DATUM["WGS_1984", | ||||||
|              SPHEROID["WGS 84",6378137,298.257223563, |         ...      SPHEROID["WGS 84",6378137,298.257223563, | ||||||
|                  AUTHORITY["EPSG","7030"]], |         ...          AUTHORITY["EPSG","7030"]], | ||||||
|              AUTHORITY["EPSG","6326"]], |         ...      AUTHORITY["EPSG","6326"]], | ||||||
|          PRIMEM["Greenwich",0, |         ... PRIMEM["Greenwich",0, | ||||||
|              AUTHORITY["EPSG","8901"]], |         ...      AUTHORITY["EPSG","8901"]], | ||||||
|          UNIT["degree",0.01745329251994328, |         ... UNIT["degree",0.01745329251994328, | ||||||
|              AUTHORITY["EPSG","9122"]], |         ...      AUTHORITY["EPSG","9122"]], | ||||||
|          AUTHORITY["EPSG","4326"]]""") # OGC WKT |         ... AUTHORITY["EPSG","4326"]]""") # OGC WKT | ||||||
|  |  | ||||||
|     .. method:: __getitem__(target) |     .. method:: __getitem__(target) | ||||||
|  |  | ||||||
| @@ -1025,7 +1025,7 @@ Coordinate System Objects | |||||||
|  |  | ||||||
|     .. code-block:: pycon |     .. code-block:: pycon | ||||||
|  |  | ||||||
|         >>> wkt = 'GEOGCS["WGS 84", DATUM["WGS_1984, ... AUTHORITY["EPSG","4326"]]') |         >>> wkt = 'GEOGCS["WGS 84", DATUM["WGS_1984, ... AUTHORITY["EPSG","4326"]]' | ||||||
|         >>> srs = SpatialReference(wkt) # could also use 'WGS84', or 4326 |         >>> srs = SpatialReference(wkt) # could also use 'WGS84', or 4326 | ||||||
|         >>> print(srs['GEOGCS']) |         >>> print(srs['GEOGCS']) | ||||||
|         WGS 84 |         WGS 84 | ||||||
|   | |||||||
| @@ -702,11 +702,13 @@ Distance Lookups | |||||||
| For an overview on performing distance queries, please refer to | For an overview on performing distance queries, please refer to | ||||||
| the :ref:`distance queries introduction <distance-queries>`. | the :ref:`distance queries introduction <distance-queries>`. | ||||||
|  |  | ||||||
| Distance lookups take the following form:: | Distance lookups take the following form: | ||||||
|  |  | ||||||
|     <field>__<distance lookup>=(<geometry/raster>, <distance value>[, 'spheroid']) | .. code-block:: text | ||||||
|     <field>__<distance lookup>=(<raster>, <band_index>, <distance value>[, 'spheroid']) |  | ||||||
|     <field>__<band_index>__<distance lookup>=(<raster>, <band_index>, <distance value>[, 'spheroid']) |     <field>__<distance lookup>=(<geometry/raster>, <distance value>[, "spheroid"]) | ||||||
|  |     <field>__<distance lookup>=(<raster>, <band_index>, <distance value>[, "spheroid"]) | ||||||
|  |     <field>__<band_index>__<distance lookup>=(<raster>, <band_index>, <distance value>[, "spheroid"]) | ||||||
|  |  | ||||||
| The value passed into a distance lookup is a tuple; the first two | The value passed into a distance lookup is a tuple; the first two | ||||||
| values are mandatory, and are the geometry to calculate distances to, | values are mandatory, and are the geometry to calculate distances to, | ||||||
|   | |||||||
| @@ -828,7 +828,7 @@ Other Properties & Methods | |||||||
|     .. code-block:: pycon |     .. code-block:: pycon | ||||||
|  |  | ||||||
|         >>> if poly_1.area > poly_2.area: |         >>> if poly_1.area > poly_2.area: | ||||||
|         >>>     pass |         ...     pass | ||||||
|  |  | ||||||
| .. _geos-geometry-collections: | .. _geos-geometry-collections: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -70,9 +70,10 @@ Example | |||||||
|  |  | ||||||
|     >>> from django.contrib.gis.utils import LayerMapping |     >>> from django.contrib.gis.utils import LayerMapping | ||||||
|     >>> from geoapp.models import TestGeo |     >>> from geoapp.models import TestGeo | ||||||
|     >>> mapping = {'name' : 'str', # The 'name' model field maps to the 'str' layer field. |     >>> mapping = { | ||||||
|                    'poly' : 'POLYGON', # For geometry fields use OGC name. |     ...     'name': 'str', # The 'name' model field maps to the 'str' layer field. | ||||||
|                    } # The mapping is a dictionary |     ...     'poly': 'POLYGON', # For geometry fields use OGC name. | ||||||
|  |     ... } # The mapping is a dictionary | ||||||
|     >>> lm = LayerMapping(TestGeo, 'test_poly.shp', mapping) |     >>> lm = LayerMapping(TestGeo, 'test_poly.shp', mapping) | ||||||
|     >>> lm.save(verbose=True) # Save the layermap, imports the data. |     >>> lm.save(verbose=True) # Save the layermap, imports the data. | ||||||
|     Saved: Name: 1 |     Saved: Name: 1 | ||||||
|   | |||||||
| @@ -584,7 +584,6 @@ Here's a sample configuration which uses a MySQL option file:: | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| .. code-block:: ini | .. code-block:: ini | ||||||
|  |  | ||||||
|     # my.cnf |     # my.cnf | ||||||
| @@ -1026,14 +1025,14 @@ using RAC or pluggable databases without ``tnsnames.ora``, for example. | |||||||
|  |  | ||||||
| Example of an Easy Connect string:: | Example of an Easy Connect string:: | ||||||
|  |  | ||||||
|     'NAME': 'localhost:1521/orclpdb1', |     'NAME': 'localhost:1521/orclpdb1' | ||||||
|  |  | ||||||
| Example of a full DSN string:: | Example of a full DSN string:: | ||||||
|  |  | ||||||
|     'NAME': ( |     'NAME': ( | ||||||
|         '(DESCRIPTION=(ADDRESS=(PROTOCOL=TCP)(HOST=localhost)(PORT=1521))' |         '(DESCRIPTION=(ADDRESS=(PROTOCOL=TCP)(HOST=localhost)(PORT=1521))' | ||||||
|         '(CONNECT_DATA=(SERVICE_NAME=orclpdb1)))' |         '(CONNECT_DATA=(SERVICE_NAME=orclpdb1)))' | ||||||
|     ), |     ) | ||||||
|  |  | ||||||
| Threaded option | Threaded option | ||||||
| --------------- | --------------- | ||||||
| @@ -1044,7 +1043,7 @@ the ``threaded`` option of your Oracle database configuration to ``True``:: | |||||||
|  |  | ||||||
|     'OPTIONS': { |     'OPTIONS': { | ||||||
|         'threaded': True, |         'threaded': True, | ||||||
|     }, |     } | ||||||
|  |  | ||||||
| Failure to do this may result in crashes and other odd behavior. | Failure to do this may result in crashes and other odd behavior. | ||||||
|  |  | ||||||
| @@ -1060,7 +1059,7 @@ The ``RETURNING INTO`` clause can be disabled by setting the | |||||||
|  |  | ||||||
|     'OPTIONS': { |     'OPTIONS': { | ||||||
|         'use_returning_into': False, |         'use_returning_into': False, | ||||||
|     }, |     } | ||||||
|  |  | ||||||
| In this case, the Oracle backend will use a separate ``SELECT`` query to | In this case, the Oracle backend will use a separate ``SELECT`` query to | ||||||
| retrieve ``AutoField`` values. | retrieve ``AutoField`` values. | ||||||
| @@ -1169,7 +1168,7 @@ file:: | |||||||
|     DATABASES = { |     DATABASES = { | ||||||
|         'default': { |         'default': { | ||||||
|             'ENGINE': 'mydbengine', |             'ENGINE': 'mydbengine', | ||||||
|             ... |             # ... | ||||||
|         }, |         }, | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1471,7 +1471,7 @@ need to bind the file data containing the mugshot image: | |||||||
|     ...         'message': 'Hi there', |     ...         'message': 'Hi there', | ||||||
|     ...         'sender': 'foo@example.com', |     ...         'sender': 'foo@example.com', | ||||||
|     ...         'cc_myself': True} |     ...         'cc_myself': True} | ||||||
|     >>> file_data = {'mugshot': SimpleUploadedFile('face.jpg', <file data>)} |     >>> file_data = {'mugshot': SimpleUploadedFile('face.jpg', b"file data")} | ||||||
|     >>> f = ContactFormWithMugshot(data, file_data) |     >>> f = ContactFormWithMugshot(data, file_data) | ||||||
|  |  | ||||||
| In practice, you will usually specify ``request.FILES`` as the source | In practice, you will usually specify ``request.FILES`` as the source | ||||||
|   | |||||||
| @@ -267,7 +267,7 @@ fields. We've specified ``auto_id=False`` to simplify the output: | |||||||
|     <tr><th>Message:</th><td><input type="text" name="message" required></td></tr> |     <tr><th>Message:</th><td><input type="text" name="message" required></td></tr> | ||||||
|     <tr><th>Sender:</th><td><input type="email" name="sender" required><br>A valid email address, please.</td></tr> |     <tr><th>Sender:</th><td><input type="email" name="sender" required><br>A valid email address, please.</td></tr> | ||||||
|     <tr><th>Cc myself:</th><td><input type="checkbox" name="cc_myself"></td></tr> |     <tr><th>Cc myself:</th><td><input type="checkbox" name="cc_myself"></td></tr> | ||||||
|     >>> print(f.as_ul())) |     >>> print(f.as_ul()) | ||||||
|     <li>Subject: <input type="text" name="subject" maxlength="100" required> <span class="helptext">100 characters max.</span></li> |     <li>Subject: <input type="text" name="subject" maxlength="100" required> <span class="helptext">100 characters max.</span></li> | ||||||
|     <li>Message: <input type="text" name="message" required></li> |     <li>Message: <input type="text" name="message" required></li> | ||||||
|     <li>Sender: <input type="email" name="sender" required> A valid email address, please.</li> |     <li>Sender: <input type="email" name="sender" required> A valid email address, please.</li> | ||||||
| @@ -764,7 +764,7 @@ For each field, we describe the default widget used if you don't specify | |||||||
|         >>> from django.core.files.uploadedfile import SimpleUploadedFile |         >>> from django.core.files.uploadedfile import SimpleUploadedFile | ||||||
|         >>> class ImageForm(forms.Form): |         >>> class ImageForm(forms.Form): | ||||||
|         ...     img = forms.ImageField() |         ...     img = forms.ImageField() | ||||||
|         >>> file_data = {'img': SimpleUploadedFile('test.png', <file data>)} |         >>> file_data = {'img': SimpleUploadedFile('test.png', b"file data")} | ||||||
|         >>> form = ImageForm({}, file_data) |         >>> form = ImageForm({}, file_data) | ||||||
|         # Pillow closes the underlying file descriptor. |         # Pillow closes the underlying file descriptor. | ||||||
|         >>> form.is_valid() |         >>> form.is_valid() | ||||||
|   | |||||||
| @@ -228,17 +228,21 @@ logger, but only to the ``django.security`` logger. | |||||||
| To silence a particular type of ``SuspiciousOperation``, you can override that | To silence a particular type of ``SuspiciousOperation``, you can override that | ||||||
| specific logger following this example:: | specific logger following this example:: | ||||||
|  |  | ||||||
|     'handlers': { |     LOGGING = { | ||||||
|         'null': { |         # ... | ||||||
|             'class': 'logging.NullHandler', |         'handlers': { | ||||||
|  |             'null': { | ||||||
|  |                 'class': 'logging.NullHandler', | ||||||
|  |             }, | ||||||
|         }, |         }, | ||||||
|     }, |         'loggers': { | ||||||
|     'loggers': { |             'django.security.DisallowedHost': { | ||||||
|         'django.security.DisallowedHost': { |                 'handlers': ['null'], | ||||||
|             'handlers': ['null'], |                 'propagate': False, | ||||||
|             'propagate': False, |             }, | ||||||
|         }, |         }, | ||||||
|     }, |         # ... | ||||||
|  |     } | ||||||
|  |  | ||||||
| Other ``django.security`` loggers not based on ``SuspiciousOperation`` are: | Other ``django.security`` loggers not based on ``SuspiciousOperation`` are: | ||||||
|  |  | ||||||
| @@ -286,7 +290,7 @@ Python logging module <python:logging.handlers>`. | |||||||
|                 'class': 'django.utils.log.AdminEmailHandler', |                 'class': 'django.utils.log.AdminEmailHandler', | ||||||
|                 'include_html': True, |                 'include_html': True, | ||||||
|             }, |             }, | ||||||
|         }, |         } | ||||||
|  |  | ||||||
|     Be aware of the :ref:`security implications of logging |     Be aware of the :ref:`security implications of logging | ||||||
|     <logging-security-implications>` when using the ``AdminEmailHandler``. |     <logging-security-implications>` when using the ``AdminEmailHandler``. | ||||||
| @@ -301,7 +305,7 @@ Python logging module <python:logging.handlers>`. | |||||||
|                 'class': 'django.utils.log.AdminEmailHandler', |                 'class': 'django.utils.log.AdminEmailHandler', | ||||||
|                 'email_backend': 'django.core.mail.backends.filebased.EmailBackend', |                 'email_backend': 'django.core.mail.backends.filebased.EmailBackend', | ||||||
|             }, |             }, | ||||||
|         }, |         } | ||||||
|  |  | ||||||
|     By default, an instance of the email backend specified in |     By default, an instance of the email backend specified in | ||||||
|     :setting:`EMAIL_BACKEND` will be used. |     :setting:`EMAIL_BACKEND` will be used. | ||||||
| @@ -318,7 +322,7 @@ Python logging module <python:logging.handlers>`. | |||||||
|                 'include_html': True, |                 'include_html': True, | ||||||
|                 'reporter_class': 'somepackage.error_reporter.CustomErrorReporter', |                 'reporter_class': 'somepackage.error_reporter.CustomErrorReporter', | ||||||
|             }, |             }, | ||||||
|         }, |         } | ||||||
|  |  | ||||||
|     .. method:: send_mail(subject, message, *args, **kwargs) |     .. method:: send_mail(subject, message, *args, **kwargs) | ||||||
|  |  | ||||||
| @@ -354,19 +358,23 @@ logging module. | |||||||
|  |  | ||||||
|     and then add it to your logging config:: |     and then add it to your logging config:: | ||||||
|  |  | ||||||
|         'filters': { |         LOGGING = { | ||||||
|             'skip_unreadable_posts': { |             # ... | ||||||
|                 '()': 'django.utils.log.CallbackFilter', |             'filters': { | ||||||
|                 'callback': skip_unreadable_post, |                 'skip_unreadable_posts': { | ||||||
|  |                     '()': 'django.utils.log.CallbackFilter', | ||||||
|  |                     'callback': skip_unreadable_post, | ||||||
|  |                 }, | ||||||
|             }, |             }, | ||||||
|         }, |             'handlers': { | ||||||
|         'handlers': { |                 'mail_admins': { | ||||||
|             'mail_admins': { |                     'level': 'ERROR', | ||||||
|                 'level': 'ERROR', |                     'filters': ['skip_unreadable_posts'], | ||||||
|                 'filters': ['skip_unreadable_posts'], |                     'class': 'django.utils.log.AdminEmailHandler', | ||||||
|                 'class': 'django.utils.log.AdminEmailHandler', |                 }, | ||||||
|             }, |             }, | ||||||
|         }, |             # ... | ||||||
|  |         } | ||||||
|  |  | ||||||
| .. class:: RequireDebugFalse() | .. class:: RequireDebugFalse() | ||||||
|  |  | ||||||
| @@ -376,18 +384,22 @@ logging module. | |||||||
|     configuration to ensure that the :class:`AdminEmailHandler` only sends |     configuration to ensure that the :class:`AdminEmailHandler` only sends | ||||||
|     error emails to admins when :setting:`DEBUG` is ``False``:: |     error emails to admins when :setting:`DEBUG` is ``False``:: | ||||||
|  |  | ||||||
|         'filters': { |         LOGGING = { | ||||||
|             'require_debug_false': { |             # ... | ||||||
|                 '()': 'django.utils.log.RequireDebugFalse', |             'filters': { | ||||||
|  |                 'require_debug_false': { | ||||||
|  |                     '()': 'django.utils.log.RequireDebugFalse', | ||||||
|  |                 }, | ||||||
|             }, |             }, | ||||||
|         }, |             'handlers': { | ||||||
|         'handlers': { |                 'mail_admins': { | ||||||
|             'mail_admins': { |                     'level': 'ERROR', | ||||||
|                 'level': 'ERROR', |                     'filters': ['require_debug_false'], | ||||||
|                 'filters': ['require_debug_false'], |                     'class': 'django.utils.log.AdminEmailHandler', | ||||||
|                 'class': 'django.utils.log.AdminEmailHandler', |                 }, | ||||||
|             }, |             }, | ||||||
|         }, |             # ... | ||||||
|  |         } | ||||||
|  |  | ||||||
| .. class:: RequireDebugTrue() | .. class:: RequireDebugTrue() | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1905,7 +1905,9 @@ more frequently. | |||||||
| .. class:: PercentRank(*expressions, **extra) | .. class:: PercentRank(*expressions, **extra) | ||||||
|  |  | ||||||
| Computes the relative rank of the rows in the frame clause. This computation is | Computes the relative rank of the rows in the frame clause. This computation is | ||||||
| equivalent to evaluating:: | equivalent to evaluating: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     (rank - 1) / (total rows - 1) |     (rank - 1) / (total rows - 1) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -21,20 +21,20 @@ constants, variables, and even other expressions. | |||||||
| Some examples | Some examples | ||||||
| ============= | ============= | ||||||
|  |  | ||||||
| .. code-block:: python | .. code-block:: pycon | ||||||
|  |  | ||||||
|     from django.db.models import Count, F, Value |     >>> from django.db.models import Count, F, Value | ||||||
|     from django.db.models.functions import Length, Upper |     >>> from django.db.models.functions import Length, Upper | ||||||
|     from django.db.models.lookups import GreaterThan |     >>> from django.db.models.lookups import GreaterThan | ||||||
|  |  | ||||||
|     # Find companies that have more employees than chairs. |     # Find companies that have more employees than chairs. | ||||||
|     Company.objects.filter(num_employees__gt=F('num_chairs')) |     >>> Company.objects.filter(num_employees__gt=F('num_chairs')) | ||||||
|  |  | ||||||
|     # Find companies that have at least twice as many employees |     # Find companies that have at least twice as many employees | ||||||
|     # as chairs. Both the querysets below are equivalent. |     # as chairs. Both the querysets below are equivalent. | ||||||
|     Company.objects.filter(num_employees__gt=F('num_chairs') * 2) |     >>> Company.objects.filter(num_employees__gt=F('num_chairs') * 2) | ||||||
|     Company.objects.filter( |     >>> Company.objects.filter( | ||||||
|         num_employees__gt=F('num_chairs') + F('num_chairs')) |     ...     num_employees__gt=F('num_chairs') + F('num_chairs')) | ||||||
|  |  | ||||||
|     # How many chairs are needed for each company to seat all employees? |     # How many chairs are needed for each company to seat all employees? | ||||||
|     >>> company = Company.objects.filter( |     >>> company = Company.objects.filter( | ||||||
| @@ -817,12 +817,12 @@ the same studio in the same genre and release year: | |||||||
|  |  | ||||||
|     >>> from django.db.models import Avg, F, Window |     >>> from django.db.models import Avg, F, Window | ||||||
|     >>> Movie.objects.annotate( |     >>> Movie.objects.annotate( | ||||||
|     >>>     avg_rating=Window( |     ...     avg_rating=Window( | ||||||
|     >>>         expression=Avg('rating'), |     ...         expression=Avg('rating'), | ||||||
|     >>>         partition_by=[F('studio'), F('genre')], |     ...         partition_by=[F('studio'), F('genre')], | ||||||
|     >>>         order_by='released__year', |     ...         order_by='released__year', | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>> ) |     ... ) | ||||||
|  |  | ||||||
| This allows you to check if a movie is rated better or worse than its peers. | This allows you to check if a movie is rated better or worse than its peers. | ||||||
|  |  | ||||||
| @@ -837,20 +837,20 @@ to reduce repetition: | |||||||
|  |  | ||||||
|     >>> from django.db.models import Avg, F, Max, Min, Window |     >>> from django.db.models import Avg, F, Max, Min, Window | ||||||
|     >>> window = { |     >>> window = { | ||||||
|     >>>    'partition_by': [F('studio'), F('genre')], |     ...    'partition_by': [F('studio'), F('genre')], | ||||||
|     >>>    'order_by': 'released__year', |     ...    'order_by': 'released__year', | ||||||
|     >>> } |     ... } | ||||||
|     >>> Movie.objects.annotate( |     >>> Movie.objects.annotate( | ||||||
|     >>>     avg_rating=Window( |     ...     avg_rating=Window( | ||||||
|     >>>         expression=Avg('rating'), **window, |     ...         expression=Avg('rating'), **window, | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>>     best=Window( |     ...     best=Window( | ||||||
|     >>>         expression=Max('rating'), **window, |     ...         expression=Max('rating'), **window, | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>>     worst=Window( |     ...     worst=Window( | ||||||
|     >>>         expression=Min('rating'), **window, |     ...         expression=Min('rating'), **window, | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>> ) |     ... ) | ||||||
|  |  | ||||||
| Filtering against window functions is supported as long as lookups are not | Filtering against window functions is supported as long as lookups are not | ||||||
| disjunctive (not using ``OR`` or ``XOR`` as a connector) and against a queryset | disjunctive (not using ``OR`` or ``XOR`` as a connector) and against a queryset | ||||||
| @@ -864,13 +864,13 @@ from groups to be included: | |||||||
| .. code-block:: pycon | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> qs = Movie.objects.annotate( |     >>> qs = Movie.objects.annotate( | ||||||
|     >>>     category_rank=Window( |     ...     category_rank=Window( | ||||||
|     >>>         Rank(), partition_by='category', order_by='-rating' |     ...         Rank(), partition_by='category', order_by='-rating' | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>>     scenes_count=Count('actors'), |     ...     scenes_count=Count('actors'), | ||||||
|     >>> ).filter( |     ... ).filter( | ||||||
|     >>>     Q(category_rank__lte=3) | Q(title__contains='Batman') |     ...     Q(category_rank__lte=3) | Q(title__contains='Batman') | ||||||
|     >>> ) |     ... ) | ||||||
|     >>> list(qs) |     >>> list(qs) | ||||||
|     NotImplementedError: Heterogeneous disjunctive predicates against window functions |     NotImplementedError: Heterogeneous disjunctive predicates against window functions | ||||||
|     are not implemented when performing conditional aggregation. |     are not implemented when performing conditional aggregation. | ||||||
| @@ -949,13 +949,13 @@ with the average rating of a movie's two prior and two following peers: | |||||||
|  |  | ||||||
|     >>> from django.db.models import Avg, F, RowRange, Window |     >>> from django.db.models import Avg, F, RowRange, Window | ||||||
|     >>> Movie.objects.annotate( |     >>> Movie.objects.annotate( | ||||||
|     >>>     avg_rating=Window( |     ...     avg_rating=Window( | ||||||
|     >>>         expression=Avg('rating'), |     ...         expression=Avg('rating'), | ||||||
|     >>>         partition_by=[F('studio'), F('genre')], |     ...         partition_by=[F('studio'), F('genre')], | ||||||
|     >>>         order_by='released__year', |     ...         order_by='released__year', | ||||||
|     >>>         frame=RowRange(start=-2, end=2), |     ...         frame=RowRange(start=-2, end=2), | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>> ) |     ... ) | ||||||
|  |  | ||||||
| If the database supports it, you can specify the start and end points based on | If the database supports it, you can specify the start and end points based on | ||||||
| values of an expression in the partition. If the ``released`` field of the | values of an expression in the partition. If the ``released`` field of the | ||||||
| @@ -967,13 +967,13 @@ released between twelve months before and twelve months after the each movie: | |||||||
|  |  | ||||||
|     >>> from django.db.models import Avg, F, ValueRange, Window |     >>> from django.db.models import Avg, F, ValueRange, Window | ||||||
|     >>> Movie.objects.annotate( |     >>> Movie.objects.annotate( | ||||||
|     >>>     avg_rating=Window( |     ...     avg_rating=Window( | ||||||
|     >>>         expression=Avg('rating'), |     ...         expression=Avg('rating'), | ||||||
|     >>>         partition_by=[F('studio'), F('genre')], |     ...         partition_by=[F('studio'), F('genre')], | ||||||
|     >>>         order_by='released__year', |     ...         order_by='released__year', | ||||||
|     >>>         frame=ValueRange(start=-12, end=12), |     ...         frame=ValueRange(start=-12, end=12), | ||||||
|     >>>     ), |     ...     ), | ||||||
|     >>> ) |     ... ) | ||||||
|  |  | ||||||
| .. currentmodule:: django.db.models | .. currentmodule:: django.db.models | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,7 +24,8 @@ Related objects reference | |||||||
|       In the above example, the methods below will be available on |       In the above example, the methods below will be available on | ||||||
|       the manager ``blog.entry_set``. |       the manager ``blog.entry_set``. | ||||||
|  |  | ||||||
|     * Both sides of a :class:`~django.db.models.ManyToManyField` relation:: |     * Both sides of a :class:`~django.db.models.ManyToManyField` relation | ||||||
|  |       :: | ||||||
|  |  | ||||||
|             class Topping(models.Model): |             class Topping(models.Model): | ||||||
|                 # ... |                 # ... | ||||||
|   | |||||||
| @@ -499,7 +499,7 @@ Use of both ``and`` and ``or`` clauses within the same tag is allowed, with | |||||||
|  |  | ||||||
| will be interpreted like: | will be interpreted like: | ||||||
|  |  | ||||||
| .. code-block:: python | .. code-block:: pycon | ||||||
|  |  | ||||||
|     if (athlete_list and coach_list) or cheerleader_list |     if (athlete_list and coach_list) or cheerleader_list | ||||||
|  |  | ||||||
| @@ -1118,14 +1118,16 @@ Example usage: | |||||||
|  |  | ||||||
| This example would return this HTML:: | This example would return this HTML:: | ||||||
|  |  | ||||||
|     <h1>José Mourinho</h1> | .. code-block:: html | ||||||
|     <p class="odd">Thibaut Courtois</p> |  | ||||||
|     <p class="even">John Terry</p> |  | ||||||
|     <p class="odd">Eden Hazard</p> |  | ||||||
|  |  | ||||||
|     <h1>Carlo Ancelotti</h1> |     <h1>Gareth</h1> | ||||||
|     <p class="odd">Manuel Neuer</p> |     <p class="odd">Harry</p> | ||||||
|     <p class="even">Thomas Müller</p> |     <p class="even">John</p> | ||||||
|  |     <p class="odd">Nick</p> | ||||||
|  |  | ||||||
|  |     <h1>John</h1> | ||||||
|  |     <p class="odd">Andrea</p> | ||||||
|  |     <p class="even">Melissa</p> | ||||||
|  |  | ||||||
| Notice how the first block ends with ``class="odd"`` and the new one starts | Notice how the first block ends with ``class="odd"`` and the new one starts | ||||||
| with ``class="odd"``. Without the ``{% resetcycle %}`` tag, the second block | with ``class="odd"``. Without the ``{% resetcycle %}`` tag, the second block | ||||||
| @@ -1264,6 +1266,8 @@ such as this: | |||||||
|  |  | ||||||
| ...then, in a template, you can create a link to this view like this:: | ...then, in a template, you can create a link to this view like this:: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% url 'app-views-client' client.id %} |     {% url 'app-views-client' client.id %} | ||||||
|  |  | ||||||
| The template tag will output the string ``/clients/client/123/``. | The template tag will output the string ``/clients/client/123/``. | ||||||
|   | |||||||
| @@ -544,14 +544,14 @@ https://web.archive.org/web/20110718035220/http://diveintomark.org/archives/2004 | |||||||
|  |  | ||||||
|         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, *args, **kwargs): | ||||||
|             # Do some conversion on string 's' |             # Do some conversion on string 's' | ||||||
|             ... |             ... | ||||||
|         fancy_utility_function = keep_lazy(str)(fancy_utility_function) |         fancy_utility_function = keep_lazy(str)(fancy_utility_function) | ||||||
|  |  | ||||||
|         # Or more succinctly: |         # Or more succinctly: | ||||||
|         @keep_lazy(str) |         @keep_lazy(str) | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, *args, **kwargs): | ||||||
|             ... |             ... | ||||||
|  |  | ||||||
|     The ``keep_lazy()`` decorator takes a number of extra arguments (``*args``) |     The ``keep_lazy()`` decorator takes a number of extra arguments (``*args``) | ||||||
| @@ -576,12 +576,12 @@ https://web.archive.org/web/20110718035220/http://diveintomark.org/archives/2004 | |||||||
|  |  | ||||||
|         # Our previous example was: |         # Our previous example was: | ||||||
|         @keep_lazy(str) |         @keep_lazy(str) | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, *args, **kwargs): | ||||||
|             ... |             ... | ||||||
|  |  | ||||||
|         # Which can be rewritten as: |         # Which can be rewritten as: | ||||||
|         @keep_lazy_text |         @keep_lazy_text | ||||||
|         def fancy_utility_function(s, ...): |         def fancy_utility_function(s, *args, **kwargs): | ||||||
|             ... |             ... | ||||||
|  |  | ||||||
| ``django.utils.html`` | ``django.utils.html`` | ||||||
|   | |||||||
| @@ -36,12 +36,16 @@ differences as a result of this change. | |||||||
| However, **users on 64-bit platforms may experience some problems** using the | However, **users on 64-bit platforms may experience some problems** using the | ||||||
| ``reset`` management command. Prior to this change, 64-bit platforms | ``reset`` management command. Prior to this change, 64-bit platforms | ||||||
| would generate a 64-bit, 16 character digest in the constraint name; for | would generate a 64-bit, 16 character digest in the constraint name; for | ||||||
| example:: | example: | ||||||
|  |  | ||||||
|  | .. code-block:: sql | ||||||
|  |  | ||||||
|     ALTER TABLE myapp_sometable ADD CONSTRAINT object_id_refs_id_5e8f10c132091d1e FOREIGN KEY ... |     ALTER TABLE myapp_sometable ADD CONSTRAINT object_id_refs_id_5e8f10c132091d1e FOREIGN KEY ... | ||||||
|  |  | ||||||
| Following this change, all platforms, regardless of word size, will generate a | Following this change, all platforms, regardless of word size, will generate a | ||||||
| 32-bit, 8 character digest in the constraint name; for example:: | 32-bit, 8 character digest in the constraint name; for example: | ||||||
|  |  | ||||||
|  | .. code-block:: sql | ||||||
|  |  | ||||||
|     ALTER TABLE myapp_sometable ADD CONSTRAINT object_id_refs_id_32091d1e FOREIGN KEY ... |     ALTER TABLE myapp_sometable ADD CONSTRAINT object_id_refs_id_32091d1e FOREIGN KEY ... | ||||||
|  |  | ||||||
|   | |||||||
| @@ -545,14 +545,18 @@ Database backend API | |||||||
| ``select_related()`` prohibits non-relational fields for nested relations | ``select_related()`` prohibits non-relational fields for nested relations | ||||||
| ------------------------------------------------------------------------- | ------------------------------------------------------------------------- | ||||||
|  |  | ||||||
| Django 1.8 added validation for non-relational fields in ``select_related()``:: | Django 1.8 added validation for non-relational fields in ``select_related()``: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Book.objects.select_related('title') |     >>> Book.objects.select_related('title') | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
|     ... |     ... | ||||||
|     FieldError: Non-relational field given in select_related: 'title' |     FieldError: Non-relational field given in select_related: 'title' | ||||||
|  |  | ||||||
| But it didn't prohibit nested non-relation fields as it does now:: | But it didn't prohibit nested non-relation fields as it does now: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Book.objects.select_related('author__name') |     >>> Book.objects.select_related('author__name') | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
| @@ -620,6 +624,8 @@ disable Django's logging configuration or override it with your own, you'll | |||||||
| need to add the appropriate logging configuration if you want to see that | need to add the appropriate logging configuration if you want to see that | ||||||
| output:: | output:: | ||||||
|  |  | ||||||
|  |   LOGGING = { | ||||||
|  |     # ... | ||||||
|     'formatters': { |     'formatters': { | ||||||
|         'django.server': { |         'django.server': { | ||||||
|             '()': 'django.utils.log.ServerFormatter', |             '()': 'django.utils.log.ServerFormatter', | ||||||
| @@ -639,7 +645,8 @@ output:: | |||||||
|             'level': 'INFO', |             'level': 'INFO', | ||||||
|             'propagate': False, |             'propagate': False, | ||||||
|         } |         } | ||||||
|     } |     }, | ||||||
|  |   } | ||||||
|  |  | ||||||
| ``auth.CustomUser`` and ``auth.ExtensionUser`` test models were removed | ``auth.CustomUser`` and ``auth.ExtensionUser`` test models were removed | ||||||
| ----------------------------------------------------------------------- | ----------------------------------------------------------------------- | ||||||
| @@ -725,7 +732,7 @@ custom lookup for it. For example:: | |||||||
|     class MyFieldExact(Exact): |     class MyFieldExact(Exact): | ||||||
|         def get_prep_lookup(self): |         def get_prep_lookup(self): | ||||||
|             # do_custom_stuff_for_myfield |             # do_custom_stuff_for_myfield | ||||||
|             .... |             ... | ||||||
|  |  | ||||||
|     MyField.register_lookup(MyFieldExact) |     MyField.register_lookup(MyFieldExact) | ||||||
|  |  | ||||||
| @@ -931,13 +938,17 @@ Features deprecated in 1.10 | |||||||
| Direct assignment to a reverse foreign key or many-to-many relation | Direct assignment to a reverse foreign key or many-to-many relation | ||||||
| ------------------------------------------------------------------- | ------------------------------------------------------------------- | ||||||
|  |  | ||||||
| Instead of assigning related objects using direct assignment:: | Instead of assigning related objects using direct assignment: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> new_list = [obj1, obj2, obj3] |     >>> new_list = [obj1, obj2, obj3] | ||||||
|     >>> e.related_set = new_list |     >>> e.related_set = new_list | ||||||
|  |  | ||||||
| Use the :meth:`~django.db.models.fields.related.RelatedManager.set` method | Use the :meth:`~django.db.models.fields.related.RelatedManager.set` method | ||||||
| added in Django 1.9:: | added in Django 1.9: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|         >>> e.related_set.set([obj1, obj2, obj3]) |         >>> e.related_set.set([obj1, obj2, obj3]) | ||||||
|  |  | ||||||
| @@ -1022,12 +1033,16 @@ Assume the following models:: | |||||||
| In older versions, :attr:`~django.db.models.Options.default_related_name` | In older versions, :attr:`~django.db.models.Options.default_related_name` | ||||||
| couldn't be used as a query lookup. This is fixed and support for the old | couldn't be used as a query lookup. This is fixed and support for the old | ||||||
| lookup name is deprecated. For example, since ``default_related_name`` is set | lookup name is deprecated. For example, since ``default_related_name`` is set | ||||||
| in model ``Bar``, instead of using the model name ``bar`` as the lookup:: | in model ``Bar``, instead of using the model name ``bar`` as the lookup: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> bar = Bar.objects.get(pk=1) |     >>> bar = Bar.objects.get(pk=1) | ||||||
|     >>> Foo.objects.get(bar=bar) |     >>> Foo.objects.get(bar=bar) | ||||||
|  |  | ||||||
| use the default_related_name ``bars``:: | use the default_related_name ``bars``: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Foo.objects.get(bars=bar) |     >>> Foo.objects.get(bars=bar) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -453,42 +453,42 @@ several functions to support conversion of Python values into | |||||||
| database-compatible values. A custom field might look something like:: | database-compatible values. A custom field might look something like:: | ||||||
|  |  | ||||||
|     class CustomModelField(models.Field): |     class CustomModelField(models.Field): | ||||||
|         # ... |         ... | ||||||
|         def db_type(self): |         def db_type(self): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_save(self, value): |         def get_db_prep_save(self, value): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_value(self, value): |         def get_db_prep_value(self, value): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_lookup(self, lookup_type, value): |         def get_db_prep_lookup(self, lookup_type, value): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
| In 1.2, these three methods have undergone a change in prototype, and | In 1.2, these three methods have undergone a change in prototype, and | ||||||
| two extra methods have been introduced:: | two extra methods have been introduced:: | ||||||
|  |  | ||||||
|     class CustomModelField(models.Field): |     class CustomModelField(models.Field): | ||||||
|         # ... |         ... | ||||||
|  |  | ||||||
|         def db_type(self, connection): |         def db_type(self, connection): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_prep_value(self, value): |         def get_prep_value(self, value): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_prep_lookup(self, lookup_type, value): |         def get_prep_lookup(self, lookup_type, value): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_save(self, value, connection): |         def get_db_prep_save(self, value, connection): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_value(self, value, connection, prepared=False): |         def get_db_prep_value(self, value, connection, prepared=False): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
|         def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False): |         def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False): | ||||||
|             # ... |             ... | ||||||
|  |  | ||||||
| These changes are required to support multiple databases -- | These changes are required to support multiple databases -- | ||||||
| ``db_type`` and ``get_db_prep_*`` can no longer make any assumptions | ``db_type`` and ``get_db_prep_*`` can no longer make any assumptions | ||||||
| @@ -537,21 +537,29 @@ You may also need to update your templates if you were relying on the | |||||||
| implementation of Django's template tags *not* being thread safe. The | implementation of Django's template tags *not* being thread safe. The | ||||||
| :ttag:`cycle` tag is the most likely to be affected in this way, | :ttag:`cycle` tag is the most likely to be affected in this way, | ||||||
| especially when used in conjunction with the :ttag:`include` tag. | especially when used in conjunction with the :ttag:`include` tag. | ||||||
| Consider the following template fragment:: | Consider the following template fragment: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% for object in object_list %} |     {% for object in object_list %} | ||||||
|         {% include "subtemplate.html" %} |         {% include "subtemplate.html" %} | ||||||
|     {% endfor %} |     {% endfor %} | ||||||
|  |  | ||||||
| with a ``subtemplate.html`` that reads:: | with a ``subtemplate.html`` that reads: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% cycle 'even' 'odd' %} |     {% cycle 'even' 'odd' %} | ||||||
|  |  | ||||||
| Using the non-thread-safe, pre-Django 1.2 renderer, this would output:: | Using the non-thread-safe, pre-Django 1.2 renderer, this would output: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     even odd even odd ... |     even odd even odd ... | ||||||
|  |  | ||||||
| Using the thread-safe Django 1.2 renderer, you will instead get:: | Using the thread-safe Django 1.2 renderer, you will instead get: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     even even even even ... |     even even even even ... | ||||||
|  |  | ||||||
| @@ -1133,14 +1141,18 @@ is using a supported spatial database backend. | |||||||
|     differs across spatial databases, the ``SpatialRefSys`` and |     differs across spatial databases, the ``SpatialRefSys`` and | ||||||
|     ``GeometryColumns`` models can no longer be associated with |     ``GeometryColumns`` models can no longer be associated with | ||||||
|     the ``gis`` application name.  Thus, no models will be returned |     the ``gis`` application name.  Thus, no models will be returned | ||||||
|     when using the ``get_models`` method in the following example:: |     when using the ``get_models`` method in the following example: | ||||||
|  |  | ||||||
|  |     .. code-block:: pycon | ||||||
|  |  | ||||||
|         >>> from django.db.models import get_app, get_models |         >>> from django.db.models import get_app, get_models | ||||||
|         >>> get_models(get_app('gis')) |         >>> get_models(get_app('gis')) | ||||||
|         [] |         [] | ||||||
|  |  | ||||||
| To get the correct ``SpatialRefSys`` and ``GeometryColumns`` | To get the correct ``SpatialRefSys`` and ``GeometryColumns`` | ||||||
| for your spatial database use the methods provided by the spatial backend:: | for your spatial database use the methods provided by the spatial backend: | ||||||
|  |  | ||||||
|  |     .. code-block:: pycon | ||||||
|  |  | ||||||
|      >>> from django.db import connections |      >>> from django.db import connections | ||||||
|      >>> SpatialRefSys = connections['my_spatialite'].ops.spatial_ref_sys() |      >>> SpatialRefSys = connections['my_spatialite'].ops.spatial_ref_sys() | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ as was reported to us recently. The Host header parsing in Django 1.3.3 and | |||||||
| Django 1.4.1 -- specifically, ``django.http.HttpRequest.get_host()`` -- was | Django 1.4.1 -- specifically, ``django.http.HttpRequest.get_host()`` -- was | ||||||
| incorrectly handling username/password information in the header. Thus, for | incorrectly handling username/password information in the header. Thus, for | ||||||
| example, the following Host header would be accepted by Django when running on | example, the following Host header would be accepted by Django when running on | ||||||
| ``validsite.com``:: | ``validsite.com``: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     Host: validsite.com:random@evilsite.com |     Host: validsite.com:random@evilsite.com | ||||||
|  |  | ||||||
|   | |||||||
| @@ -141,7 +141,7 @@ Users of Python 2.5 and above may now use transaction management functions as | |||||||
| context managers. For example:: | context managers. For example:: | ||||||
|  |  | ||||||
|     with transaction.autocommit(): |     with transaction.autocommit(): | ||||||
|         # ... |         ... | ||||||
|  |  | ||||||
| Configurable delete-cascade | Configurable delete-cascade | ||||||
| --------------------------- | --------------------------- | ||||||
| @@ -419,7 +419,9 @@ If you have an existing project that is using the database session | |||||||
| backend, you don't have to do anything to accommodate this change. | backend, you don't have to do anything to accommodate this change. | ||||||
| However, you may get a significant performance boost if you manually | However, you may get a significant performance boost if you manually | ||||||
| add the new index to the session table. The SQL that will add the | add the new index to the session table. The SQL that will add the | ||||||
| index can be found by running the ``sqlindexes`` admin command:: | index can be found by running the ``sqlindexes`` admin command: | ||||||
|  |  | ||||||
|  | .. code-block:: shell | ||||||
|  |  | ||||||
|     python manage.py sqlindexes sessions |     python manage.py sqlindexes sessions | ||||||
|  |  | ||||||
| @@ -476,14 +478,18 @@ passed empty dictionary. This was inconsistent with behavior in other parts of | |||||||
| the framework. Starting with 1.3 if you pass in empty dictionary the | the framework. Starting with 1.3 if you pass in empty dictionary the | ||||||
| ``FormSet`` will raise a ``ValidationError``. | ``FormSet`` will raise a ``ValidationError``. | ||||||
|  |  | ||||||
| For example with a ``FormSet``:: | For example with a ``FormSet``: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> class ArticleForm(Form): |     >>> class ArticleForm(Form): | ||||||
|     ...     title = CharField() |     ...     title = CharField() | ||||||
|     ...     pub_date = DateField() |     ...     pub_date = DateField() | ||||||
|     >>> ArticleFormSet = formset_factory(ArticleForm) |     >>> ArticleFormSet = formset_factory(ArticleForm) | ||||||
|  |  | ||||||
| the following code will raise a ``ValidationError``:: | the following code will raise a ``ValidationError``: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> ArticleFormSet({}) |     >>> ArticleFormSet({}) | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
| @@ -491,7 +497,9 @@ the following code will raise a ``ValidationError``:: | |||||||
|     ValidationError: [u'ManagementForm data is missing or has been tampered with'] |     ValidationError: [u'ManagementForm data is missing or has been tampered with'] | ||||||
|  |  | ||||||
| if you need to instantiate an empty ``FormSet``, don't pass in the data or use | if you need to instantiate an empty ``FormSet``, don't pass in the data or use | ||||||
| ``None``:: | ``None``: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> formset = ArticleFormSet() |     >>> formset = ArticleFormSet() | ||||||
|     >>> formset = ArticleFormSet(data=None) |     >>> formset = ArticleFormSet(data=None) | ||||||
| @@ -502,7 +510,9 @@ Callables in templates | |||||||
| Previously, a callable in a template would only be called automatically as part | Previously, a callable in a template would only be called automatically as part | ||||||
| of the variable resolution process if it was retrieved via attribute | of the variable resolution process if it was retrieved via attribute | ||||||
| lookup. This was an inconsistency that could result in confusing and unhelpful | lookup. This was an inconsistency that could result in confusing and unhelpful | ||||||
| behavior:: | behavior: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Template("{{ user.get_full_name }}").render(Context({'user': user})) |     >>> Template("{{ user.get_full_name }}").render(Context({'user': user})) | ||||||
|     u'Joe Bloggs' |     u'Joe Bloggs' | ||||||
| @@ -720,12 +730,16 @@ Changes to ``url`` and ``ssi`` | |||||||
| ------------------------------ | ------------------------------ | ||||||
|  |  | ||||||
| Most template tags will allow you to pass in either constants or | Most template tags will allow you to pass in either constants or | ||||||
| variables as arguments -- for example:: | variables as arguments -- for example: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% extends "base.html" %} |     {% extends "base.html" %} | ||||||
|  |  | ||||||
| allows you to specify a base template as a constant, but if you have a | allows you to specify a base template as a constant, but if you have a | ||||||
| context variable ``templ`` that contains the value ``base.html``:: | context variable ``templ`` that contains the value ``base.html``: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% extends templ %} |     {% extends templ %} | ||||||
|  |  | ||||||
| @@ -741,11 +755,15 @@ accident. Django 1.3 adds a new template library -- ``future`` -- that | |||||||
| provides alternate implementations of the ``url`` and ``ssi`` | provides alternate implementations of the ``url`` and ``ssi`` | ||||||
| template tags. This ``future`` library implement behavior that makes | template tags. This ``future`` library implement behavior that makes | ||||||
| the handling of the first argument consistent with the handling of all | the handling of the first argument consistent with the handling of all | ||||||
| other variables. So, an existing template that contains:: | other variables. So, an existing template that contains: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% url sample %} |     {% url sample %} | ||||||
|  |  | ||||||
| should be replaced with:: | should be replaced with: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% load url from future %} |     {% load url from future %} | ||||||
|     {% url 'sample' %} |     {% url 'sample' %} | ||||||
| @@ -840,7 +858,9 @@ Rationale for this decision: | |||||||
|   generate spurious error messages when the project directory is added |   generate spurious error messages when the project directory is added | ||||||
|   to the Python path (``manage.py runserver`` does this) and then it |   to the Python path (``manage.py runserver`` does this) and then it | ||||||
|   clashes with the equally named standard library module, this is a |   clashes with the equally named standard library module, this is a | ||||||
|   typical warning message:: |   typical warning message: | ||||||
|  |  | ||||||
|  |   .. code-block:: pytb | ||||||
|  |  | ||||||
|      /usr/lib/python2.6/gettext.py:49: ImportWarning: Not importing directory '/path/to/project/locale': missing __init__.py. |      /usr/lib/python2.6/gettext.py:49: ImportWarning: Not importing directory '/path/to/project/locale': missing __init__.py. | ||||||
|      import locale, copy, os, re, struct, sys |      import locale, copy, os, re, struct, sys | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ as was reported to us recently. The Host header parsing in Django 1.3.3 and | |||||||
| Django 1.4.1 -- specifically, ``django.http.HttpRequest.get_host()`` -- was | Django 1.4.1 -- specifically, ``django.http.HttpRequest.get_host()`` -- was | ||||||
| incorrectly handling username/password information in the header. Thus, for | incorrectly handling username/password information in the header. Thus, for | ||||||
| example, the following Host header would be accepted by Django when running on | example, the following Host header would be accepted by Django when running on | ||||||
| ``validsite.com``:: | ``validsite.com``: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     Host: validsite.com:random@evilsite.com |     Host: validsite.com:random@evilsite.com | ||||||
|  |  | ||||||
|   | |||||||
| @@ -152,7 +152,9 @@ using the project name prefix (e.g. ``myproject.settings``, ``ROOT_URLCONF = | |||||||
| directory up, so it is outside the project package rather than adjacent to | directory up, so it is outside the project package rather than adjacent to | ||||||
| ``settings.py`` and ``urls.py``. | ``settings.py`` and ``urls.py``. | ||||||
|  |  | ||||||
| For instance, with the following layout:: | For instance, with the following layout: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     manage.py |     manage.py | ||||||
|     mysite/ |     mysite/ | ||||||
| @@ -168,7 +170,9 @@ but not ``settings``, ``urls``, or ``myapp`` as top-level modules. | |||||||
|  |  | ||||||
| Anything imported as a top-level module can be placed adjacent to the new | Anything imported as a top-level module can be placed adjacent to the new | ||||||
| ``manage.py``. For instance, to decouple ``myapp`` from the project module and | ``manage.py``. For instance, to decouple ``myapp`` from the project module and | ||||||
| import it as just ``myapp``, place it outside the ``mysite/`` directory:: | import it as just ``myapp``, place it outside the ``mysite/`` directory: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     manage.py |     manage.py | ||||||
|     myapp/ |     myapp/ | ||||||
| @@ -191,12 +195,16 @@ now have a ``--template`` option for specifying a path or URL to a custom app | |||||||
| or project template. | or project template. | ||||||
|  |  | ||||||
| For example, Django will use the ``/path/to/my_project_template`` directory | For example, Django will use the ``/path/to/my_project_template`` directory | ||||||
| when you run the following command:: | when you run the following command: | ||||||
|  |  | ||||||
|  | .. code-block:: shell | ||||||
|  |  | ||||||
|     django-admin.py startproject --template=/path/to/my_project_template myproject |     django-admin.py startproject --template=/path/to/my_project_template myproject | ||||||
|  |  | ||||||
| You can also now provide a destination directory as the second | You can also now provide a destination directory as the second | ||||||
| argument to both :djadmin:`startapp` and :djadmin:`startproject`:: | argument to both :djadmin:`startapp` and :djadmin:`startproject`: | ||||||
|  |  | ||||||
|  | .. code-block:: shell | ||||||
|  |  | ||||||
|     django-admin.py startapp myapp /path/to/new/app |     django-admin.py startapp myapp /path/to/new/app | ||||||
|     django-admin.py startproject myproject /path/to/new/project |     django-admin.py startproject myproject /path/to/new/project | ||||||
| @@ -1136,7 +1144,9 @@ Development Server Multithreading | |||||||
|  |  | ||||||
| The development server is now is multithreaded by default. Use the | The development server is now is multithreaded by default. Use the | ||||||
| :option:`runserver --nothreading` option to disable the use of threading in the | :option:`runserver --nothreading` option to disable the use of threading in the | ||||||
| development server:: | development server: | ||||||
|  |  | ||||||
|  | .. code-block:: shell | ||||||
|  |  | ||||||
|     django-admin.py runserver --nothreading |     django-admin.py runserver --nothreading | ||||||
|  |  | ||||||
| @@ -1199,18 +1209,21 @@ To increase the flexibility of error logging for requests, the | |||||||
| separate filter attached to :class:`django.utils.log.AdminEmailHandler` to | separate filter attached to :class:`django.utils.log.AdminEmailHandler` to | ||||||
| prevent admin error emails in ``DEBUG`` mode:: | prevent admin error emails in ``DEBUG`` mode:: | ||||||
|  |  | ||||||
|    'filters': { |     LOGGING = { | ||||||
|         'require_debug_false': { |         # ... | ||||||
|             '()': 'django.utils.log.RequireDebugFalse' |         'filters': { | ||||||
|         } |             'require_debug_false': { | ||||||
|     }, |                 '()': 'django.utils.log.RequireDebugFalse', | ||||||
|     'handlers': { |             } | ||||||
|         'mail_admins': { |         }, | ||||||
|             'level': 'ERROR', |         'handlers': { | ||||||
|             'filters': ['require_debug_false'], |             'mail_admins': { | ||||||
|             'class': 'django.utils.log.AdminEmailHandler' |                 'level': 'ERROR', | ||||||
|         } |                 'filters': ['require_debug_false'], | ||||||
|     }, |                 'class': 'django.utils.log.AdminEmailHandler' | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |     } | ||||||
|  |  | ||||||
| If your project was created prior to this change, your :setting:`LOGGING` | If your project was created prior to this change, your :setting:`LOGGING` | ||||||
| setting will not include this new filter. In order to maintain | setting will not include this new filter. In order to maintain | ||||||
|   | |||||||
| @@ -155,7 +155,9 @@ Caching of related model instances | |||||||
| ---------------------------------- | ---------------------------------- | ||||||
|  |  | ||||||
| When traversing relations, the ORM will avoid re-fetching objects that were | When traversing relations, the ORM will avoid re-fetching objects that were | ||||||
| previously loaded. For example, with the tutorial's models:: | previously loaded. For example, with the tutorial's models: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> first_poll = Poll.objects.all()[0] |     >>> first_poll = Poll.objects.all()[0] | ||||||
|     >>> first_choice = first_poll.choice_set.all()[0] |     >>> first_choice = first_poll.choice_set.all()[0] | ||||||
| @@ -403,7 +405,9 @@ Context in year and month archive class-based views | |||||||
| provide a ``date_list`` sorted in ascending order in the context, like their | provide a ``date_list`` sorted in ascending order in the context, like their | ||||||
| function-based predecessors, but it actually was in descending order. In 1.5, | function-based predecessors, but it actually was in descending order. In 1.5, | ||||||
| the documented order was restored. You may want to add (or remove) the | the documented order was restored. You may want to add (or remove) the | ||||||
| ``reversed`` keyword when you're iterating on ``date_list`` in a template:: | ``reversed`` keyword when you're iterating on ``date_list`` in a template: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% for date in date_list reversed %} |     {% for date in date_list reversed %} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -463,11 +463,14 @@ explicitly import the User model in your test module:: | |||||||
|     @override_settings(AUTH_USER_MODEL='auth.CustomUser') |     @override_settings(AUTH_USER_MODEL='auth.CustomUser') | ||||||
|     class CustomUserFeatureTests(TestCase): |     class CustomUserFeatureTests(TestCase): | ||||||
|         def test_something(self): |         def test_something(self): | ||||||
|             # Test code here ... |             # Test code here | ||||||
|  |             ... | ||||||
|  |  | ||||||
| This import forces the custom user model to be registered. Without this import, | This import forces the custom user model to be registered. Without this import, | ||||||
| the test will be unable to swap in the custom user model, and you will get an | the test will be unable to swap in the custom user model, and you will get an | ||||||
| error reporting:: | error reporting: | ||||||
|  |  | ||||||
|  | .. code-block:: pytb | ||||||
|  |  | ||||||
|     ImproperlyConfigured: AUTH_USER_MODEL refers to model 'auth.CustomUser' that has not been installed |     ImproperlyConfigured: AUTH_USER_MODEL refers to model 'auth.CustomUser' that has not been installed | ||||||
|  |  | ||||||
| @@ -1010,11 +1013,15 @@ Django 1.6 starts a process to correct this inconsistency. The ``future`` | |||||||
| template library provides alternate implementations of :ttag:`cycle` and | template library provides alternate implementations of :ttag:`cycle` and | ||||||
| :ttag:`firstof` that autoescape their inputs. If you're using these tags, | :ttag:`firstof` that autoescape their inputs. If you're using these tags, | ||||||
| you're encouraged to include the following line at the top of your templates to | you're encouraged to include the following line at the top of your templates to | ||||||
| enable the new behavior:: | enable the new behavior: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% load cycle from future %} |     {% load cycle from future %} | ||||||
|  |  | ||||||
| or:: | or: | ||||||
|  |  | ||||||
|  | .. code-block:: html+django | ||||||
|  |  | ||||||
|     {% load firstof from future %} |     {% load firstof from future %} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1009,7 +1009,9 @@ Standalone scripts | |||||||
| If you're using Django in a plain Python script — rather than a management | If you're using Django in a plain Python script — rather than a management | ||||||
| command — and you rely on the :envvar:`DJANGO_SETTINGS_MODULE` environment | command — and you rely on the :envvar:`DJANGO_SETTINGS_MODULE` environment | ||||||
| variable, you must now explicitly initialize Django at the beginning of your | variable, you must now explicitly initialize Django at the beginning of your | ||||||
| script with:: | script with: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> import django |     >>> import django | ||||||
|     >>> django.setup() |     >>> django.setup() | ||||||
| @@ -1134,13 +1136,13 @@ method:: | |||||||
|  |  | ||||||
|     # Old pattern: |     # Old pattern: | ||||||
|     try: |     try: | ||||||
|         # ... |         ... | ||||||
|     except ValidationError as e: |     except ValidationError as e: | ||||||
|         self._errors = e.update_error_dict(self._errors) |         self._errors = e.update_error_dict(self._errors) | ||||||
|  |  | ||||||
|     # New pattern: |     # New pattern: | ||||||
|     try: |     try: | ||||||
|         # ... |         ... | ||||||
|     except ValidationError as e: |     except ValidationError as e: | ||||||
|         self.add_error(None, e) |         self.add_error(None, e) | ||||||
|  |  | ||||||
| @@ -1150,7 +1152,7 @@ wasn't available before Django 1.7, but you can use the following | |||||||
| workaround to convert any ``list`` into ``ErrorList``:: | workaround to convert any ``list`` into ``ErrorList``:: | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         # ... |         ... | ||||||
|     except ValidationError as e: |     except ValidationError as e: | ||||||
|         self._errors = e.update_error_dict(self._errors) |         self._errors = e.update_error_dict(self._errors) | ||||||
|  |  | ||||||
| @@ -1195,8 +1197,8 @@ manager will *not* be reset. This was necessary to resolve an inconsistency in | |||||||
| the way routing information cascaded over joins. See :ticket:`13724` for more | the way routing information cascaded over joins. See :ticket:`13724` for more | ||||||
| details. | details. | ||||||
|  |  | ||||||
| pytz may be required | ``pytz`` may be required | ||||||
| -------------------- | ------------------------ | ||||||
|  |  | ||||||
| If your project handles datetimes before 1970 or after 2037 and Django raises | If your project handles datetimes before 1970 or after 2037 and Django raises | ||||||
| a :exc:`ValueError` when encountering them, you will have to install pytz_. You | a :exc:`ValueError` when encountering them, you will have to install pytz_. You | ||||||
|   | |||||||
| @@ -696,7 +696,9 @@ Assigning unsaved objects to relations raises an error | |||||||
| .. note:: | .. note:: | ||||||
|  |  | ||||||
|     To more easily allow in-memory usage of models, this change was reverted in |     To more easily allow in-memory usage of models, this change was reverted in | ||||||
|     Django 1.8.4 and replaced with a check during ``model.save()``. For example:: |     Django 1.8.4 and replaced with a check during ``model.save()``. For example: | ||||||
|  |  | ||||||
|  |     .. code-block:: pycon | ||||||
|  |  | ||||||
|         >>> book = Book.objects.create(name="Django") |         >>> book = Book.objects.create(name="Django") | ||||||
|         >>> book.author = Author(name="John") |         >>> book.author = Author(name="John") | ||||||
| @@ -713,7 +715,9 @@ Assigning unsaved objects to a :class:`~django.db.models.ForeignKey`, | |||||||
| :class:`~django.db.models.OneToOneField` now raises a :exc:`ValueError`. | :class:`~django.db.models.OneToOneField` now raises a :exc:`ValueError`. | ||||||
|  |  | ||||||
| Previously, the assignment of an unsaved object would be silently ignored. | Previously, the assignment of an unsaved object would be silently ignored. | ||||||
| For example:: | For example: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> book = Book.objects.create(name="Django") |     >>> book = Book.objects.create(name="Django") | ||||||
|     >>> book.author = Author(name="John") |     >>> book.author = Author(name="John") | ||||||
| @@ -724,7 +728,9 @@ For example:: | |||||||
|     >>> book.author |     >>> book.author | ||||||
|     >>> |     >>> | ||||||
|  |  | ||||||
| Now, an error will be raised to prevent data loss:: | Now, an error will be raised to prevent data loss: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> book.author = Author(name="john") |     >>> book.author = Author(name="john") | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
| @@ -790,7 +796,9 @@ Querying for model lookups now checks if the object passed is of correct type | |||||||
| and raises a :exc:`ValueError` if not. Previously, Django didn't care if the | and raises a :exc:`ValueError` if not. Previously, Django didn't care if the | ||||||
| object was of correct type; it just used the object's related field attribute | object was of correct type; it just used the object's related field attribute | ||||||
| (e.g. ``id``) for the lookup. Now, an error is raised to prevent incorrect | (e.g. ``id``) for the lookup. Now, an error is raised to prevent incorrect | ||||||
| lookups:: | lookups: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> book = Book.objects.create(name="Django") |     >>> book = Book.objects.create(name="Django") | ||||||
|     >>> book = Book.objects.filter(author=book) |     >>> book = Book.objects.filter(author=book) | ||||||
| @@ -802,14 +810,18 @@ lookups:: | |||||||
| -------------------------------------------- | -------------------------------------------- | ||||||
|  |  | ||||||
| ``select_related()`` now validates that the given fields actually exist. | ``select_related()`` now validates that the given fields actually exist. | ||||||
| Previously, nonexistent fields were silently ignored. Now, an error is raised:: | Previously, nonexistent fields were silently ignored. Now, an error is raised: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> book = Book.objects.select_related('nonexistent_field') |     >>> book = Book.objects.select_related('nonexistent_field') | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
|     ... |     ... | ||||||
|     FieldError: Invalid field name(s) given in select_related: 'nonexistent_field' |     FieldError: Invalid field name(s) given in select_related: 'nonexistent_field' | ||||||
|  |  | ||||||
| The validation also makes sure that the given field is relational:: | The validation also makes sure that the given field is relational: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> book = Book.objects.select_related('name') |     >>> book = Book.objects.select_related('name') | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
| @@ -966,7 +978,9 @@ returns the relationship as ``django.db.models.fields.related.ManyToOneRel`` | |||||||
| is set to the target of the relationship instead of the source. The source | is set to the target of the relationship instead of the source. The source | ||||||
| model is accessible on the ``related_model`` attribute instead. | model is accessible on the ``related_model`` attribute instead. | ||||||
|  |  | ||||||
| Consider this example from the tutorial in Django 1.8:: | Consider this example from the tutorial in Django 1.8: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> p = Poll.objects.get(pk=1) |     >>> p = Poll.objects.get(pk=1) | ||||||
|     >>> p._meta.get_all_related_objects() |     >>> p._meta.get_all_related_objects() | ||||||
| @@ -976,7 +990,9 @@ Consider this example from the tutorial in Django 1.8:: | |||||||
|     >>> p._meta.get_all_related_objects()[0].related_model |     >>> p._meta.get_all_related_objects()[0].related_model | ||||||
|     <class 'polls.models.Choice'> |     <class 'polls.models.Choice'> | ||||||
|  |  | ||||||
| and compare it to the behavior on older versions:: | and compare it to the behavior on older versions: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> p._meta.get_all_related_objects() |     >>> p._meta.get_all_related_objects() | ||||||
|     [<RelatedObject: polls:choice related to poll>] |     [<RelatedObject: polls:choice related to poll>] | ||||||
|   | |||||||
| @@ -1018,7 +1018,9 @@ to 2.1.4. jQuery 2.x has the same API as jQuery 1.x, but does not support | |||||||
| Internet Explorer 6, 7, or 8, allowing for better performance and a smaller | Internet Explorer 6, 7, or 8, allowing for better performance and a smaller | ||||||
| file size. If you need to support IE8 and must also use the latest version of | file size. If you need to support IE8 and must also use the latest version of | ||||||
| Django, you can override the admin's copy of jQuery with your own by creating | Django, you can override the admin's copy of jQuery with your own by creating | ||||||
| a Django application with this structure:: | a Django application with this structure: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     app/static/admin/js/vendor/ |     app/static/admin/js/vendor/ | ||||||
|         jquery.js |         jquery.js | ||||||
| @@ -1029,7 +1031,9 @@ a Django application with this structure:: | |||||||
| ``SyntaxError`` when installing Django setuptools 5.5.x | ``SyntaxError`` when installing Django setuptools 5.5.x | ||||||
| ------------------------------------------------------- | ------------------------------------------------------- | ||||||
|  |  | ||||||
| When installing Django 1.9 or 1.9.1 with setuptools 5.5.x, you'll see:: | When installing Django 1.9 or 1.9.1 with setuptools 5.5.x, you'll see: | ||||||
|  |  | ||||||
|  | .. code-block:: shell | ||||||
|  |  | ||||||
|     Compiling django/conf/app_template/apps.py ... |     Compiling django/conf/app_template/apps.py ... | ||||||
|       File "django/conf/app_template/apps.py", line 4 |       File "django/conf/app_template/apps.py", line 4 | ||||||
| @@ -1313,7 +1317,9 @@ the included URLconf sets an application namespace. | |||||||
| ``current_app`` parameter to ``contrib.auth`` views | ``current_app`` parameter to ``contrib.auth`` views | ||||||
| --------------------------------------------------- | --------------------------------------------------- | ||||||
|  |  | ||||||
| All views in ``django.contrib.auth.views`` have the following structure:: | All views in ``django.contrib.auth.views`` have the following structure: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     def view(request, ..., current_app=None, ...): |     def view(request, ..., current_app=None, ...): | ||||||
|  |  | ||||||
|   | |||||||
| @@ -454,7 +454,9 @@ If you wish to keep this restriction in the admin when editing users, set | |||||||
|  |  | ||||||
| Calling ``QuerySet.reverse()`` or ``last()`` on a sliced queryset leads to | Calling ``QuerySet.reverse()`` or ``last()`` on a sliced queryset leads to | ||||||
| unexpected results due to the slice being applied after reordering. This is | unexpected results due to the slice being applied after reordering. This is | ||||||
| now prohibited, e.g.:: | now prohibited, e.g.: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Model.objects.all()[:2].reverse() |     >>> Model.objects.all()[:2].reverse() | ||||||
|     Traceback (most recent call last): |     Traceback (most recent call last): | ||||||
|   | |||||||
| @@ -392,7 +392,9 @@ existing row will result in an ``IntegrityError``. | |||||||
|  |  | ||||||
| In order to update an existing model for a specific primary key value, use the | In order to update an existing model for a specific primary key value, use the | ||||||
| :meth:`~django.db.models.query.QuerySet.update_or_create` method or | :meth:`~django.db.models.query.QuerySet.update_or_create` method or | ||||||
| ``QuerySet.filter(pk=…).update(…)`` instead. For example:: | ``QuerySet.filter(pk=…).update(…)`` instead. For example: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> MyModel.objects.update_or_create(pk=existing_pk, defaults={'name': 'new name'}) |     >>> MyModel.objects.update_or_create(pk=existing_pk, defaults={'name': 'new name'}) | ||||||
|     >>> MyModel.objects.filter(pk=existing_pk).update(name='new name') |     >>> MyModel.objects.filter(pk=existing_pk).update(name='new name') | ||||||
|   | |||||||
| @@ -706,8 +706,8 @@ Miscellaneous | |||||||
|   ``SimpleTestCase.assertFormError()`` and ``assertFormsetError()`` is |   ``SimpleTestCase.assertFormError()`` and ``assertFormsetError()`` is | ||||||
|   deprecated. Use:: |   deprecated. Use:: | ||||||
|  |  | ||||||
|     assertFormError(response.context['form_name'], …) |     assertFormError(response.context['form_name'], ...) | ||||||
|     assertFormsetError(response.context['formset_name'], …) |     assertFormsetError(response.context['formset_name'], ...) | ||||||
|  |  | ||||||
|   or pass the form/formset object directly instead. |   or pass the form/formset object directly instead. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -93,11 +93,11 @@ Detailed notes can be found in :ref:`async-queries`, but in short: | |||||||
|  |  | ||||||
| Django also supports some asynchronous model methods that use the database:: | Django also supports some asynchronous model methods that use the database:: | ||||||
|  |  | ||||||
|     async def make_book(...): |     async def make_book(*args, **kwargs): | ||||||
|         book = Book(...) |         book = Book(...) | ||||||
|         await book.asave(using="secondary") |         await book.asave(using="secondary") | ||||||
|  |  | ||||||
|     async def make_book_with_tags(tags, ...): |     async def make_book_with_tags(tags, *args, **kwargs): | ||||||
|         book = await Book.objects.acreate(...) |         book = await Book.objects.acreate(...) | ||||||
|         await book.tags.aset(tags) |         await book.tags.aset(tags) | ||||||
|  |  | ||||||
| @@ -229,13 +229,13 @@ as either a direct wrapper or a decorator:: | |||||||
|  |  | ||||||
|     from asgiref.sync import async_to_sync |     from asgiref.sync import async_to_sync | ||||||
|  |  | ||||||
|     async def get_data(...): |     async def get_data(): | ||||||
|         ... |         ... | ||||||
|  |  | ||||||
|     sync_get_data = async_to_sync(get_data) |     sync_get_data = async_to_sync(get_data) | ||||||
|  |  | ||||||
|     @async_to_sync |     @async_to_sync | ||||||
|     async def get_other_data(...): |     async def get_other_data(): | ||||||
|         ... |         ... | ||||||
|  |  | ||||||
| The async function is run in the event loop for the current thread, if one is | The async function is run in the event loop for the current thread, if one is | ||||||
| @@ -266,7 +266,7 @@ as either a direct wrapper or a decorator:: | |||||||
|     async_function = sync_to_async(sensitive_sync_function, thread_sensitive=True) |     async_function = sync_to_async(sensitive_sync_function, thread_sensitive=True) | ||||||
|  |  | ||||||
|     @sync_to_async |     @sync_to_async | ||||||
|     def sync_function(...): |     def sync_function(): | ||||||
|         ... |         ... | ||||||
|  |  | ||||||
| Threadlocals and contextvars values are preserved across the boundary in both | Threadlocals and contextvars values are preserved across the boundary in both | ||||||
|   | |||||||
| @@ -134,8 +134,10 @@ Authenticating users | |||||||
|         user = authenticate(username='john', password='secret') |         user = authenticate(username='john', password='secret') | ||||||
|         if user is not None: |         if user is not None: | ||||||
|             # A backend authenticated the credentials |             # A backend authenticated the credentials | ||||||
|  |             ... | ||||||
|         else: |         else: | ||||||
|             # No backend authenticated the credentials |             # No backend authenticated the credentials | ||||||
|  |             ... | ||||||
|  |  | ||||||
|     ``request`` is an optional :class:`~django.http.HttpRequest` which is |     ``request`` is an optional :class:`~django.http.HttpRequest` which is | ||||||
|     passed on the ``authenticate()`` method of the authentication backends. |     passed on the ``authenticate()`` method of the authentication backends. | ||||||
| @@ -950,7 +952,9 @@ in your own URLconf, for example:: | |||||||
|         path('accounts/', include('django.contrib.auth.urls')), |         path('accounts/', include('django.contrib.auth.urls')), | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
| This will include the following URL patterns:: | This will include the following URL patterns: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     accounts/login/ [name='login'] |     accounts/login/ [name='login'] | ||||||
|     accounts/logout/ [name='logout'] |     accounts/logout/ [name='logout'] | ||||||
|   | |||||||
| @@ -24,7 +24,9 @@ How Django stores passwords | |||||||
| Django provides a flexible password storage system and uses PBKDF2 by default. | Django provides a flexible password storage system and uses PBKDF2 by default. | ||||||
|  |  | ||||||
| The :attr:`~django.contrib.auth.models.User.password` attribute of a | The :attr:`~django.contrib.auth.models.User.password` attribute of a | ||||||
| :class:`~django.contrib.auth.models.User` object is a string in this format:: | :class:`~django.contrib.auth.models.User` object is a string in this format: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     <algorithm>$<iterations>$<salt>$<hash> |     <algorithm>$<iterations>$<salt>$<hash> | ||||||
|  |  | ||||||
| @@ -215,7 +217,8 @@ parameter (use the ``rounds`` parameter when subclassing a bcrypt hasher). For | |||||||
| example, to increase the number of iterations used by the default PBKDF2 | example, to increase the number of iterations used by the default PBKDF2 | ||||||
| algorithm: | algorithm: | ||||||
|  |  | ||||||
| #. Create a subclass of ``django.contrib.auth.hashers.PBKDF2PasswordHasher``:: | #. Create a subclass of ``django.contrib.auth.hashers.PBKDF2PasswordHasher`` | ||||||
|  |    :: | ||||||
|  |  | ||||||
|         from django.contrib.auth.hashers import PBKDF2PasswordHasher |         from django.contrib.auth.hashers import PBKDF2PasswordHasher | ||||||
|  |  | ||||||
|   | |||||||
| @@ -18,7 +18,9 @@ That's where caching comes in. | |||||||
|  |  | ||||||
| To cache something is to save the result of an expensive calculation so that | To cache something is to save the result of an expensive calculation so that | ||||||
| you don't have to perform the calculation next time. Here's some pseudocode | you don't have to perform the calculation next time. Here's some pseudocode | ||||||
| explaining how this would work for a dynamically generated web page:: | explaining how this would work for a dynamically generated web page: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     given a URL, try finding that page in the cache |     given a URL, try finding that page in the cache | ||||||
|     if the page is in the cache: |     if the page is in the cache: | ||||||
|   | |||||||
| @@ -150,7 +150,7 @@ this would lead to incorrect behavior. | |||||||
|     @etag(etag_func) |     @etag(etag_func) | ||||||
|     @last_modified(last_modified_func) |     @last_modified(last_modified_func) | ||||||
|     def my_view(request): |     def my_view(request): | ||||||
|         # ... |         ... | ||||||
|  |  | ||||||
|     # End of bad code. |     # End of bad code. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -74,9 +74,9 @@ p2 doesn't have an associated restaurant: | |||||||
|  |  | ||||||
|     >>> from django.core.exceptions import ObjectDoesNotExist |     >>> from django.core.exceptions import ObjectDoesNotExist | ||||||
|     >>> try: |     >>> try: | ||||||
|     >>>     p2.restaurant |     ...     p2.restaurant | ||||||
|     >>> except ObjectDoesNotExist: |     ... except ObjectDoesNotExist: | ||||||
|     >>>     print("There is no restaurant here.") |     ...     print("There is no restaurant here.") | ||||||
|     There is no restaurant here. |     There is no restaurant here. | ||||||
|  |  | ||||||
| You can also use ``hasattr`` to avoid the need for exception catching: | You can also use ``hasattr`` to avoid the need for exception catching: | ||||||
|   | |||||||
| @@ -675,7 +675,9 @@ Field name restrictions | |||||||
| Django places some restrictions on model field names: | Django places some restrictions on model field names: | ||||||
|  |  | ||||||
| #. A field name cannot be a Python reserved word, because that would result | #. A field name cannot be a Python reserved word, because that would result | ||||||
|    in a Python syntax error. For example:: |    in a Python syntax error. For example: | ||||||
|  |  | ||||||
|  |    .. code-block:: text | ||||||
|  |  | ||||||
|        class Example(models.Model): |        class Example(models.Model): | ||||||
|            pass = models.IntegerField() # 'pass' is a reserved word! |            pass = models.IntegerField() # 'pass' is a reserved word! | ||||||
| @@ -1221,7 +1223,9 @@ subclass with a :class:`~django.db.models.ManyToManyField`:: | |||||||
|     class Supplier(Place): |     class Supplier(Place): | ||||||
|         customers = models.ManyToManyField(Place) |         customers = models.ManyToManyField(Place) | ||||||
|  |  | ||||||
| This results in the error:: | This results in the error: | ||||||
|  |  | ||||||
|  | .. code-block:: pytb | ||||||
|  |  | ||||||
|     Reverse query name for 'Supplier.customers' clashes with reverse query |     Reverse query name for 'Supplier.customers' clashes with reverse query | ||||||
|     name for 'Supplier.place_ptr'. |     name for 'Supplier.place_ptr'. | ||||||
|   | |||||||
| @@ -1075,7 +1075,7 @@ query for SQL ``NULL``, use :lookup:`isnull`: | |||||||
|     <Dog: Archie> |     <Dog: Archie> | ||||||
|     >>> Dog.objects.filter(data=None) |     >>> Dog.objects.filter(data=None) | ||||||
|     <QuerySet [<Dog: Archie>]> |     <QuerySet [<Dog: Archie>]> | ||||||
|     >>> Dog.objects.filter(data=Value(None, JSONField()) |     >>> Dog.objects.filter(data=Value(None, JSONField())) | ||||||
|     <QuerySet [<Dog: Archie>]> |     <QuerySet [<Dog: Archie>]> | ||||||
|     >>> Dog.objects.filter(data__isnull=True) |     >>> Dog.objects.filter(data__isnull=True) | ||||||
|     <QuerySet [<Dog: Max>]> |     <QuerySet [<Dog: Max>]> | ||||||
| @@ -1364,7 +1364,9 @@ For example, this statement yields a single ``Q`` object that represents the | |||||||
|  |  | ||||||
|     Q(question__startswith='Who') | Q(question__startswith='What') |     Q(question__startswith='Who') | Q(question__startswith='What') | ||||||
|  |  | ||||||
| This is equivalent to the following SQL ``WHERE`` clause:: | This is equivalent to the following SQL ``WHERE`` clause: | ||||||
|  |  | ||||||
|  | .. code-block: sql | ||||||
|  |  | ||||||
|     WHERE question LIKE 'Who%' OR question LIKE 'What%' |     WHERE question LIKE 'Who%' OR question LIKE 'What%' | ||||||
|  |  | ||||||
|   | |||||||
| @@ -309,7 +309,8 @@ alias:: | |||||||
|  |  | ||||||
|     from django.db import connections |     from django.db import connections | ||||||
|     with connections['my_db_alias'].cursor() as cursor: |     with connections['my_db_alias'].cursor() as cursor: | ||||||
|         # Your code here... |         # Your code here | ||||||
|  |         ... | ||||||
|  |  | ||||||
| By default, the Python DB API will return results without their field names, | By default, the Python DB API will return results without their field names, | ||||||
| which means you end up with a ``list`` of values, rather than a ``dict``. At a | which means you end up with a ``list`` of values, rather than a ``dict``. At a | ||||||
|   | |||||||
| @@ -111,7 +111,7 @@ requirements:: | |||||||
|         css = { |         css = { | ||||||
|             'screen': ['pretty.css'], |             'screen': ['pretty.css'], | ||||||
|             'tv,projector': ['lo_res.css'], |             'tv,projector': ['lo_res.css'], | ||||||
|             'print': ['newspaper.css], |             'print': ['newspaper.css'], | ||||||
|         } |         } | ||||||
|  |  | ||||||
| If this last CSS definition were to be rendered, it would become the following HTML: | If this last CSS definition were to be rendered, it would become the following HTML: | ||||||
| @@ -287,7 +287,7 @@ outputting the complete HTML ``<script>`` or ``<link>`` tag content: | |||||||
|     >>> from django.utils.html import html_safe |     >>> from django.utils.html import html_safe | ||||||
|     >>> |     >>> | ||||||
|     >>> @html_safe |     >>> @html_safe | ||||||
|     >>> class JSPath: |     ... class JSPath: | ||||||
|     ...     def __str__(self): |     ...     def __str__(self): | ||||||
|     ...         return '<script src="https://example.org/asset.js" rel="stylesheet">' |     ...         return '<script src="https://example.org/asset.js" rel="stylesheet">' | ||||||
|  |  | ||||||
|   | |||||||
| @@ -778,7 +778,9 @@ keyword arguments, or the corresponding attributes on the ``ModelForm`` inner | |||||||
| ``Meta`` class. Please see the ``ModelForm`` :ref:`modelforms-selecting-fields` | ``Meta`` class. Please see the ``ModelForm`` :ref:`modelforms-selecting-fields` | ||||||
| documentation. | documentation. | ||||||
|  |  | ||||||
| ... or enable localization for specific fields:: | ... or enable localization for specific fields: | ||||||
|  |  | ||||||
|  | .. code-block:: pycon | ||||||
|  |  | ||||||
|     >>> Form = modelform_factory(Author, form=AuthorForm, localized_fields=["birth_date"]) |     >>> Form = modelform_factory(Author, form=AuthorForm, localized_fields=["birth_date"]) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -127,13 +127,15 @@ You can use the :func:`redirect` function in a number of ways. | |||||||
|             ... |             ... | ||||||
|             return redirect('some-view-name', foo='bar') |             return redirect('some-view-name', foo='bar') | ||||||
|  |  | ||||||
| #. By passing a hardcoded URL to redirect to:: | #. By passing a hardcoded URL to redirect to: | ||||||
|  |    :: | ||||||
|  |  | ||||||
|         def my_view(request): |         def my_view(request): | ||||||
|             ... |             ... | ||||||
|             return redirect('/some/url/') |             return redirect('/some/url/') | ||||||
|  |  | ||||||
|    This also works with full URLs:: |    This also works with full URLs: | ||||||
|  |    :: | ||||||
|  |  | ||||||
|         def my_view(request): |         def my_view(request): | ||||||
|             ... |             ... | ||||||
|   | |||||||
| @@ -865,7 +865,9 @@ Secondly, you can include an object that contains embedded namespace data. If | |||||||
| you ``include()`` a list of :func:`~django.urls.path` or | you ``include()`` a list of :func:`~django.urls.path` or | ||||||
| :func:`~django.urls.re_path` instances, the URLs contained in that object | :func:`~django.urls.re_path` instances, the URLs contained in that object | ||||||
| will be added to the global namespace. However, you can also ``include()`` a | will be added to the global namespace. However, you can also ``include()`` a | ||||||
| 2-tuple containing:: | 2-tuple containing: | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|     (<list of path()/re_path() instances>, <application namespace>) |     (<list of path()/re_path() instances>, <application namespace>) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -262,7 +262,9 @@ In a case like this, consider something like the following:: | |||||||
|         } |         } | ||||||
|  |  | ||||||
|     You would get an error when running :djadmin:`django-admin |     You would get an error when running :djadmin:`django-admin | ||||||
|     compilemessages <compilemessages>`:: |     compilemessages <compilemessages>`: | ||||||
|  |  | ||||||
|  |     .. code-block: pytb | ||||||
|  |  | ||||||
|         a format specification for argument 'name', as in 'msgstr[0]', doesn't exist in 'msgid' |         a format specification for argument 'name', as in 'msgstr[0]', doesn't exist in 'msgid' | ||||||
|  |  | ||||||
| @@ -1094,7 +1096,9 @@ interface within your Python code:: | |||||||
| ~~~~~~~~~~~~ | ~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| The ``ngettext`` function provides an interface to pluralize words and | The ``ngettext`` function provides an interface to pluralize words and | ||||||
| phrases:: | phrases: | ||||||
|  |  | ||||||
|  | .. code-block:: javascript | ||||||
|  |  | ||||||
|     const objectCount = 1 // or 0, or 2, or 3, ... |     const objectCount = 1 // or 0, or 2, or 3, ... | ||||||
|     const string = ngettext( |     const string = ngettext( | ||||||
| @@ -1113,7 +1117,9 @@ function supports both positional and named interpolation: | |||||||
| * Positional interpolation: ``obj`` contains a JavaScript Array object | * Positional interpolation: ``obj`` contains a JavaScript Array object | ||||||
|   whose elements values are then sequentially interpolated in their |   whose elements values are then sequentially interpolated in their | ||||||
|   corresponding ``fmt`` placeholders in the same order they appear. |   corresponding ``fmt`` placeholders in the same order they appear. | ||||||
|   For example:: |   For example: | ||||||
|  |  | ||||||
|  |   .. code-block:: javascript | ||||||
|  |  | ||||||
|     const formats = ngettext( |     const formats = ngettext( | ||||||
|       'There is %s object. Remaining: %s', |       'There is %s object. Remaining: %s', | ||||||
| @@ -1125,7 +1131,9 @@ function supports both positional and named interpolation: | |||||||
|  |  | ||||||
| * Named interpolation: This mode is selected by passing the optional | * Named interpolation: This mode is selected by passing the optional | ||||||
|   boolean ``named`` parameter as ``true``. ``obj`` contains a JavaScript |   boolean ``named`` parameter as ``true``. ``obj`` contains a JavaScript | ||||||
|   object or associative array. For example:: |   object or associative array. For example: | ||||||
|  |  | ||||||
|  |   .. code-block:: javascript | ||||||
|  |  | ||||||
|     const data = { |     const data = { | ||||||
|       count: 10, |       count: 10, | ||||||
| @@ -1149,7 +1157,9 @@ to produce proper pluralizations). | |||||||
| ~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| The ``get_format`` function has access to the configured i18n formatting | The ``get_format`` function has access to the configured i18n formatting | ||||||
| settings and can retrieve the format string for a given setting name:: | settings and can retrieve the format string for a given setting name: | ||||||
|  |  | ||||||
|  | .. code-block:: javascript | ||||||
|  |  | ||||||
|     document.write(get_format('DATE_FORMAT')); |     document.write(get_format('DATE_FORMAT')); | ||||||
|     // 'N j, Y' |     // 'N j, Y' | ||||||
| @@ -1199,7 +1209,9 @@ translated word:: | |||||||
|  |  | ||||||
| The ``npgettext`` function also behaves like the Python variant | The ``npgettext`` function also behaves like the Python variant | ||||||
| (:func:`~django.utils.translation.npgettext()`), providing a **pluralized** | (:func:`~django.utils.translation.npgettext()`), providing a **pluralized** | ||||||
| contextually translated word:: | contextually translated word: | ||||||
|  |  | ||||||
|  | .. code-block:: javascript | ||||||
|  |  | ||||||
|     document.write(npgettext('group', 'party', 1)); |     document.write(npgettext('group', 'party', 1)); | ||||||
|     // party |     // party | ||||||
| @@ -1211,7 +1223,9 @@ contextually translated word:: | |||||||
|  |  | ||||||
| The ``pluralidx`` function works in a similar way to the :tfilter:`pluralize` | The ``pluralidx`` function works in a similar way to the :tfilter:`pluralize` | ||||||
| template filter, determining if a given ``count`` should use a plural form of | template filter, determining if a given ``count`` should use a plural form of | ||||||
| a word or not:: | a word or not: | ||||||
|  |  | ||||||
|  | .. code-block:: javascript | ||||||
|  |  | ||||||
|     document.write(pluralidx(0)); |     document.write(pluralidx(0)); | ||||||
|     // true |     // true | ||||||
|   | |||||||
| @@ -245,7 +245,7 @@ JSON in the following way:: | |||||||
|             "model": "sessions.session", |             "model": "sessions.session", | ||||||
|             "fields": { |             "fields": { | ||||||
|                 "expire_date": "2013-01-16T08:16:59.844Z", |                 "expire_date": "2013-01-16T08:16:59.844Z", | ||||||
|                 ... |                 # ... | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     ] |     ] | ||||||
|   | |||||||
| @@ -125,6 +125,7 @@ In your Django apps, use settings by importing the object | |||||||
|  |  | ||||||
|     if settings.DEBUG: |     if settings.DEBUG: | ||||||
|         # Do something |         # Do something | ||||||
|  |         ... | ||||||
|  |  | ||||||
| Note that ``django.conf.settings`` isn't a module -- it's an object. So | Note that ``django.conf.settings`` isn't a module -- it's an object. So | ||||||
| importing individual settings is not possible:: | importing individual settings is not possible:: | ||||||
|   | |||||||
| @@ -248,7 +248,9 @@ Use the ``django.test.Client`` class to make requests. | |||||||
|             >>> c = Client() |             >>> c = Client() | ||||||
|             >>> c.post('/login/', {'name': 'fred', 'passwd': 'secret'}) |             >>> c.post('/login/', {'name': 'fred', 'passwd': 'secret'}) | ||||||
|  |  | ||||||
|         ...will result in the evaluation of a POST request to this URL:: |         ...will result in the evaluation of a POST request to this URL: | ||||||
|  |  | ||||||
|  |         .. code-block:: text | ||||||
|  |  | ||||||
|             /login/ |             /login/ | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user