mirror of
https://github.com/django/django.git
synced 2025-10-24 14:16:09 +00:00
Fixed #34761 -- Dropped support for MySQL < 8.0.11.
This commit is contained in:
@@ -11,6 +11,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
# MySQL doesn't support sliced subqueries with IN/ALL/ANY/SOME.
|
# MySQL doesn't support sliced subqueries with IN/ALL/ANY/SOME.
|
||||||
allow_sliced_subqueries_with_in = False
|
allow_sliced_subqueries_with_in = False
|
||||||
has_select_for_update = True
|
has_select_for_update = True
|
||||||
|
has_select_for_update_nowait = True
|
||||||
supports_forward_references = False
|
supports_forward_references = False
|
||||||
supports_regex_backreferencing = False
|
supports_regex_backreferencing = False
|
||||||
supports_date_lookup_using_string = False
|
supports_date_lookup_using_string = False
|
||||||
@@ -23,6 +24,8 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
supports_temporal_subtraction = True
|
supports_temporal_subtraction = True
|
||||||
supports_slicing_ordering_in_compound = True
|
supports_slicing_ordering_in_compound = True
|
||||||
supports_index_on_text_field = False
|
supports_index_on_text_field = False
|
||||||
|
supports_over_clause = True
|
||||||
|
supports_frame_range_fixed_distance = True
|
||||||
supports_update_conflicts = True
|
supports_update_conflicts = True
|
||||||
delete_can_self_reference_subquery = False
|
delete_can_self_reference_subquery = False
|
||||||
create_test_procedure_without_params_sql = """
|
create_test_procedure_without_params_sql = """
|
||||||
@@ -62,7 +65,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
if self.connection.mysql_is_mariadb:
|
if self.connection.mysql_is_mariadb:
|
||||||
return (10, 4)
|
return (10, 4)
|
||||||
else:
|
else:
|
||||||
return (8,)
|
return (8, 0, 11)
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def test_collations(self):
|
def test_collations(self):
|
||||||
@@ -225,16 +228,6 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
def is_sql_auto_is_null_enabled(self):
|
def is_sql_auto_is_null_enabled(self):
|
||||||
return self.connection.mysql_server_data["sql_auto_is_null"]
|
return self.connection.mysql_server_data["sql_auto_is_null"]
|
||||||
|
|
||||||
@cached_property
|
|
||||||
def supports_over_clause(self):
|
|
||||||
if self.connection.mysql_is_mariadb:
|
|
||||||
return True
|
|
||||||
return self.connection.mysql_version >= (8, 0, 2)
|
|
||||||
|
|
||||||
supports_frame_range_fixed_distance = property(
|
|
||||||
operator.attrgetter("supports_over_clause")
|
|
||||||
)
|
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def supports_column_check_constraints(self):
|
def supports_column_check_constraints(self):
|
||||||
if self.connection.mysql_is_mariadb:
|
if self.connection.mysql_is_mariadb:
|
||||||
@@ -255,20 +248,11 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
def has_select_for_update_skip_locked(self):
|
def has_select_for_update_skip_locked(self):
|
||||||
if self.connection.mysql_is_mariadb:
|
if self.connection.mysql_is_mariadb:
|
||||||
return self.connection.mysql_version >= (10, 6)
|
return self.connection.mysql_version >= (10, 6)
|
||||||
return self.connection.mysql_version >= (8, 0, 1)
|
return True
|
||||||
|
|
||||||
@cached_property
|
|
||||||
def has_select_for_update_nowait(self):
|
|
||||||
if self.connection.mysql_is_mariadb:
|
|
||||||
return True
|
|
||||||
return self.connection.mysql_version >= (8, 0, 1)
|
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def has_select_for_update_of(self):
|
def has_select_for_update_of(self):
|
||||||
return (
|
return not self.connection.mysql_is_mariadb
|
||||||
not self.connection.mysql_is_mariadb
|
|
||||||
and self.connection.mysql_version >= (8, 0, 1)
|
|
||||||
)
|
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def supports_explain_analyze(self):
|
def supports_explain_analyze(self):
|
||||||
@@ -319,7 +303,7 @@ class DatabaseFeatures(BaseDatabaseFeatures):
|
|||||||
return False
|
return False
|
||||||
if self.connection.mysql_is_mariadb:
|
if self.connection.mysql_is_mariadb:
|
||||||
return self.connection.mysql_version >= (10, 8)
|
return self.connection.mysql_version >= (10, 8)
|
||||||
return self.connection.mysql_version >= (8, 0, 1)
|
return True
|
||||||
|
|
||||||
@cached_property
|
@cached_property
|
||||||
def supports_expression_indexes(self):
|
def supports_expression_indexes(self):
|
||||||
|
@@ -45,14 +45,12 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def sql_rename_column(self):
|
def sql_rename_column(self):
|
||||||
# MariaDB >= 10.5.2 and MySQL >= 8.0.4 support an
|
is_mariadb = self.connection.mysql_is_mariadb
|
||||||
# "ALTER TABLE ... RENAME COLUMN" statement.
|
if is_mariadb and self.connection.mysql_version < (10, 5, 2):
|
||||||
if self.connection.mysql_is_mariadb:
|
# MariaDB < 10.5.2 doesn't support an
|
||||||
if self.connection.mysql_version >= (10, 5, 2):
|
# "ALTER TABLE ... RENAME COLUMN" statement.
|
||||||
return super().sql_rename_column
|
return "ALTER TABLE %(table)s CHANGE %(old_column)s %(new_column)s %(type)s"
|
||||||
elif self.connection.mysql_version >= (8, 0, 4):
|
return super().sql_rename_column
|
||||||
return super().sql_rename_column
|
|
||||||
return "ALTER TABLE %(table)s CHANGE %(old_column)s %(new_column)s %(type)s"
|
|
||||||
|
|
||||||
def quote_value(self, value):
|
def quote_value(self, value):
|
||||||
self.connection.ensure_connection()
|
self.connection.ensure_connection()
|
||||||
|
@@ -57,7 +57,7 @@ supported versions, and any notes for each of the supported database backends:
|
|||||||
Database Library Requirements Supported Versions Notes
|
Database Library Requirements Supported Versions Notes
|
||||||
================== ============================== ================== =========================================
|
================== ============================== ================== =========================================
|
||||||
PostgreSQL GEOS, GDAL, PROJ, PostGIS 12+ Requires PostGIS.
|
PostgreSQL GEOS, GDAL, PROJ, PostGIS 12+ Requires PostGIS.
|
||||||
MySQL GEOS, GDAL 8+ :ref:`Limited functionality <mysql-spatial-limitations>`.
|
MySQL GEOS, GDAL 8.0.11+ :ref:`Limited functionality <mysql-spatial-limitations>`.
|
||||||
Oracle GEOS, GDAL 19+ XE not supported.
|
Oracle GEOS, GDAL 19+ XE not supported.
|
||||||
SQLite GEOS, GDAL, PROJ, SpatiaLite 3.21.0+ Requires SpatiaLite 4.3+
|
SQLite GEOS, GDAL, PROJ, SpatiaLite 3.21.0+ Requires SpatiaLite 4.3+
|
||||||
================== ============================== ================== =========================================
|
================== ============================== ================== =========================================
|
||||||
|
@@ -412,7 +412,7 @@ MySQL notes
|
|||||||
Version support
|
Version support
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
Django supports MySQL 8 and higher.
|
Django supports MySQL 8.0.11 and higher.
|
||||||
|
|
||||||
Django's ``inspectdb`` feature uses the ``information_schema`` database, which
|
Django's ``inspectdb`` feature uses the ``information_schema`` database, which
|
||||||
contains detailed data on all database schemas.
|
contains detailed data on all database schemas.
|
||||||
@@ -755,14 +755,14 @@ MySQL and MariaDB do not support some options to the ``SELECT ... FOR UPDATE``
|
|||||||
statement. If ``select_for_update()`` is used with an unsupported option, then
|
statement. If ``select_for_update()`` is used with an unsupported option, then
|
||||||
a :exc:`~django.db.NotSupportedError` is raised.
|
a :exc:`~django.db.NotSupportedError` is raised.
|
||||||
|
|
||||||
=============== ========= ==========
|
=============== ========= =====
|
||||||
Option MariaDB MySQL
|
Option MariaDB MySQL
|
||||||
=============== ========= ==========
|
=============== ========= =====
|
||||||
``SKIP LOCKED`` X (≥10.6) X (≥8.0.1)
|
``SKIP LOCKED`` X (≥10.6) X
|
||||||
``NOWAIT`` X X (≥8.0.1)
|
``NOWAIT`` X X
|
||||||
``OF`` X (≥8.0.1)
|
``OF`` X
|
||||||
``NO KEY``
|
``NO KEY``
|
||||||
=============== ========= ==========
|
=============== ========= =====
|
||||||
|
|
||||||
When using ``select_for_update()`` on MySQL, make sure you filter a queryset
|
When using ``select_for_update()`` on MySQL, make sure you filter a queryset
|
||||||
against at least a set of fields contained in unique constraints or only
|
against at least a set of fields contained in unique constraints or only
|
||||||
|
@@ -880,7 +880,7 @@ from groups to be included:
|
|||||||
|
|
||||||
Support for filtering against window functions was added.
|
Support for filtering against window functions was added.
|
||||||
|
|
||||||
Among Django's built-in database backends, MySQL 8.0.2+, PostgreSQL, and Oracle
|
Among Django's built-in database backends, MySQL, PostgreSQL, and Oracle
|
||||||
support window expressions. Support for different window expression features
|
support window expressions. Support for different window expression features
|
||||||
varies among the different databases. For example, the options in
|
varies among the different databases. For example, the options in
|
||||||
:meth:`~django.db.models.Expression.asc` and
|
:meth:`~django.db.models.Expression.asc` and
|
||||||
|
@@ -82,10 +82,10 @@ field's name.
|
|||||||
For example ``Index(fields=['headline', '-pub_date'])`` would create SQL with
|
For example ``Index(fields=['headline', '-pub_date'])`` would create SQL with
|
||||||
``(headline, pub_date DESC)``.
|
``(headline, pub_date DESC)``.
|
||||||
|
|
||||||
.. admonition:: MySQL and MariaDB
|
.. admonition:: MariaDB
|
||||||
|
|
||||||
Index ordering isn't supported on MySQL < 8.0.1 and MariaDB < 10.8. In that
|
Index ordering isn't supported on MariaDB < 10.8. In that case, a
|
||||||
case, a descending index is created as a normal index.
|
descending index is created as a normal index.
|
||||||
|
|
||||||
``name``
|
``name``
|
||||||
--------
|
--------
|
||||||
|
@@ -1928,8 +1928,8 @@ them:
|
|||||||
The ``postgresql``, ``oracle``, and ``mysql`` database backends support
|
The ``postgresql``, ``oracle``, and ``mysql`` database backends support
|
||||||
``select_for_update()``. However, MariaDB only supports the ``nowait``
|
``select_for_update()``. However, MariaDB only supports the ``nowait``
|
||||||
argument, MariaDB 10.6+ also supports the ``skip_locked`` argument, and MySQL
|
argument, MariaDB 10.6+ also supports the ``skip_locked`` argument, and MySQL
|
||||||
8.0.1+ supports the ``nowait``, ``skip_locked``, and ``of`` arguments. The
|
supports the ``nowait``, ``skip_locked``, and ``of`` arguments. The ``no_key``
|
||||||
``no_key`` argument is only supported on PostgreSQL.
|
argument is only supported on PostgreSQL.
|
||||||
|
|
||||||
Passing ``nowait=True``, ``skip_locked=True``, ``no_key=True``, or ``of`` to
|
Passing ``nowait=True``, ``skip_locked=True``, ``no_key=True``, or ``of`` to
|
||||||
``select_for_update()`` using database backends that do not support these
|
``select_for_update()`` using database backends that do not support these
|
||||||
|
@@ -463,6 +463,12 @@ backends.
|
|||||||
``False`` if the database doesn't support the ``DEFAULT`` keyword in bulk
|
``False`` if the database doesn't support the ``DEFAULT`` keyword in bulk
|
||||||
``INSERT`` queries.
|
``INSERT`` queries.
|
||||||
|
|
||||||
|
Dropped support for MySQL < 8.0.11
|
||||||
|
----------------------------------
|
||||||
|
|
||||||
|
Support for pre-releases of MySQL 8.0.x series is removed. Django 5.0 supports
|
||||||
|
MySQL 8.0.11 and higher.
|
||||||
|
|
||||||
:mod:`django.contrib.gis`
|
:mod:`django.contrib.gis`
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
@@ -23,20 +23,6 @@ class TestFeatures(TestCase):
|
|||||||
self.assertFalse(connection.features.supports_transactions)
|
self.assertFalse(connection.features.supports_transactions)
|
||||||
del connection.features.supports_transactions
|
del connection.features.supports_transactions
|
||||||
|
|
||||||
def test_skip_locked_no_wait(self):
|
|
||||||
with mock.MagicMock() as _connection:
|
|
||||||
_connection.mysql_version = (8, 0, 1)
|
|
||||||
_connection.mysql_is_mariadb = False
|
|
||||||
database_features = DatabaseFeatures(_connection)
|
|
||||||
self.assertTrue(database_features.has_select_for_update_skip_locked)
|
|
||||||
self.assertTrue(database_features.has_select_for_update_nowait)
|
|
||||||
with mock.MagicMock() as _connection:
|
|
||||||
_connection.mysql_version = (8, 0, 0)
|
|
||||||
_connection.mysql_is_mariadb = False
|
|
||||||
database_features = DatabaseFeatures(_connection)
|
|
||||||
self.assertFalse(database_features.has_select_for_update_skip_locked)
|
|
||||||
self.assertFalse(database_features.has_select_for_update_nowait)
|
|
||||||
|
|
||||||
def test_allows_auto_pk_0(self):
|
def test_allows_auto_pk_0(self):
|
||||||
with mock.MagicMock() as _connection:
|
with mock.MagicMock() as _connection:
|
||||||
_connection.sql_mode = {"NO_AUTO_VALUE_ON_ZERO"}
|
_connection.sql_mode = {"NO_AUTO_VALUE_ON_ZERO"}
|
||||||
|
@@ -109,8 +109,8 @@ class Tests(TestCase):
|
|||||||
mocked_get_database_version.return_value = (10, 3)
|
mocked_get_database_version.return_value = (10, 3)
|
||||||
msg = "MariaDB 10.4 or later is required (found 10.3)."
|
msg = "MariaDB 10.4 or later is required (found 10.3)."
|
||||||
else:
|
else:
|
||||||
mocked_get_database_version.return_value = (5, 7)
|
mocked_get_database_version.return_value = (8, 0, 4)
|
||||||
msg = "MySQL 8 or later is required (found 5.7)."
|
msg = "MySQL 8.0.11 or later is required (found 8.0.4)."
|
||||||
|
|
||||||
with self.assertRaisesMessage(NotSupportedError, msg):
|
with self.assertRaisesMessage(NotSupportedError, msg):
|
||||||
connection.check_database_version_supported()
|
connection.check_database_version_supported()
|
||||||
|
Reference in New Issue
Block a user