mirror of https://github.com/django/django.git
[3.1.x] Fixed #31866 -- Fixed locking proxy models in QuerySet.select_for_update(of=()).
Backport of 60626162f7
from master
This commit is contained in:
parent
02b474f2a4
commit
f139372491
|
@ -975,7 +975,8 @@ class SQLCompiler:
|
||||||
the query.
|
the query.
|
||||||
"""
|
"""
|
||||||
def _get_parent_klass_info(klass_info):
|
def _get_parent_klass_info(klass_info):
|
||||||
for parent_model, parent_link in klass_info['model']._meta.parents.items():
|
concrete_model = klass_info['model']._meta.concrete_model
|
||||||
|
for parent_model, parent_link in concrete_model._meta.parents.items():
|
||||||
parent_list = parent_model._meta.get_parent_list()
|
parent_list = parent_model._meta.get_parent_list()
|
||||||
yield {
|
yield {
|
||||||
'model': parent_model,
|
'model': parent_model,
|
||||||
|
@ -1000,8 +1001,9 @@ class SQLCompiler:
|
||||||
select_fields is filled recursively, so it also contains fields
|
select_fields is filled recursively, so it also contains fields
|
||||||
from the parent models.
|
from the parent models.
|
||||||
"""
|
"""
|
||||||
|
concrete_model = klass_info['model']._meta.concrete_model
|
||||||
for select_index in klass_info['select_fields']:
|
for select_index in klass_info['select_fields']:
|
||||||
if self.select[select_index][0].target.model == klass_info['model']:
|
if self.select[select_index][0].target.model == concrete_model:
|
||||||
return self.select[select_index][0]
|
return self.select[select_index][0]
|
||||||
|
|
||||||
def _get_field_choices():
|
def _get_field_choices():
|
||||||
|
|
|
@ -9,4 +9,7 @@ Django 2.2.16 fixes a data loss bug in 2.2.15.
|
||||||
Bugfixes
|
Bugfixes
|
||||||
========
|
========
|
||||||
|
|
||||||
* ...
|
* Fixed a data loss possibility in the
|
||||||
|
:meth:`~django.db.models.query.QuerySet.select_for_update()`. When using
|
||||||
|
related fields pointing to a proxy model in the ``of`` argument, the
|
||||||
|
corresponding model was not locked (:ticket:`31866`).
|
||||||
|
|
|
@ -9,4 +9,7 @@ Django 3.0.10 fixes a data loss bug in 3.0.9.
|
||||||
Bugfixes
|
Bugfixes
|
||||||
========
|
========
|
||||||
|
|
||||||
* ...
|
* Fixed a data loss possibility in the
|
||||||
|
:meth:`~django.db.models.query.QuerySet.select_for_update()`. When using
|
||||||
|
related fields pointing to a proxy model in the ``of`` argument, the
|
||||||
|
corresponding model was not locked (:ticket:`31866`).
|
||||||
|
|
|
@ -20,3 +20,8 @@ Bugfixes
|
||||||
|
|
||||||
* Adjusted admin's navigation sidebar template to reduce debug logging when
|
* Adjusted admin's navigation sidebar template to reduce debug logging when
|
||||||
rendering (:ticket:`31865`).
|
rendering (:ticket:`31865`).
|
||||||
|
|
||||||
|
* Fixed a data loss possibility in the
|
||||||
|
:meth:`~django.db.models.query.QuerySet.select_for_update()`. When using
|
||||||
|
related fields pointing to a proxy model in the ``of`` argument, the
|
||||||
|
corresponding model was not locked (:ticket:`31866`).
|
||||||
|
|
|
@ -23,6 +23,20 @@ class EUCity(models.Model):
|
||||||
country = models.ForeignKey(EUCountry, models.CASCADE)
|
country = models.ForeignKey(EUCountry, models.CASCADE)
|
||||||
|
|
||||||
|
|
||||||
|
class CountryProxy(Country):
|
||||||
|
class Meta:
|
||||||
|
proxy = True
|
||||||
|
|
||||||
|
|
||||||
|
class CountryProxyProxy(CountryProxy):
|
||||||
|
class Meta:
|
||||||
|
proxy = True
|
||||||
|
|
||||||
|
|
||||||
|
class CityCountryProxy(models.Model):
|
||||||
|
country = models.ForeignKey(CountryProxyProxy, models.CASCADE)
|
||||||
|
|
||||||
|
|
||||||
class Person(models.Model):
|
class Person(models.Model):
|
||||||
name = models.CharField(max_length=30)
|
name = models.CharField(max_length=30)
|
||||||
born = models.ForeignKey(City, models.CASCADE, related_name='+')
|
born = models.ForeignKey(City, models.CASCADE, related_name='+')
|
||||||
|
|
|
@ -15,7 +15,9 @@ from django.test import (
|
||||||
)
|
)
|
||||||
from django.test.utils import CaptureQueriesContext
|
from django.test.utils import CaptureQueriesContext
|
||||||
|
|
||||||
from .models import City, Country, EUCity, EUCountry, Person, PersonProfile
|
from .models import (
|
||||||
|
City, CityCountryProxy, Country, EUCity, EUCountry, Person, PersonProfile,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class SelectForUpdateTests(TransactionTestCase):
|
class SelectForUpdateTests(TransactionTestCase):
|
||||||
|
@ -195,6 +197,21 @@ class SelectForUpdateTests(TransactionTestCase):
|
||||||
expected = [connection.ops.quote_name(value) for value in expected]
|
expected = [connection.ops.quote_name(value) for value in expected]
|
||||||
self.assertTrue(self.has_for_update_sql(ctx.captured_queries, of=expected))
|
self.assertTrue(self.has_for_update_sql(ctx.captured_queries, of=expected))
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('has_select_for_update_of')
|
||||||
|
def test_for_update_sql_model_proxy_generated_of(self):
|
||||||
|
with transaction.atomic(), CaptureQueriesContext(connection) as ctx:
|
||||||
|
list(CityCountryProxy.objects.select_related(
|
||||||
|
'country',
|
||||||
|
).select_for_update(
|
||||||
|
of=('country',),
|
||||||
|
))
|
||||||
|
if connection.features.select_for_update_of_column:
|
||||||
|
expected = ['select_for_update_country"."entity_ptr_id']
|
||||||
|
else:
|
||||||
|
expected = ['select_for_update_country']
|
||||||
|
expected = [connection.ops.quote_name(value) for value in expected]
|
||||||
|
self.assertTrue(self.has_for_update_sql(ctx.captured_queries, of=expected))
|
||||||
|
|
||||||
@skipUnlessDBFeature('has_select_for_update_of')
|
@skipUnlessDBFeature('has_select_for_update_of')
|
||||||
def test_for_update_of_followed_by_values(self):
|
def test_for_update_of_followed_by_values(self):
|
||||||
with transaction.atomic():
|
with transaction.atomic():
|
||||||
|
@ -353,6 +370,19 @@ class SelectForUpdateTests(TransactionTestCase):
|
||||||
with transaction.atomic():
|
with transaction.atomic():
|
||||||
EUCountry.objects.select_for_update(of=('name',)).get()
|
EUCountry.objects.select_for_update(of=('name',)).get()
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('has_select_for_update', 'has_select_for_update_of')
|
||||||
|
def test_model_proxy_of_argument_raises_error_proxy_field_in_choices(self):
|
||||||
|
msg = (
|
||||||
|
'Invalid field name(s) given in select_for_update(of=(...)): '
|
||||||
|
'name. Only relational fields followed in the query are allowed. '
|
||||||
|
'Choices are: self, country, country__entity_ptr.'
|
||||||
|
)
|
||||||
|
with self.assertRaisesMessage(FieldError, msg):
|
||||||
|
with transaction.atomic():
|
||||||
|
CityCountryProxy.objects.select_related(
|
||||||
|
'country',
|
||||||
|
).select_for_update(of=('name',)).get()
|
||||||
|
|
||||||
@skipUnlessDBFeature('has_select_for_update', 'has_select_for_update_of')
|
@skipUnlessDBFeature('has_select_for_update', 'has_select_for_update_of')
|
||||||
def test_reverse_one_to_one_of_arguments(self):
|
def test_reverse_one_to_one_of_arguments(self):
|
||||||
"""
|
"""
|
||||||
|
|
Loading…
Reference in New Issue