Fixed #34558 -- Fixed QuerySet.bulk_create() crash with Now() on Oracle.

This commit is contained in:
Mariusz Felisiak 2023-05-11 18:22:55 +02:00 committed by GitHub
parent 2b11740e1d
commit 72a86ceb33
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 32 additions and 1 deletions

View File

@ -236,6 +236,11 @@ class Now(Func):
**extra_context, **extra_context,
) )
def as_oracle(self, compiler, connection, **extra_context):
return self.as_sql(
compiler, connection, template="LOCALTIMESTAMP", **extra_context
)
class TruncBase(TimezoneMixin, Transform): class TruncBase(TimezoneMixin, Transform):
kind = None kind = None

View File

@ -562,11 +562,21 @@ Usage example:
``Now()`` uses ``STATEMENT_TIMESTAMP`` instead. If you need the transaction ``Now()`` uses ``STATEMENT_TIMESTAMP`` instead. If you need the transaction
timestamp, use :class:`django.contrib.postgres.functions.TransactionNow`. timestamp, use :class:`django.contrib.postgres.functions.TransactionNow`.
.. admonition:: Oracle
On Oracle, the SQL ``LOCALTIMESTAMP`` is used to avoid issues with casting
``CURRENT_TIMESTAMP`` to ``DateTimeField``.
.. versionchanged:: 4.2 .. versionchanged:: 4.2
Support for microsecond precision on MySQL and millisecond precision on Support for microsecond precision on MySQL and millisecond precision on
SQLite were added. SQLite were added.
.. versionchanged:: 5.0
In older versions, the SQL ``CURRENT_TIMESTAMP`` was used on Oracle instead
of ``LOCALTIMESTAMP``.
``Trunc`` ``Trunc``
--------- ---------

View File

@ -382,6 +382,9 @@ Miscellaneous
* ``ImageField.update_dimension_fields()`` is no longer called on the * ``ImageField.update_dimension_fields()`` is no longer called on the
``post_init`` signal if ``width_field`` and ``height_field`` are not set. ``post_init`` signal if ``width_field`` and ``height_field`` are not set.
* :class:`~django.db.models.functions.Now` database function now uses
``LOCALTIMESTAMP`` instead of ``CURRENT_TIMESTAMP`` on Oracle.
.. _deprecated-features-5.0: .. _deprecated-features-5.0:
Features deprecated in 5.0 Features deprecated in 5.0

View File

@ -10,7 +10,7 @@ from django.db import (
connection, connection,
) )
from django.db.models import FileField, Value from django.db.models import FileField, Value
from django.db.models.functions import Lower from django.db.models.functions import Lower, Now
from django.test import ( from django.test import (
TestCase, TestCase,
override_settings, override_settings,
@ -300,6 +300,19 @@ class BulkCreateTests(TestCase):
bbb = Restaurant.objects.filter(name="betty's beetroot bar") bbb = Restaurant.objects.filter(name="betty's beetroot bar")
self.assertEqual(bbb.count(), 1) self.assertEqual(bbb.count(), 1)
@skipUnlessDBFeature("has_bulk_insert")
def test_bulk_insert_now(self):
NullableFields.objects.bulk_create(
[
NullableFields(datetime_field=Now()),
NullableFields(datetime_field=Now()),
]
)
self.assertEqual(
NullableFields.objects.filter(datetime_field__isnull=False).count(),
2,
)
@skipUnlessDBFeature("has_bulk_insert") @skipUnlessDBFeature("has_bulk_insert")
def test_bulk_insert_nullable_fields(self): def test_bulk_insert_nullable_fields(self):
fk_to_auto_fields = { fk_to_auto_fields = {