From 0fcee1676c7f14bb08e2cc662898dee56d9cf207 Mon Sep 17 00:00:00 2001 From: Simon Charette Date: Sat, 13 Jan 2024 16:16:58 -0500 Subject: [PATCH] Fixed #35111 -- Fixed compilation of DateField __in/__range rhs on SQLite and MySQL. Also removed tests that ensured that adapt_(date)timefield backend operations where able to deal with expressions when it's not the case for any other adapt methods. --- django/db/backends/base/operations.py | 8 -------- django/db/backends/oracle/operations.py | 8 -------- django/db/backends/sqlite3/operations.py | 8 -------- django/db/models/lookups.py | 11 ++++++++--- tests/backends/base/test_operations.py | 10 +--------- tests/expressions/tests.py | 18 ++++++++++++++---- 6 files changed, 23 insertions(+), 40 deletions(-) diff --git a/django/db/backends/base/operations.py b/django/db/backends/base/operations.py index d5a40eb46e..9f40ec5e4f 100644 --- a/django/db/backends/base/operations.py +++ b/django/db/backends/base/operations.py @@ -562,10 +562,6 @@ class BaseDatabaseOperations: """ if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - return str(value) def adapt_timefield_value(self, value): @@ -575,10 +571,6 @@ class BaseDatabaseOperations: """ if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - if timezone.is_aware(value): raise ValueError("Django does not support timezone-aware times.") return str(value) diff --git a/django/db/backends/oracle/operations.py b/django/db/backends/oracle/operations.py index 9e8172b80a..aedfeea236 100644 --- a/django/db/backends/oracle/operations.py +++ b/django/db/backends/oracle/operations.py @@ -590,10 +590,6 @@ END; if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - # oracledb doesn't support tz-aware datetimes if timezone.is_aware(value): if settings.USE_TZ: @@ -610,10 +606,6 @@ END; if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - if isinstance(value, str): return datetime.datetime.strptime(value, "%H:%M:%S") diff --git a/django/db/backends/sqlite3/operations.py b/django/db/backends/sqlite3/operations.py index dfc9857b84..29a5c0391e 100644 --- a/django/db/backends/sqlite3/operations.py +++ b/django/db/backends/sqlite3/operations.py @@ -263,10 +263,6 @@ class DatabaseOperations(BaseDatabaseOperations): if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - # SQLite doesn't support tz-aware datetimes if timezone.is_aware(value): if settings.USE_TZ: @@ -283,10 +279,6 @@ class DatabaseOperations(BaseDatabaseOperations): if value is None: return None - # Expression values are adapted by the database. - if hasattr(value, "resolve_expression"): - return value - # SQLite doesn't support tz-aware datetimes if timezone.is_aware(value): raise ValueError("SQLite backend does not support timezone-aware times.") diff --git a/django/db/models/lookups.py b/django/db/models/lookups.py index c0bcc1b3bf..4a6e2b3241 100644 --- a/django/db/models/lookups.py +++ b/django/db/models/lookups.py @@ -268,11 +268,16 @@ class FieldGetDbPrepValueMixin: getattr(field, "get_db_prep_value", None) or self.lhs.output_field.get_db_prep_value ) + if not self.get_db_prep_lookup_value_is_iterable: + value = [value] return ( "%s", - [get_db_prep_value(v, connection, prepared=True) for v in value] - if self.get_db_prep_lookup_value_is_iterable - else [get_db_prep_value(value, connection, prepared=True)], + [ + v + if hasattr(v, "as_sql") + else get_db_prep_value(v, connection, prepared=True) + for v in value + ], ) diff --git a/tests/backends/base/test_operations.py b/tests/backends/base/test_operations.py index d2a3fb6765..8df02ee76b 100644 --- a/tests/backends/base/test_operations.py +++ b/tests/backends/base/test_operations.py @@ -4,7 +4,7 @@ from unittest import mock from django.core.management.color import no_style from django.db import NotSupportedError, connection, transaction from django.db.backends.base.operations import BaseDatabaseOperations -from django.db.models import DurationField, Value +from django.db.models import DurationField from django.db.models.expressions import Col from django.db.models.lookups import Exact from django.test import ( @@ -89,17 +89,9 @@ class SimpleDatabaseOperationTests(SimpleTestCase): def test_adapt_timefield_value_none(self): self.assertIsNone(self.ops.adapt_timefield_value(None)) - def test_adapt_timefield_value_expression(self): - value = Value(timezone.now().time()) - self.assertEqual(self.ops.adapt_timefield_value(value), value) - def test_adapt_datetimefield_value_none(self): self.assertIsNone(self.ops.adapt_datetimefield_value(None)) - def test_adapt_datetimefield_value_expression(self): - value = Value(timezone.now()) - self.assertEqual(self.ops.adapt_datetimefield_value(value), value) - def test_adapt_timefield_value(self): msg = "Django does not support timezone-aware times." with self.assertRaisesMessage(ValueError, msg): diff --git a/tests/expressions/tests.py b/tests/expressions/tests.py index 5ba786db13..909e317dca 100644 --- a/tests/expressions/tests.py +++ b/tests/expressions/tests.py @@ -1225,7 +1225,7 @@ class IterableLookupInnerExpressionsTests(TestCase): queryset = Company.objects.filter(name__in=[F("num_chairs") + "1)) OR ((1==1"]) self.assertQuerySetEqual(queryset, [], ordered=False) - def test_range_lookup_allows_F_expressions_and_expressions_for_datetimes(self): + def test_range_lookup_allows_F_expressions_and_expressions_for_dates(self): start = datetime.datetime(2016, 2, 3, 15, 0, 0) end = datetime.datetime(2016, 2, 5, 15, 0, 0) experiment_1 = Experiment.objects.create( @@ -1256,9 +1256,19 @@ class IterableLookupInnerExpressionsTests(TestCase): experiment=experiment_2, result_time=datetime.datetime(2016, 1, 8, 5, 0, 0), ) - within_experiment_time = [F("experiment__start"), F("experiment__end")] - queryset = Result.objects.filter(result_time__range=within_experiment_time) - self.assertSequenceEqual(queryset, [r1]) + tests = [ + # Datetimes. + ([F("experiment__start"), F("experiment__end")], "result_time__range"), + # Dates. + ( + [F("experiment__start__date"), F("experiment__end__date")], + "result_time__date__range", + ), + ] + for within_experiment_time, lookup in tests: + with self.subTest(lookup=lookup): + queryset = Result.objects.filter(**{lookup: within_experiment_time}) + self.assertSequenceEqual(queryset, [r1]) class FTests(SimpleTestCase):