Fixed #30199 -- Adjusted QuerySet.get_or_create() docs to highlight atomicity warning.

This commit is contained in:
Alex 2019-05-17 03:23:10 -07:00 committed by Carlton Gibson
parent de4832c49b
commit 1686dce06c
3 changed files with 19 additions and 18 deletions

View File

@ -38,6 +38,7 @@ answer newbie questions, and generally made Django that much better:
Alexander Dutton <dev@alexdutton.co.uk> Alexander Dutton <dev@alexdutton.co.uk>
Alexander Myodov <alex@myodov.com> Alexander Myodov <alex@myodov.com>
Alexandr Tatarinov <tatarinov1997@gmail.com> Alexandr Tatarinov <tatarinov1997@gmail.com>
Alex Becker <https://alexcbecker.net/>
Alex Couper <http://alexcouper.com/> Alex Couper <http://alexcouper.com/>
Alex Dedul Alex Dedul
Alex Gaynor <alex.gaynor@gmail.com> Alex Gaynor <alex.gaynor@gmail.com>

View File

@ -488,7 +488,10 @@ this entry are the four standard isolation levels:
or ``None`` to use the server's configured isolation level. However, Django or ``None`` to use the server's configured isolation level. However, Django
works best with and defaults to read committed rather than MySQL's default, works best with and defaults to read committed rather than MySQL's default,
repeatable read. Data loss is possible with repeatable read. repeatable read. Data loss is possible with repeatable read. In particular,
you may see cases where :meth:`~django.db.models.query.QuerySet.get_or_create`
will raise an :exc:`~django.db.IntegrityError` but the object won't appear in
a subsequent :meth:`~django.db.models.query.QuerySet.get` call.
.. _transaction isolation level: https://dev.mysql.com/doc/refman/en/innodb-transaction-isolation-levels.html .. _transaction isolation level: https://dev.mysql.com/doc/refman/en/innodb-transaction-isolation-levels.html

View File

@ -1879,7 +1879,8 @@ Returns a tuple of ``(object, created)``, where ``object`` is the retrieved or
created object and ``created`` is a boolean specifying whether a new object was created object and ``created`` is a boolean specifying whether a new object was
created. created.
This is meant as a shortcut to boilerplatish code. For example:: This is meant to prevent duplicate objects from being created when requests are
made in parallel, and as a shortcut to boilerplatish code. For example::
try: try:
obj = Person.objects.get(first_name='John', last_name='Lennon') obj = Person.objects.get(first_name='John', last_name='Lennon')
@ -1887,8 +1888,9 @@ This is meant as a shortcut to boilerplatish code. For example::
obj = Person(first_name='John', last_name='Lennon', birthday=date(1940, 10, 9)) obj = Person(first_name='John', last_name='Lennon', birthday=date(1940, 10, 9))
obj.save() obj.save()
This pattern gets quite unwieldy as the number of fields in a model goes up. Here, with concurrent requests, multiple attempts to save a ``Person`` with
The above example can be rewritten using ``get_or_create()`` like so:: the same parameters may be made. To avoid this race condition, the above
example can be rewritten using ``get_or_create()`` like so::
obj, created = Person.objects.get_or_create( obj, created = Person.objects.get_or_create(
first_name='John', first_name='John',
@ -1900,6 +1902,15 @@ Any keyword arguments passed to ``get_or_create()`` — *except* an optional one
called ``defaults`` — will be used in a :meth:`get()` call. If an object is called ``defaults`` — will be used in a :meth:`get()` call. If an object is
found, ``get_or_create()`` returns a tuple of that object and ``False``. found, ``get_or_create()`` returns a tuple of that object and ``False``.
.. warning::
This method is atomic assuming that the database enforces uniqueness of the
keyword arguments (see :attr:`~django.db.models.Field.unique` or
:attr:`~django.db.models.Options.unique_together`). If the fields used in the
keyword arguments do not have a uniqueness constraint, concurrent calls to
this method may result in multiple rows with the same parameters being
inserted.
You can specify more complex conditions for the retrieved object by chaining You can specify more complex conditions for the retrieved object by chaining
``get_or_create()`` with ``filter()`` and using :class:`Q objects ``get_or_create()`` with ``filter()`` and using :class:`Q objects
<django.db.models.Q>`. For example, to retrieve Robert or Bob Marley if either <django.db.models.Q>`. For example, to retrieve Robert or Bob Marley if either
@ -1941,20 +1952,6 @@ when you're using manually specified primary keys. If an object needs to be
created and the key already exists in the database, an created and the key already exists in the database, an
:exc:`~django.db.IntegrityError` will be raised. :exc:`~django.db.IntegrityError` will be raised.
This method is atomic assuming correct usage, correct database configuration,
and correct behavior of the underlying database. However, if uniqueness is not
enforced at the database level for the ``kwargs`` used in a ``get_or_create``
call (see :attr:`~django.db.models.Field.unique` or
:attr:`~django.db.models.Options.unique_together`), this method is prone to a
race-condition which can result in multiple rows with the same parameters being
inserted simultaneously.
If you are using MySQL, be sure to use the ``READ COMMITTED`` isolation level
rather than ``REPEATABLE READ`` (the default), otherwise you may see cases
where ``get_or_create`` will raise an :exc:`~django.db.IntegrityError` but the
object won't appear in a subsequent :meth:`~django.db.models.query.QuerySet.get`
call.
Finally, a word on using ``get_or_create()`` in Django views. Please make sure Finally, a word on using ``get_or_create()`` in Django views. Please make sure
to use it only in ``POST`` requests unless you have a good reason not to. to use it only in ``POST`` requests unless you have a good reason not to.
``GET`` requests shouldn't have any effect on data. Instead, use ``POST`` ``GET`` requests shouldn't have any effect on data. Instead, use ``POST``