#33397: Arithmetic operations on DateField/DateTimeField/DurationField
expressions
are buggy.
-------------------------------------+-------------------------------------
Reporter: Luke | Owner: Luke Plant
Plant |
Type: Bug | Status: assigned
Component: Database | Version: 4.0
layer (models, ORM) |
Severity: Normal | Keywords:
Triage Stage: | Has patch: 0
Unreviewed |
Needs documentation: 0 | Needs tests: 0
Patch needs improvement: 0 | Easy pickings: 0
UI/UX: 0 |
-------------------------------------+-------------------------------------
There are two main sets of bugs here:
1. Nonsensical operations involving `DateField` or `DateTimeField`
expressions (such as adding two dates) do not raise the expected
`FieldError` exceptions. They usually raise exceptions later that vary
depending on the database backend.
2. Well-defined operations, which work in SQL, such as ’date + duration’,
require using `ExpressionWrapper(output_field=…)` when this could be
inferred.
Although we could technically split this into two bugs, I’m filing as one
since the two parts are closely related and fixing part 2 (which is the
real reason I’m
here) will require some changes that impinge on part 1.
== Part 1
Test case
{{{#!python
# tests/experiments/tests.py
class FTimeDeltaTests(TestCase):
def test_nonsensical_date_operations(self):
queryset = Experiment.objects.annotate(nonsense=F('name') +
F('assigned'))
with self.assertRaises(FieldError):
list(queryset)
queryset = Experiment.objects.annotate(nonsense=F('assigned') +
F('completed'))
with self.assertRaises(FieldError):
list(queryset)
}}}
The first part works as expected (it makes no sense to add a string to a
date, and a `FieldError` is raised), but the second doesn’t.
Expected behaviour: `FieldError` should be raised
Actual behaviour:
With Postgres:
{{{
File "/home/luke/devel/django/main/django/db/models/query.py", line 51,
in __iter__
results = compiler.execute_sql(chunked_fetch=self.chunked_fetch,
chunk_size=self.chunk_size)
File "/home/luke/devel/django/main/django/db/models/sql/compiler.py",
line 1211, in execute_sql
cursor.execute(sql, params)
File "/home/luke/devel/django/main/django/db/backends/utils.py", line
67, in execute
return self._execute_with_wrappers(sql, params, many=False,
executor=self._execute)
File "/home/luke/devel/django/main/django/db/backends/utils.py", line
76, in _execute_with_wrappers
return executor(sql, params, many, context)
File "/home/luke/devel/django/main/django/db/backends/utils.py", line
85, in _execute
return self.cursor.execute(sql, params)
File "/home/luke/devel/django/main/django/db/utils.py", line 90, in
__exit__
raise dj_exc_value.with_traceback(traceback) from exc_value
File "/home/luke/devel/django/main/django/db/backends/utils.py", line
85, in _execute
return self.cursor.execute(sql, params)
psycopg2.errors.UndefinedFunction: operator does not exist: date + date
LINE 1: ...t"."scalar", ("expressions_ExPeRiMeNt"."assigned" + "express...
^
HINT: No operator matches the given name and argument types. You might
need to add explicit type casts.
}}}
With SQLite:
{{{
File "/home/luke/devel/django/main/django/db/models/query.py", line 68,
in __iter__
for row in compiler.results_iter(results):
File "/home/luke/devel/django/main/django/db/models/sql/compiler.py",
line 1158, in apply_converters
value = converter(value, expression, connection)
File
"/home/luke/devel/django/main/django/db/backends/sqlite3/operations.py",
line 305, in convert_datefield_value
value = parse_date(value)
File "/home/luke/devel/django/main/django/utils/dateparse.py", line 76,
in parse_date
return datetime.date.fromisoformat(value)
TypeError: fromisoformat: argument must be str
}}}
I have not tested on other databases.
=== Additional notes
There is a related bug in some context. For example, in contrast to the
above test case `Experiment.objects.filter(name=F('name') +
F('assigned'))` does not raise `FieldError`, despite the attempt to add a
date to a string. Instead you get backend dependent results - SQLite
silently does some type coercion, Postgres fails with `UndefinedFunction`.
Tackling this may need to be done separately - there are different code
paths involved when using `QuerySet.annotate()` compared to
`QuerySet.filter()`
== Part 2
Test case:
{{{#!python
# tests/experiments/tests.py
class FTimeDeltaTests(TestCase):
def
test_datetime_and_duration_field_addition_without_output_field(self):
test_set = Experiment.objects.annotate(estimated_end=F('start') +
F('estimated_time'))
self.assertEqual(
[e.estimated_end for e in test_set],
[e.start + e.estimated_time for e in test_set]
)
}}}
Expected behaviour: Django should infer the output type, like it does for
other expressions such as integer field addition.
Actual behaviour: Django raises `django.core.exceptions.FieldError:
Expression contains mixed types: DateTimeField, DurationField. You must
set output_field.`
=== Additional motivation
If we have this code:
{{{#!python
Experiment.objects.filter(end__gt=F('start') + F('estimated_time'))
}}}
we should be able to refactor to:
{{{#!python
Experiment.objects.alias(estimated_end=F('start') +
F('estimated_time')).filter(end__gt=F('estimated_end'))
}}}
But the latter fails with `FieldError`. The former succeeds because in
that context the ORM doesn't need to do any type inference.
== Notes
* Above tests have been run against `main` branch.
* There are a bunch of other cases, like "date multiplied by date" etc.
that don't work as expected.
--
Ticket URL: <https://code.djangoproject.com/ticket/33397>
Django <https://code.djangoproject.com/>
The Web framework for perfectionists with deadlines.
--
You received this message because you are subscribed to the Google Groups
"Django updates" group.
To unsubscribe from this group and stop receiving emails from it, send an email
to [email protected].
To view this discussion on the web visit
https://groups.google.com/d/msgid/django-updates/054.8cef7c506d545eb453397613540ce316%40djangoproject.com.