diff --git a/docs/Makefile b/docs/Makefile
index d97a7ff07c..596e7fc2dd 100644
--- a/docs/Makefile
+++ b/docs/Makefile
@@ -61,7 +61,7 @@ html:
@echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
htmlview: html
- $(PYTHON) -c "import webbrowser; webbrowser.open('_build/html/index.html')"
+ $(PYTHON) -m webbrowser "$(BUILDDIR)/html/index.html"
dirhtml:
$(SPHINXBUILD) -b dirhtml $(ALLSPHINXOPTS) $(BUILDDIR)/dirhtml
diff --git a/docs/conf.py b/docs/conf.py
index b72b1afcf5..9289e821fa 100644
--- a/docs/conf.py
+++ b/docs/conf.py
@@ -13,6 +13,8 @@ import functools
import sys
from os.path import abspath, dirname, join
+from sphinx import version_info as sphinx_version
+
# Workaround for sphinx-build recursion limit overflow:
# pickle.dump(doctree, f, pickle.HIGHEST_PROTOCOL)
# RuntimeError: maximum recursion depth exceeded while pickling an object
@@ -138,13 +140,15 @@ django_next_version = "5.2"
extlinks = {
"bpo": ("https://bugs.python.org/issue?@action=redirect&bpo=%s", "bpo-%s"),
"commit": ("https://github.com/django/django/commit/%s", "%s"),
- "cve": ("https://nvd.nist.gov/vuln/detail/CVE-%s", "CVE-%s"),
"pypi": ("https://pypi.org/project/%s/", "%s"),
# A file or directory. GitHub redirects from blob to tree if needed.
"source": ("https://github.com/django/django/blob/main/%s", "%s"),
"ticket": ("https://code.djangoproject.com/ticket/%s", "#%s"),
}
+if sphinx_version < (8, 1):
+ extlinks["cve"] = ("https://www.cve.org/CVERecord?id=CVE-%s", "CVE-%s")
+
# The language for content autogenerated by Sphinx. Refer to documentation
# for a list of supported languages.
# language = None
diff --git a/docs/howto/auth-remote-user.txt b/docs/howto/auth-remote-user.txt
index 19b25432fe..f8492e367a 100644
--- a/docs/howto/auth-remote-user.txt
+++ b/docs/howto/auth-remote-user.txt
@@ -6,12 +6,11 @@ This document describes how to make use of external authentication sources
(where the web server sets the ``REMOTE_USER`` environment variable) in your
Django applications. This type of authentication solution is typically seen on
intranet sites, with single sign-on solutions such as IIS and Integrated
-Windows Authentication or Apache and `mod_authnz_ldap`_, `CAS`_, `Cosign`_,
-`WebAuth`_, `mod_auth_sspi`_, etc.
+Windows Authentication or Apache and `mod_authnz_ldap`_, `CAS`_, `WebAuth`_,
+`mod_auth_sspi`_, etc.
-.. _mod_authnz_ldap: https://httpd.apache.org/docs/2.2/mod/mod_authnz_ldap.html
+.. _mod_authnz_ldap: https://httpd.apache.org/docs/current/mod/mod_authnz_ldap.html
.. _CAS: https://www.apereo.org/projects/cas
-.. _Cosign: http://weblogin.org
.. _WebAuth: https://uit.stanford.edu/service/authentication
.. _mod_auth_sspi: https://sourceforge.net/projects/mod-auth-sspi
diff --git a/docs/howto/deployment/asgi/hypercorn.txt b/docs/howto/deployment/asgi/hypercorn.txt
index ea5ce3cc72..3abd2d54ef 100644
--- a/docs/howto/deployment/asgi/hypercorn.txt
+++ b/docs/howto/deployment/asgi/hypercorn.txt
@@ -17,7 +17,7 @@ You can install Hypercorn with ``pip``:
Running Django in Hypercorn
===========================
-When Hypercorn is installed, a ``hypercorn`` command is available
+When :pypi:`Hypercorn` is installed, a ``hypercorn`` command is available
which runs ASGI applications. Hypercorn needs to be called with the
location of a module containing an ASGI application object, followed
by what the application is called (separated by a colon).
@@ -35,4 +35,4 @@ this command from the same directory as your ``manage.py`` file.
For more advanced usage, please read the `Hypercorn documentation
`_.
-.. _Hypercorn: https://pgjones.gitlab.io/hypercorn/
+.. _Hypercorn: https://hypercorn.readthedocs.io/
diff --git a/docs/howto/index.txt b/docs/howto/index.txt
index 0034032ce2..d799ca7906 100644
--- a/docs/howto/index.txt
+++ b/docs/howto/index.txt
@@ -1,11 +1,57 @@
-===============
-"How-to" guides
-===============
+=============
+How-to guides
+=============
-Here you'll find short answers to "How do I....?" types of questions. These
-how-to guides don't cover topics in depth -- you'll find that material in the
-:doc:`/topics/index` and the :doc:`/ref/index`. However, these guides will help
-you quickly accomplish common tasks.
+Practical guides covering common tasks and problems.
+
+Models, data and databases
+==========================
+
+.. toctree::
+ :maxdepth: 1
+
+ initial-data
+ legacy-databases
+ custom-model-fields
+ writing-migrations
+ custom-lookups
+
+Templates and output
+====================
+
+.. toctree::
+ :maxdepth: 1
+
+ outputting-csv
+ outputting-pdf
+ overriding-templates
+ custom-template-backend
+ custom-template-tags
+
+Project configuration and management
+====================================
+
+.. toctree::
+ :maxdepth: 1
+
+ static-files/index
+ logging
+ error-reporting
+ delete-app
+
+Installing, deploying and upgrading
+===================================
+
+.. toctree::
+ :maxdepth: 1
+
+ upgrade-version
+ windows
+ deployment/index
+ static-files/deployment
+
+Other guides
+============
.. toctree::
:maxdepth: 1
@@ -13,25 +59,7 @@ you quickly accomplish common tasks.
auth-remote-user
csrf
custom-management-commands
- custom-model-fields
- custom-lookups
- custom-template-backend
- custom-template-tags
custom-file-storage
- deployment/index
- upgrade-version
- error-reporting
- initial-data
- legacy-databases
- logging
- outputting-csv
- outputting-pdf
- overriding-templates
- static-files/index
- static-files/deployment
- windows
- writing-migrations
- delete-app
.. seealso::
diff --git a/docs/howto/overriding-templates.txt b/docs/howto/overriding-templates.txt
index f636948a20..f99a1203a8 100644
--- a/docs/howto/overriding-templates.txt
+++ b/docs/howto/overriding-templates.txt
@@ -111,15 +111,15 @@ reimplement the entire template.
For example, you can use this technique to add a custom logo to the
``admin/base_site.html`` template:
- .. code-block:: html+django
- :caption: ``templates/admin/base_site.html``
+.. code-block:: html+django
+ :caption: ``templates/admin/base_site.html``
- {% extends "admin/base_site.html" %}
+ {% extends "admin/base_site.html" %}
- {% block branding %}
-
- {{ block.super }}
- {% endblock %}
+ {% block branding %}
+
+ {{ block.super }}
+ {% endblock %}
Key points to note:
diff --git a/docs/internals/_images/triage_process.svg b/docs/internals/_images/triage_process.svg
index 2b5e0d3ced..6fbf1cbcc7 100644
--- a/docs/internals/_images/triage_process.svg
+++ b/docs/internals/_images/triage_process.svg
@@ -232,47 +232,47 @@
-
-
- The ticket was already reported, was
- already rejected, isn't a bug, doesn't contain
- enough information, or can't be reproduced.
+
+
+ The ticket was already reported, was
+ already rejected, isn't a bug, doesn't contain
+ enough information, or can't be reproduced.
-
+
-
+
-
-
- The ticket is a
- bug and should
- be fixed.
+
+
+ The ticket is a
+ bug and should
+ be fixed.
-
+
-
+
-
-
- The ticket has a patch which applies cleanly and includes all
- needed tests and docs. A merger can commit it as is.
+
+
+ The ticket has a patch which applies cleanly and includes all
+ needed tests and docs. A merger can commit it as is.
-
+
-
+
diff --git a/docs/internals/contributing/triaging-tickets.txt b/docs/internals/contributing/triaging-tickets.txt
index 852219c96c..7987d63e9a 100644
--- a/docs/internals/contributing/triaging-tickets.txt
+++ b/docs/internals/contributing/triaging-tickets.txt
@@ -49,8 +49,8 @@ attribute easily tells us what and who each ticket is waiting on.
Since a picture is worth a thousand words, let's start there:
.. image:: /internals/_images/triage_process.*
- :height: 501
- :width: 400
+ :height: 750
+ :width: 600
:alt: Django's ticket triage workflow
We've got two roles in this diagram:
diff --git a/docs/internals/contributing/writing-code/coding-style.txt b/docs/internals/contributing/writing-code/coding-style.txt
index c1838b77a3..20605aef56 100644
--- a/docs/internals/contributing/writing-code/coding-style.txt
+++ b/docs/internals/contributing/writing-code/coding-style.txt
@@ -417,7 +417,7 @@ Model style
* All database fields
* Custom manager attributes
* ``class Meta``
- * ``def __str__()``
+ * ``def __str__()`` and other Python magic methods
* ``def save()``
* ``def get_absolute_url()``
* Any custom methods
diff --git a/docs/internals/contributing/writing-code/submitting-patches.txt b/docs/internals/contributing/writing-code/submitting-patches.txt
index cac6848d04..799292e3fd 100644
--- a/docs/internals/contributing/writing-code/submitting-patches.txt
+++ b/docs/internals/contributing/writing-code/submitting-patches.txt
@@ -114,7 +114,7 @@ requirements:
feature, the change should also contain documentation.
When you think your work is ready to be reviewed, send :doc:`a GitHub pull
-request `.
+request `.
If you can't send a pull request for some reason, you can also use patches in
Trac. When using this style, follow these guidelines.
@@ -140,20 +140,63 @@ Regardless of the way you submit your work, follow these steps.
.. _ticket tracker: https://code.djangoproject.com/
.. _Development dashboard: https://dashboard.djangoproject.com/
-Non-trivial contributions
-=========================
+Contributions which require community feedback
+==============================================
-A "non-trivial" contribution is one that is more than a small bug fix. It's a
-change that introduces new Django functionality and makes some sort of design
-decision.
+A wider community discussion is required when a patch introduces new Django
+functionality and makes some sort of design decision. This is especially
+important if the approach involves a :ref:`deprecation `
+or introduces breaking changes.
-If you provide a non-trivial change, include evidence that alternatives have
-been discussed on the `Django Forum`_ or |django-developers| list.
+The following are different approaches for gaining feedback from the community.
-If you're not sure whether your contribution should be considered non-trivial,
-ask on the ticket for opinions.
+The Django Forum or django-developers mailing list
+--------------------------------------------------
+
+You can propose a change on the `Django Forum`_ or |django-developers| mailing
+list. You should explain the need for the change, go into details of the
+approach and discuss alternatives.
+
+Please include a link to such discussions in your contributions.
+
+Third party package
+-------------------
+
+Django does not accept experimental features. All features must follow our
+:ref:`deprecation policy `. Hence, it can
+take months or years for Django to iterate on an API design.
+
+If you need user feedback on a public interface, it is better to create a
+third-party package first. You can iterate on the public API much faster, while
+also validating the need for the feature.
+
+Once this package becomes stable and there are clear benefits of incorporating
+aspects into Django core, starting a discussion on the `Django Forum`_ or
+|django-developers| mailing list would be the next step.
+
+Django Enhancement Proposal (DEP)
+---------------------------------
+
+Similar to Python’s PEPs, Django has `Django Enhancement Proposals`_ or DEPs. A
+DEP is a design document which provides information to the Django community, or
+describes a new feature or process for Django. They provide concise technical
+specifications of features, along with rationales. DEPs are also the primary
+mechanism for proposing and collecting community input on major new features.
+
+Before considering writing a DEP, it is recommended to first open a discussion
+on the `Django Forum`_ or |django-developers| mailing list. This allows the
+community to provide feedback and helps refine the proposal. Once the DEP is
+ready the :ref:`Steering Council ` votes on whether to accept
+it.
+
+Some examples of DEPs that have been approved and fully implemented:
+
+* `DEP 181: ORM Expressions `_
+* `DEP 182: Multiple Template Engines `_
+* `DEP 201: Simplified routing syntax `_
.. _Django Forum: https://forum.djangoproject.com/
+.. _Django Enhancement Proposals: https://github.com/django/deps
.. _deprecating-a-feature:
diff --git a/docs/internals/contributing/writing-documentation.txt b/docs/internals/contributing/writing-documentation.txt
index 763039e61a..a8db5d93fd 100644
--- a/docs/internals/contributing/writing-documentation.txt
+++ b/docs/internals/contributing/writing-documentation.txt
@@ -159,9 +159,14 @@ Spelling check
Before you commit your docs, it's a good idea to run the spelling checker.
You'll need to install :pypi:`sphinxcontrib-spelling` first. Then from the
-``docs`` directory, run ``make spelling``. Wrong words (if any) along with the
-file and line number where they occur will be saved to
-``_build/spelling/output.txt``.
+``docs`` directory, run:
+
+.. console::
+
+ $ make spelling
+
+Wrong words (if any) along with the file and line number where they occur will
+be saved to ``_build/spelling/output.txt``.
If you encounter false-positives (error output that actually is correct), do
one of the following:
@@ -179,10 +184,21 @@ Link check
Links in documentation can become broken or changed such that they are no
longer the canonical link. Sphinx provides a builder that can check whether the
-links in the documentation are working. From the ``docs`` directory, run ``make
-linkcheck``. Output is printed to the terminal, but can also be found in
+links in the documentation are working. From the ``docs`` directory, run:
+
+.. console::
+
+ $ make linkcheck
+
+Output is printed to the terminal, but can also be found in
``_build/linkcheck/output.txt`` and ``_build/linkcheck/output.json``.
+.. warning::
+
+ The execution of the command requires an internet connection and takes
+ several minutes to complete, because the command tests all the links
+ that are found in the documentation.
+
Entries that have a status of "working" are fine, those that are "unchecked" or
"ignored" have been skipped because they either cannot be checked or have
matched ignore rules in the configuration.
@@ -290,7 +306,8 @@ documentation:
display a link with the title "auth".
* All Python code blocks should be formatted using the :pypi:`blacken-docs`
- auto-formatter. This will be run by ``pre-commit`` if that is configured.
+ auto-formatter. This will be run by :ref:`pre-commit
+ ` if that is configured.
* Use :mod:`~sphinx.ext.intersphinx` to reference Python's and Sphinx'
documentation.
@@ -324,8 +341,9 @@ documentation:
Five
^^^^
-* Use :rst:role:`:rfc:` to reference RFC and try to link to the relevant
- section if possible. For example, use ``:rfc:`2324#section-2.3.2``` or
+* Use :rst:role:`:rfc:` to reference a Request for Comments (RFC) and
+ try to link to the relevant section if possible. For example, use
+ ``:rfc:`2324#section-2.3.2``` or
``:rfc:`Custom link text <2324#section-2.3.2>```.
* Use :rst:role:`:pep:` to reference a Python Enhancement Proposal (PEP)
@@ -339,6 +357,9 @@ documentation:
also need to define a reference to the documentation for that environment
variable using :rst:dir:`.. envvar:: `.
+* Use :rst:role:`:cve:` to reference a Common Vulnerabilities and
+ Exposures (CVE) identifier. For example, use ``:cve:`2019-14232```.
+
Django-specific markup
======================
@@ -518,7 +539,7 @@ Minimizing images
Optimize image compression where possible. For PNG files, use OptiPNG and
AdvanceCOMP's ``advpng``:
-.. code-block:: console
+.. console::
$ cd docs
$ optipng -o7 -zm1-9 -i0 -strip all `find . -type f -not -path "./_build/*" -name "*.png"`
@@ -619,6 +640,10 @@ included in the Django repository and the releases as
``docs/man/django-admin.1``. There isn't a need to update this file when
updating the documentation, as it's updated once as part of the release process.
-To generate an updated version of the man page, run ``make man`` in the
-``docs`` directory. The new man page will be written in
-``docs/_build/man/django-admin.1``.
+To generate an updated version of the man page, in the ``docs`` directory, run:
+
+.. console::
+
+ $ make man
+
+The new man page will be written in ``docs/_build/man/django-admin.1``.
diff --git a/docs/internals/security.txt b/docs/internals/security.txt
index 55300b01e1..6aac9a6b66 100644
--- a/docs/internals/security.txt
+++ b/docs/internals/security.txt
@@ -38,6 +38,41 @@ action to be taken, you may receive further followup emails.
.. _our public Trac instance: https://code.djangoproject.com/query
+.. _security-report-evaluation:
+
+How does Django evaluate a report
+=================================
+
+These are criteria used by the security team when evaluating whether a report
+requires a security release:
+
+* The vulnerability is within a :ref:`supported version ` of
+ Django.
+
+* The vulnerability applies to a production-grade Django application. This means
+ the following do not require a security release:
+
+ * Exploits that only affect local development, for example when using
+ :djadmin:`runserver`.
+ * Exploits which fail to follow security best practices, such as failure to
+ sanitize user input. For other examples, see our :ref:`security
+ documentation `.
+ * Exploits in AI generated code that do not adhere to security best practices.
+
+The security team may conclude that the source of the vulnerability is within
+the Python standard library, in which case the reporter will be asked to report
+the vulnerability to the Python core team. For further details see the `Python
+security guidelines `_.
+
+On occasion, a security release may be issued to help resolve a security
+vulnerability within a popular third-party package. These reports should come
+from the package maintainers.
+
+If you are unsure whether your finding meets these criteria, please still report
+it :ref:`privately by emailing security@djangoproject.com
+`. The security team will review your report and
+recommend the correct course of action.
+
.. _security-support:
Supported versions
diff --git a/docs/intro/contributing.txt b/docs/intro/contributing.txt
index 7d590e76a2..0900fdae37 100644
--- a/docs/intro/contributing.txt
+++ b/docs/intro/contributing.txt
@@ -217,8 +217,7 @@ a dependency for one or more of the Python packages. Consult the failing
package's documentation or search the web with the error message that you
encounter.
-Now we are ready to run the test suite. If you're using GNU/Linux, macOS, or
-some other flavor of Unix, run:
+Now we are ready to run the test suite:
.. console::
diff --git a/docs/intro/overview.txt b/docs/intro/overview.txt
index 0c41446d01..af87a01bb4 100644
--- a/docs/intro/overview.txt
+++ b/docs/intro/overview.txt
@@ -309,7 +309,7 @@ Here's what the "base.html" template, including the use of :doc:`static files
:caption: ``templates/base.html``
{% load static %}
-
+
{% block title %}{% endblock %}
diff --git a/docs/intro/reusable-apps.txt b/docs/intro/reusable-apps.txt
index a9c0768e3b..5acf8c2b18 100644
--- a/docs/intro/reusable-apps.txt
+++ b/docs/intro/reusable-apps.txt
@@ -6,7 +6,7 @@ This advanced tutorial begins where :doc:`Tutorial 8 `
left off. We'll be turning our web-poll into a standalone Python package
you can reuse in new projects and share with other people.
-If you haven't recently completed Tutorials 1–7, we encourage you to review
+If you haven't recently completed Tutorials 1–8, we encourage you to review
these so that your example project matches the one described below.
Reusability matters
diff --git a/docs/intro/tutorial01.txt b/docs/intro/tutorial01.txt
index 041da0a404..c5ac5a1107 100644
--- a/docs/intro/tutorial01.txt
+++ b/docs/intro/tutorial01.txt
@@ -293,7 +293,8 @@ app will still work.
.. admonition:: When to use :func:`~django.urls.include()`
You should always use ``include()`` when you include other URL patterns.
- ``admin.site.urls`` is the only exception to this.
+ The only exception is ``admin.site.urls``, which is a pre-built URLconf
+ provided by Django for the default admin site.
You have now wired an ``index`` view into the URLconf. Verify it's working with
the following command:
diff --git a/docs/ref/checks.txt b/docs/ref/checks.txt
index d78a6f76b2..2308a854c7 100644
--- a/docs/ref/checks.txt
+++ b/docs/ref/checks.txt
@@ -77,6 +77,7 @@ Django's system checks are organized using the following tags:
* ``async_support``: Checks asynchronous-related configuration.
* ``caches``: Checks cache related configuration.
* ``compatibility``: Flags potential problems with version upgrades.
+* ``commands``: Checks custom management commands related configuration.
* ``database``: Checks database-related configuration issues. Database checks
are not run by default because they do more than static code analysis as
regular checks do. They are only run by the :djadmin:`migrate` command or if
@@ -428,6 +429,14 @@ Models
* **models.W047**: ```` does not support unique constraints with
nulls distinct.
+Management Commands
+-------------------
+
+The following checks verify custom management commands are correctly configured:
+
+* **commands.E001**: The ``migrate`` and ``makemigrations`` commands must have
+ the same ``autodetector``.
+
Security
--------
diff --git a/docs/ref/contrib/admin/index.txt b/docs/ref/contrib/admin/index.txt
index 407dd88e71..d9e12f0165 100644
--- a/docs/ref/contrib/admin/index.txt
+++ b/docs/ref/contrib/admin/index.txt
@@ -337,7 +337,8 @@ subclass::
If neither ``fields`` nor :attr:`~ModelAdmin.fieldsets` options are present,
Django will default to displaying each field that isn't an ``AutoField`` and
has ``editable=True``, in a single fieldset, in the same order as the fields
- are defined in the model.
+ are defined in the model, followed by any fields defined in
+ :attr:`~ModelAdmin.readonly_fields`.
.. attribute:: ModelAdmin.fieldsets
@@ -1465,6 +1466,27 @@ templates used by the :class:`ModelAdmin` views:
See also :ref:`saving-objects-in-the-formset`.
+.. warning::
+
+ All hooks that return a ``ModelAdmin`` property return the property itself
+ rather than a copy of its value. Dynamically modifying the value can lead
+ to surprising results.
+
+ Let's take :meth:`ModelAdmin.get_readonly_fields` as an example::
+
+ class PersonAdmin(admin.ModelAdmin):
+ readonly_fields = ["name"]
+
+ def get_readonly_fields(self, request, obj=None):
+ readonly = super().get_readonly_fields(request, obj)
+ if not request.user.is_superuser:
+ readonly.append("age") # Edits the class attribute.
+ return readonly
+
+ This results in ``readonly_fields`` becoming
+ ``["name", "age", "age", ...]``, even for a superuser, as ``"age"`` is added
+ each time non-superuser visits the page.
+
.. method:: ModelAdmin.get_ordering(request)
The ``get_ordering`` method takes a ``request`` as parameter and
diff --git a/docs/ref/contrib/auth.txt b/docs/ref/contrib/auth.txt
index c8699a2913..103aff8e0b 100644
--- a/docs/ref/contrib/auth.txt
+++ b/docs/ref/contrib/auth.txt
@@ -54,7 +54,8 @@ Fields
Required. A hash of, and metadata about, the password. (Django doesn't
store the raw password.) Raw passwords can be arbitrarily long and can
- contain any character. See the :doc:`password documentation
+ contain any character. The metadata in this field may mark the password
+ as unusable. See the :doc:`password documentation
`.
.. attribute:: groups
@@ -175,8 +176,9 @@ Methods
.. method:: set_unusable_password()
- Marks the user as having no password set. This isn't the same as
- having a blank string for a password.
+ Marks the user as having no password set by updating the metadata in
+ the :attr:`~django.contrib.auth.models.User.password` field. This isn't
+ the same as having a blank string for a password.
:meth:`~django.contrib.auth.models.User.check_password()` for this user
will never return ``True``. Doesn't save the
:class:`~django.contrib.auth.models.User` object.
diff --git a/docs/ref/contrib/flatpages.txt b/docs/ref/contrib/flatpages.txt
index c82fb5de85..01e5553ff3 100644
--- a/docs/ref/contrib/flatpages.txt
+++ b/docs/ref/contrib/flatpages.txt
@@ -256,7 +256,7 @@ Here's a sample :file:`flatpages/default.html` template:
.. code-block:: html+django
-
+
{{ flatpage.title }}
diff --git a/docs/ref/contrib/gis/db-api.txt b/docs/ref/contrib/gis/db-api.txt
index bce6f2efcc..e33d9a514f 100644
--- a/docs/ref/contrib/gis/db-api.txt
+++ b/docs/ref/contrib/gis/db-api.txt
@@ -339,42 +339,42 @@ divided into the three categories described in the :ref:`raster lookup details
`: native support ``N``, bilateral native support ``B``,
and geometry conversion support ``C``.
-================================= ========= ======== ========= ============ ========== ========
-Lookup Type PostGIS Oracle MariaDB MySQL [#]_ SpatiaLite PGRaster
-================================= ========= ======== ========= ============ ========== ========
-:lookup:`bbcontains` X X X X N
-:lookup:`bboverlaps` X X X X N
-:lookup:`contained` X X X X N
-:lookup:`contains ` X X X X X B
-:lookup:`contains_properly` X B
-:lookup:`coveredby` X X X B
-:lookup:`covers` X X X B
-:lookup:`crosses` X X X X C
-:lookup:`disjoint` X X X X X B
-:lookup:`distance_gt` X X X X X N
-:lookup:`distance_gte` X X X X X N
-:lookup:`distance_lt` X X X X X N
-:lookup:`distance_lte` X X X X X N
-:lookup:`dwithin` X X X B
-:lookup:`equals` X X X X X C
-:lookup:`exact ` X X X X X B
-:lookup:`intersects` X X X X X B
+================================= ========= ======== ========== ============ ========== ========
+Lookup Type PostGIS Oracle MariaDB MySQL [#]_ SpatiaLite PGRaster
+================================= ========= ======== ========== ============ ========== ========
+:lookup:`bbcontains` X X X X N
+:lookup:`bboverlaps` X X X X N
+:lookup:`contained` X X X X N
+:lookup:`contains ` X X X X X B
+:lookup:`contains_properly` X B
+:lookup:`coveredby` X X X (≥ 11.7) X X B
+:lookup:`covers` X X X X B
+:lookup:`crosses` X X X X C
+:lookup:`disjoint` X X X X X B
+:lookup:`distance_gt` X X X X X N
+:lookup:`distance_gte` X X X X X N
+:lookup:`distance_lt` X X X X X N
+:lookup:`distance_lte` X X X X X N
+:lookup:`dwithin` X X X B
+:lookup:`equals` X X X X X C
+:lookup:`exact ` X X X X X B
+:lookup:`intersects` X X X X X B
:lookup:`isempty` X
-:lookup:`isvalid` X X X X
-:lookup:`overlaps` X X X X X B
-:lookup:`relate` X X X X C
-:lookup:`same_as` X X X X X B
-:lookup:`touches` X X X X X B
-:lookup:`within` X X X X X B
-:lookup:`left` X C
-:lookup:`right` X C
-:lookup:`overlaps_left` X B
-:lookup:`overlaps_right` X B
-:lookup:`overlaps_above` X C
-:lookup:`overlaps_below` X C
-:lookup:`strictly_above` X C
-:lookup:`strictly_below` X C
-================================= ========= ======== ========= ============ ========== ========
+:lookup:`isvalid` X X X (≥ 11.7) X X
+:lookup:`overlaps` X X X X X B
+:lookup:`relate` X X X X C
+:lookup:`same_as` X X X X X B
+:lookup:`touches` X X X X X B
+:lookup:`within` X X X X X B
+:lookup:`left` X C
+:lookup:`right` X C
+:lookup:`overlaps_left` X B
+:lookup:`overlaps_right` X B
+:lookup:`overlaps_above` X C
+:lookup:`overlaps_below` X C
+:lookup:`strictly_above` X C
+:lookup:`strictly_below` X C
+================================= ========= ======== ========== ============ ========== ========
.. _database-functions-compatibility:
@@ -406,10 +406,10 @@ Function PostGIS Oracle MariaDB MySQL
:class:`ForcePolygonCW` X X
:class:`FromWKB` X X X X X
:class:`FromWKT` X X X X X
-:class:`GeoHash` X X X (LWGEOM/RTTOPO)
+:class:`GeoHash` X X (≥ 11.7) X X (LWGEOM/RTTOPO)
:class:`Intersection` X X X X X
:class:`IsEmpty` X
-:class:`IsValid` X X X X
+:class:`IsValid` X X X (≥ 11.7) X X
:class:`Length` X X X X X
:class:`LineLocatePoint` X X
:class:`MakeValid` X X (LWGEOM/RTTOPO)
@@ -431,20 +431,19 @@ Aggregate Functions
-------------------
The following table provides a summary of what GIS-specific aggregate functions
-are available on each spatial backend. Please note that MariaDB does not
-support any of these aggregates, and is thus excluded from the table.
+are available on each spatial backend.
.. currentmodule:: django.contrib.gis.db.models
-======================= ======= ====== ============ ==========
-Aggregate PostGIS Oracle MySQL SpatiaLite
-======================= ======= ====== ============ ==========
-:class:`Collect` X X (≥ 8.0.24) X
-:class:`Extent` X X X
+======================= ======= ====== ========== ============ ==========
+Aggregate PostGIS Oracle MariaDB MySQL SpatiaLite
+======================= ======= ====== ========== ============ ==========
+:class:`Collect` X X (≥ 11.7) X (≥ 8.0.24) X
+:class:`Extent` X X X
:class:`Extent3D` X
-:class:`MakeLine` X X
-:class:`Union` X X X
-======================= ======= ====== ============ ==========
+:class:`MakeLine` X X
+:class:`Union` X X X
+======================= ======= ====== ========== ============ ==========
.. rubric:: Footnotes
.. [#fnwkt] *See* Open Geospatial Consortium, Inc., `OpenGIS Simple Feature Specification For SQL `_, Document 99-049 (May 5, 1999), at Ch. 3.2.5, p. 3-11 (SQL Textual Representation of Geometry).
diff --git a/docs/ref/contrib/gis/functions.txt b/docs/ref/contrib/gis/functions.txt
index ff05d0ec96..ff62c17580 100644
--- a/docs/ref/contrib/gis/functions.txt
+++ b/docs/ref/contrib/gis/functions.txt
@@ -393,7 +393,7 @@ Creates geometry from `Well-known text (WKT)`_ representation. The optional
.. class:: GeoHash(expression, precision=None, **extra)
-*Availability*: `MySQL
+*Availability*: MariaDB, `MySQL
`__,
`PostGIS `__, SpatiaLite
(LWGEOM/RTTOPO)
@@ -406,6 +406,10 @@ result.
__ https://en.wikipedia.org/wiki/Geohash
+.. versionchanged:: 5.2
+
+ MariaDB 11.7+ support was added.
+
``GeometryDistance``
====================
@@ -444,13 +448,17 @@ geometry. Returns ``True`` if its value is empty and ``False`` otherwise.
.. class:: IsValid(expr)
-*Availability*: `MySQL
+*Availability*: MariaDB, `MySQL
`__,
`PostGIS `__, Oracle, SpatiaLite
Accepts a geographic field or expression and tests if the value is well formed.
Returns ``True`` if its value is a valid geometry and ``False`` otherwise.
+.. versionchanged:: 5.2
+
+ MariaDB 11.7+ support was added.
+
``Length``
==========
diff --git a/docs/ref/contrib/gis/gdal.txt b/docs/ref/contrib/gis/gdal.txt
index c2a333f895..726cd83756 100644
--- a/docs/ref/contrib/gis/gdal.txt
+++ b/docs/ref/contrib/gis/gdal.txt
@@ -611,6 +611,26 @@ coordinate transformation:
>>> polygon.geom_count
1
+ .. attribute:: has_curve
+
+ .. versionadded:: 5.2
+
+ A boolean indicating if this geometry is or contains a curve geometry.
+
+ .. method:: get_linear_geometry
+
+ .. versionadded:: 5.2
+
+ Returns a linear version of the geometry. If no conversion can be made, the
+ original geometry is returned.
+
+ .. method:: get_curve_geometry
+
+ .. versionadded:: 5.2
+
+ Returns a curved version of the geometry. If no conversion can be made, the
+ original geometry is returned.
+
.. attribute:: point_count
Returns the number of points used to describe this geometry:
diff --git a/docs/ref/contrib/gis/geoquerysets.txt b/docs/ref/contrib/gis/geoquerysets.txt
index b639c5271e..19411b7304 100644
--- a/docs/ref/contrib/gis/geoquerysets.txt
+++ b/docs/ref/contrib/gis/geoquerysets.txt
@@ -183,7 +183,7 @@ PostGIS ``ST_ContainsProperly(poly, geom)``
-------------
*Availability*: `PostGIS `__,
-Oracle, PGRaster (Bilateral), SpatiaLite
+Oracle, MariaDB 11.7+, MySQL, PGRaster (Bilateral), SpatiaLite
Tests if no point in the geometry field is outside the lookup geometry.
[#fncovers]_
@@ -197,16 +197,22 @@ Backend SQL Equivalent
========== =============================
PostGIS ``ST_CoveredBy(poly, geom)``
Oracle ``SDO_COVEREDBY(poly, geom)``
+MariaDB ``MBRCoveredBy(poly, geom)``
+MySQL ``MBRCoveredBy(poly, geom)``
SpatiaLite ``CoveredBy(poly, geom)``
========== =============================
+.. versionchanged:: 5.2
+
+ MySQL and MariaDB 11.7+ support was added.
+
.. fieldlookup:: covers
``covers``
----------
*Availability*: `PostGIS `__,
-Oracle, PGRaster (Bilateral), SpatiaLite
+Oracle, MySQL, PGRaster (Bilateral), SpatiaLite
Tests if no point in the lookup geometry is outside the geometry field.
[#fncovers]_
@@ -220,9 +226,14 @@ Backend SQL Equivalent
========== ==========================
PostGIS ``ST_Covers(poly, geom)``
Oracle ``SDO_COVERS(poly, geom)``
+MySQL ``MBRCovers(poly, geom)``
SpatiaLite ``Covers(poly, geom)``
========== ==========================
+.. versionchanged:: 5.2
+
+ MySQL support was added.
+
.. fieldlookup:: crosses
``crosses``
@@ -364,8 +375,8 @@ Example::
``isvalid``
-----------
-*Availability*: MySQL, `PostGIS `__,
-Oracle, SpatiaLite
+*Availability*: MariaDB, MySQL,
+`PostGIS `__, Oracle, SpatiaLite
Tests if the geometry is valid.
@@ -373,12 +384,16 @@ Example::
Zipcode.objects.filter(poly__isvalid=True)
-========================== ================================================================
-Backend SQL Equivalent
-========================== ================================================================
-MySQL, PostGIS, SpatiaLite ``ST_IsValid(poly)``
-Oracle ``SDO_GEOM.VALIDATE_GEOMETRY_WITH_CONTEXT(poly, 0.05) = 'TRUE'``
-========================== ================================================================
+=================================== ================================================================
+Backend SQL Equivalent
+=================================== ================================================================
+MariaDB, MySQL, PostGIS, SpatiaLite ``ST_IsValid(poly)``
+Oracle ``SDO_GEOM.VALIDATE_GEOMETRY_WITH_CONTEXT(poly, 0.05) = 'TRUE'``
+=================================== ================================================================
+
+.. versionchanged:: 5.2
+
+ MariaDB 11.7+ support was added.
.. fieldlookup:: overlaps
@@ -870,8 +885,8 @@ Example:
.. class:: Collect(geo_field, filter=None)
-*Availability*: `PostGIS `__, MySQL,
-SpatiaLite
+*Availability*: `PostGIS `__,
+MariaDB, MySQL, SpatiaLite
Returns a ``GEOMETRYCOLLECTION`` or a ``MULTI`` geometry object from the geometry
column. This is analogous to a simplified version of the :class:`Union`
@@ -883,6 +898,10 @@ caring about dissolving boundaries.
MySQL 8.0.24+ support was added.
+.. versionchanged:: 5.2
+
+ MariaDB 11.7+ support was added.
+
``Extent``
~~~~~~~~~~
diff --git a/docs/ref/files/storage.txt b/docs/ref/files/storage.txt
index f7c290a150..52c8f90427 100644
--- a/docs/ref/files/storage.txt
+++ b/docs/ref/files/storage.txt
@@ -11,7 +11,25 @@ Django provides convenient ways to access the default storage class:
.. data:: storages
- Storage instances as defined by :setting:`STORAGES`.
+ A dictionary-like object that allows retrieving a storage instance using
+ its alias as defined by :setting:`STORAGES`.
+
+ ``storages`` has an attribute ``backends``, which defaults to the raw value
+ provided in :setting:`STORAGES`.
+
+ Additionally, ``storages`` provides a ``create_storage()`` method that
+ accepts the dictionary used in :setting:`STORAGES` for a backend, and
+ returns a storage instance based on that backend definition. This may be
+ useful for third-party packages needing to instantiate storages in tests:
+
+ .. code-block:: pycon
+
+ >>> from django.core.files.storage import storages
+ >>> storages.backends
+ {'default': {'BACKEND': 'django.core.files.storage.FileSystemStorage'},
+ 'staticfiles': {'BACKEND': 'django.contrib.staticfiles.storage.StaticFilesStorage'},
+ 'custom': {'BACKEND': 'package.storage.CustomStorage'}}
+ >>> storage_instance = storages.create_storage({"BACKEND": "package.storage.CustomStorage"})
.. class:: DefaultStorage
diff --git a/docs/ref/forms/api.txt b/docs/ref/forms/api.txt
index 33d0806859..9ce16ff2ab 100644
--- a/docs/ref/forms/api.txt
+++ b/docs/ref/forms/api.txt
@@ -406,8 +406,8 @@ process:
.. code-block:: pycon
>>> f.base_fields["subject"].label_suffix = "?"
- >>> another_f = CommentForm(auto_id=False)
- >>> f.as_div().split(" ")[0]
+ >>> another_f = ContactForm(auto_id=False)
+ >>> another_f.as_div().split("