Refs #30897 -- Added support for TREE format to Queryset.explain() on MySQL 8.0.16+.
This commit is contained in:
parent
37f02c47f8
commit
742961332e
|
@ -48,8 +48,6 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
END;
|
END;
|
||||||
"""
|
"""
|
||||||
db_functions_convert_bytes_to_str = True
|
db_functions_convert_bytes_to_str = True
|
||||||
# Alias MySQL's TRADITIONAL to TEXT for consistency with other backends.
|
|
||||||
supported_explain_formats = {'JSON', 'TEXT', 'TRADITIONAL'}
|
|
||||||
# Neither MySQL nor MariaDB support partial indexes.
|
# Neither MySQL nor MariaDB support partial indexes.
|
||||||
supports_partial_indexes = False
|
supports_partial_indexes = False
|
||||||
|
|
||||||
|
@ -119,6 +117,15 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
||||||
# EXTENDED is deprecated (and not required) in MySQL 5.7.
|
# EXTENDED is deprecated (and not required) in MySQL 5.7.
|
||||||
return not self.connection.mysql_is_mariadb and self.connection.mysql_version < (5, 7)
|
return not self.connection.mysql_is_mariadb and self.connection.mysql_version < (5, 7)
|
||||||
|
|
||||||
|
@cached_property
|
||||||
|
def supported_explain_formats(self):
|
||||||
|
# Alias MySQL's TRADITIONAL to TEXT for consistency with other
|
||||||
|
# backends.
|
||||||
|
formats = {'JSON', 'TEXT', 'TRADITIONAL'}
|
||||||
|
if not self.connection.mysql_is_mariadb and self.connection.mysql_version >= (8, 0, 16):
|
||||||
|
formats.add('TREE')
|
||||||
|
return formats
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def supports_transactions(self):
|
def supports_transactions(self):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -296,6 +296,9 @@ class DatabaseOperations(BaseDatabaseOperations):
|
||||||
# Alias MySQL's TRADITIONAL to TEXT for consistency with other backends.
|
# Alias MySQL's TRADITIONAL to TEXT for consistency with other backends.
|
||||||
if format and format.upper() == 'TEXT':
|
if format and format.upper() == 'TEXT':
|
||||||
format = 'TRADITIONAL'
|
format = 'TRADITIONAL'
|
||||||
|
elif not format and 'TREE' in self.connection.features.supported_explain_formats:
|
||||||
|
# Use TREE by default (if supported) as it's more informative.
|
||||||
|
format = 'TREE'
|
||||||
prefix = super().explain_query_prefix(format, **options)
|
prefix = super().explain_query_prefix(format, **options)
|
||||||
if format:
|
if format:
|
||||||
prefix += ' FORMAT=%s' % format
|
prefix += ' FORMAT=%s' % format
|
||||||
|
|
|
@ -2571,8 +2571,10 @@ because an implementation there isn't straightforward.
|
||||||
|
|
||||||
The ``format`` parameter changes the output format from the databases's default,
|
The ``format`` parameter changes the output format from the databases's default,
|
||||||
usually text-based. PostgreSQL supports ``'TEXT'``, ``'JSON'``, ``'YAML'``, and
|
usually text-based. PostgreSQL supports ``'TEXT'``, ``'JSON'``, ``'YAML'``, and
|
||||||
``'XML'``. MySQL supports ``'TEXT'`` (also called ``'TRADITIONAL'``) and
|
``'XML'`` formats. MariaDB and MySQL support ``'TEXT'`` (also called
|
||||||
``'JSON'``.
|
``'TRADITIONAL'``) and ``'JSON'`` formats. MySQL 8.0.16+ also supports an
|
||||||
|
improved ``'TREE'`` format, which is similar to PostgreSQL's ``'TEXT'`` output
|
||||||
|
and is used by default, if supported.
|
||||||
|
|
||||||
Some databases accept flags that can return more information about the query.
|
Some databases accept flags that can return more information about the query.
|
||||||
Pass these flags as keyword arguments. For example, when using PostgreSQL::
|
Pass these flags as keyword arguments. For example, when using PostgreSQL::
|
||||||
|
@ -2589,6 +2591,10 @@ adverse effects on your database. For example, PostgreSQL's ``ANALYZE`` flag
|
||||||
could result in changes to data if there are triggers or if a function is
|
could result in changes to data if there are triggers or if a function is
|
||||||
called, even for a ``SELECT`` query.
|
called, even for a ``SELECT`` query.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
|
Support for the ``'TREE'`` format on MySQL 8.0.16+ was added.
|
||||||
|
|
||||||
.. _field-lookups:
|
.. _field-lookups:
|
||||||
|
|
||||||
``Field`` lookups
|
``Field`` lookups
|
||||||
|
|
|
@ -169,6 +169,8 @@ Models
|
||||||
:class:`~django.db.models.DateTimeField`, and the new :lookup:`iso_week_day`
|
:class:`~django.db.models.DateTimeField`, and the new :lookup:`iso_week_day`
|
||||||
lookup allows querying by an ISO-8601 day of week.
|
lookup allows querying by an ISO-8601 day of week.
|
||||||
|
|
||||||
|
* :meth:`.QuerySet.explain` now supports ``TREE`` format on MySQL 8.0.16+.
|
||||||
|
|
||||||
Pagination
|
Pagination
|
||||||
~~~~~~~~~~
|
~~~~~~~~~~
|
||||||
|
|
||||||
|
|
|
@ -71,9 +71,10 @@ class ExplainTests(TestCase):
|
||||||
|
|
||||||
@unittest.skipUnless(connection.vendor == 'mysql', 'MySQL specific')
|
@unittest.skipUnless(connection.vendor == 'mysql', 'MySQL specific')
|
||||||
def test_mysql_text_to_traditional(self):
|
def test_mysql_text_to_traditional(self):
|
||||||
# Initialize the cached property, if needed, to prevent a query for
|
# Ensure these cached properties are initialized to prevent queries for
|
||||||
# the MySQL version during the QuerySet evaluation.
|
# the MariaDB or MySQL version during the QuerySet evaluation.
|
||||||
connection.features.needs_explain_extended
|
connection.features.needs_explain_extended
|
||||||
|
connection.features.supported_explain_formats
|
||||||
with CaptureQueriesContext(connection) as captured_queries:
|
with CaptureQueriesContext(connection) as captured_queries:
|
||||||
Tag.objects.filter(name='test').explain(format='text')
|
Tag.objects.filter(name='test').explain(format='text')
|
||||||
self.assertEqual(len(captured_queries), 1)
|
self.assertEqual(len(captured_queries), 1)
|
||||||
|
|
Loading…
Reference in New Issue