mirror of https://github.com/django/django.git
Refs #373 -- Added Model._is_pk_set() abstraction to check if a Model's PK is set.
This commit is contained in:
parent
cdbd31960e
commit
5865ff5adc
|
@ -31,7 +31,7 @@ class BaseGenericInlineFormSet(BaseModelFormSet):
|
||||||
+ self.ct_fk_field.name
|
+ self.ct_fk_field.name
|
||||||
)
|
)
|
||||||
self.save_as_new = save_as_new
|
self.save_as_new = save_as_new
|
||||||
if self.instance is None or self.instance.pk is None:
|
if self.instance is None or not self.instance._is_pk_set():
|
||||||
qs = self.model._default_manager.none()
|
qs = self.model._default_manager.none()
|
||||||
else:
|
else:
|
||||||
if queryset is None:
|
if queryset is None:
|
||||||
|
|
|
@ -198,7 +198,7 @@ class ExclusionConstraint(BaseConstraint):
|
||||||
lookups.append(lookup)
|
lookups.append(lookup)
|
||||||
queryset = queryset.filter(*lookups)
|
queryset = queryset.filter(*lookups)
|
||||||
model_class_pk = instance._get_pk_val(model._meta)
|
model_class_pk = instance._get_pk_val(model._meta)
|
||||||
if not instance._state.adding and model_class_pk is not None:
|
if not instance._state.adding and instance._is_pk_set(model._meta):
|
||||||
queryset = queryset.exclude(pk=model_class_pk)
|
queryset = queryset.exclude(pk=model_class_pk)
|
||||||
if not self.condition:
|
if not self.condition:
|
||||||
if queryset.exists():
|
if queryset.exists():
|
||||||
|
|
|
@ -601,7 +601,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
return my_pk == other.pk
|
return my_pk == other.pk
|
||||||
|
|
||||||
def __hash__(self):
|
def __hash__(self):
|
||||||
if self.pk is None:
|
if not self._is_pk_set():
|
||||||
raise TypeError("Model instances without primary key value are unhashable")
|
raise TypeError("Model instances without primary key value are unhashable")
|
||||||
return hash(self.pk)
|
return hash(self.pk)
|
||||||
|
|
||||||
|
@ -662,6 +662,9 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
|
|
||||||
pk = property(_get_pk_val, _set_pk_val)
|
pk = property(_get_pk_val, _set_pk_val)
|
||||||
|
|
||||||
|
def _is_pk_set(self, meta=None):
|
||||||
|
return self._get_pk_val(meta) is not None
|
||||||
|
|
||||||
def get_deferred_fields(self):
|
def get_deferred_fields(self):
|
||||||
"""
|
"""
|
||||||
Return a set containing names of deferred fields for this instance.
|
Return a set containing names of deferred fields for this instance.
|
||||||
|
@ -1094,11 +1097,10 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
if f.name in update_fields or f.attname in update_fields
|
if f.name in update_fields or f.attname in update_fields
|
||||||
]
|
]
|
||||||
|
|
||||||
pk_val = self._get_pk_val(meta)
|
if not self._is_pk_set(meta):
|
||||||
if pk_val is None:
|
|
||||||
pk_val = meta.pk.get_pk_value_on_save(self)
|
pk_val = meta.pk.get_pk_value_on_save(self)
|
||||||
setattr(self, meta.pk.attname, pk_val)
|
setattr(self, meta.pk.attname, pk_val)
|
||||||
pk_set = pk_val is not None
|
pk_set = self._is_pk_set(meta)
|
||||||
if not pk_set and (force_update or update_fields):
|
if not pk_set and (force_update or update_fields):
|
||||||
raise ValueError("Cannot force an update in save() with no primary key.")
|
raise ValueError("Cannot force an update in save() with no primary key.")
|
||||||
updated = False
|
updated = False
|
||||||
|
@ -1126,6 +1128,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
for f in non_pks_non_generated
|
for f in non_pks_non_generated
|
||||||
]
|
]
|
||||||
forced_update = update_fields or force_update
|
forced_update = update_fields or force_update
|
||||||
|
pk_val = self._get_pk_val(meta)
|
||||||
updated = self._do_update(
|
updated = self._do_update(
|
||||||
base_qs, using, pk_val, values, update_fields, forced_update
|
base_qs, using, pk_val, values, update_fields, forced_update
|
||||||
)
|
)
|
||||||
|
@ -1226,7 +1229,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
# database to raise an IntegrityError if applicable. If
|
# database to raise an IntegrityError if applicable. If
|
||||||
# constraints aren't supported by the database, there's the
|
# constraints aren't supported by the database, there's the
|
||||||
# unavoidable risk of data corruption.
|
# unavoidable risk of data corruption.
|
||||||
if obj.pk is None:
|
if not obj._is_pk_set():
|
||||||
# Remove the object from a related instance cache.
|
# Remove the object from a related instance cache.
|
||||||
if not field.remote_field.multiple:
|
if not field.remote_field.multiple:
|
||||||
field.remote_field.delete_cached_value(obj)
|
field.remote_field.delete_cached_value(obj)
|
||||||
|
@ -1254,14 +1257,14 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
and hasattr(field, "fk_field")
|
and hasattr(field, "fk_field")
|
||||||
):
|
):
|
||||||
obj = field.get_cached_value(self, default=None)
|
obj = field.get_cached_value(self, default=None)
|
||||||
if obj and obj.pk is None:
|
if obj and not obj._is_pk_set():
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"{operation_name}() prohibited to prevent data loss due to "
|
f"{operation_name}() prohibited to prevent data loss due to "
|
||||||
f"unsaved related object '{field.name}'."
|
f"unsaved related object '{field.name}'."
|
||||||
)
|
)
|
||||||
|
|
||||||
def delete(self, using=None, keep_parents=False):
|
def delete(self, using=None, keep_parents=False):
|
||||||
if self.pk is None:
|
if not self._is_pk_set():
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"%s object can't be deleted because its %s attribute is set "
|
"%s object can't be deleted because its %s attribute is set "
|
||||||
"to None." % (self._meta.object_name, self._meta.pk.attname)
|
"to None." % (self._meta.object_name, self._meta.pk.attname)
|
||||||
|
@ -1367,7 +1370,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
return field_map
|
return field_map
|
||||||
|
|
||||||
def prepare_database_save(self, field):
|
def prepare_database_save(self, field):
|
||||||
if self.pk is None:
|
if not self._is_pk_set():
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Unsaved model instance %r cannot be used in an ORM query." % self
|
"Unsaved model instance %r cannot be used in an ORM query." % self
|
||||||
)
|
)
|
||||||
|
@ -1497,7 +1500,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
# allows single model to have effectively multiple primary keys.
|
# allows single model to have effectively multiple primary keys.
|
||||||
# Refs #17615.
|
# Refs #17615.
|
||||||
model_class_pk = self._get_pk_val(model_class._meta)
|
model_class_pk = self._get_pk_val(model_class._meta)
|
||||||
if not self._state.adding and model_class_pk is not None:
|
if not self._state.adding and self._is_pk_set(model_class._meta):
|
||||||
qs = qs.exclude(pk=model_class_pk)
|
qs = qs.exclude(pk=model_class_pk)
|
||||||
if qs.exists():
|
if qs.exists():
|
||||||
if len(unique_check) == 1:
|
if len(unique_check) == 1:
|
||||||
|
@ -1532,7 +1535,7 @@ class Model(AltersData, metaclass=ModelBase):
|
||||||
qs = model_class._default_manager.filter(**lookup_kwargs)
|
qs = model_class._default_manager.filter(**lookup_kwargs)
|
||||||
# Exclude the current object from the query if we are editing an
|
# Exclude the current object from the query if we are editing an
|
||||||
# instance (as opposed to creating a new one)
|
# instance (as opposed to creating a new one)
|
||||||
if not self._state.adding and self.pk is not None:
|
if not self._state.adding and self._is_pk_set():
|
||||||
qs = qs.exclude(pk=self.pk)
|
qs = qs.exclude(pk=self.pk)
|
||||||
|
|
||||||
if qs.exists():
|
if qs.exists():
|
||||||
|
|
|
@ -686,7 +686,7 @@ class UniqueConstraint(BaseConstraint):
|
||||||
filters.append(condition)
|
filters.append(condition)
|
||||||
queryset = queryset.filter(*filters)
|
queryset = queryset.filter(*filters)
|
||||||
model_class_pk = instance._get_pk_val(model._meta)
|
model_class_pk = instance._get_pk_val(model._meta)
|
||||||
if not instance._state.adding and model_class_pk is not None:
|
if not instance._state.adding and instance._is_pk_set(model._meta):
|
||||||
queryset = queryset.exclude(pk=model_class_pk)
|
queryset = queryset.exclude(pk=model_class_pk)
|
||||||
if not self.condition:
|
if not self.condition:
|
||||||
if queryset.exists():
|
if queryset.exists():
|
||||||
|
|
|
@ -1969,7 +1969,7 @@ class ManyToManyField(RelatedField):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def value_from_object(self, obj):
|
def value_from_object(self, obj):
|
||||||
return [] if obj.pk is None else list(getattr(obj, self.attname).all())
|
return list(getattr(obj, self.attname).all()) if obj._is_pk_set() else []
|
||||||
|
|
||||||
def save_form_data(self, instance, data):
|
def save_form_data(self, instance, data):
|
||||||
getattr(instance, self.attname).set(data)
|
getattr(instance, self.attname).set(data)
|
||||||
|
|
|
@ -511,8 +511,7 @@ class ReverseOneToOneDescriptor:
|
||||||
try:
|
try:
|
||||||
rel_obj = self.related.get_cached_value(instance)
|
rel_obj = self.related.get_cached_value(instance)
|
||||||
except KeyError:
|
except KeyError:
|
||||||
related_pk = instance.pk
|
if not instance._is_pk_set():
|
||||||
if related_pk is None:
|
|
||||||
rel_obj = None
|
rel_obj = None
|
||||||
else:
|
else:
|
||||||
filter_args = self.related.field.get_forward_related_filter(instance)
|
filter_args = self.related.field.get_forward_related_filter(instance)
|
||||||
|
@ -753,7 +752,7 @@ def create_reverse_many_to_one_manager(superclass, rel):
|
||||||
# Even if this relation is not to pk, we require still pk value.
|
# Even if this relation is not to pk, we require still pk value.
|
||||||
# The wish is that the instance has been already saved to DB,
|
# The wish is that the instance has been already saved to DB,
|
||||||
# although having a pk value isn't a guarantee of that.
|
# although having a pk value isn't a guarantee of that.
|
||||||
if self.instance.pk is None:
|
if not self.instance._is_pk_set():
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"{self.instance.__class__.__name__!r} instance needs to have a "
|
f"{self.instance.__class__.__name__!r} instance needs to have a "
|
||||||
f"primary key value before this relationship can be used."
|
f"primary key value before this relationship can be used."
|
||||||
|
@ -1081,7 +1080,7 @@ def create_forward_many_to_many_manager(superclass, rel, reverse):
|
||||||
# Even if this relation is not to pk, we require still pk value.
|
# Even if this relation is not to pk, we require still pk value.
|
||||||
# The wish is that the instance has been already saved to DB,
|
# The wish is that the instance has been already saved to DB,
|
||||||
# although having a pk value isn't a guarantee of that.
|
# although having a pk value isn't a guarantee of that.
|
||||||
if instance.pk is None:
|
if not instance._is_pk_set():
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"%r instance needs to have a primary key value before "
|
"%r instance needs to have a primary key value before "
|
||||||
"a many-to-many relationship can be used."
|
"a many-to-many relationship can be used."
|
||||||
|
|
|
@ -16,7 +16,7 @@ def get_normalized_value(value, lhs):
|
||||||
from django.db.models import Model
|
from django.db.models import Model
|
||||||
|
|
||||||
if isinstance(value, Model):
|
if isinstance(value, Model):
|
||||||
if value.pk is None:
|
if not value._is_pk_set():
|
||||||
raise ValueError("Model instances passed to related filters must be saved.")
|
raise ValueError("Model instances passed to related filters must be saved.")
|
||||||
value_list = []
|
value_list = []
|
||||||
sources = lhs.output_field.path_infos[-1].target_fields
|
sources = lhs.output_field.path_infos[-1].target_fields
|
||||||
|
|
|
@ -668,7 +668,7 @@ class QuerySet(AltersData):
|
||||||
|
|
||||||
connection = connections[self.db]
|
connection = connections[self.db]
|
||||||
for obj in objs:
|
for obj in objs:
|
||||||
if obj.pk is None:
|
if not obj._is_pk_set():
|
||||||
# Populate new PK values.
|
# Populate new PK values.
|
||||||
obj.pk = obj._meta.pk.get_pk_value_on_save(obj)
|
obj.pk = obj._meta.pk.get_pk_value_on_save(obj)
|
||||||
if not connection.features.supports_default_keyword_in_bulk_insert:
|
if not connection.features.supports_default_keyword_in_bulk_insert:
|
||||||
|
@ -794,7 +794,7 @@ class QuerySet(AltersData):
|
||||||
objs = list(objs)
|
objs = list(objs)
|
||||||
self._prepare_for_bulk_create(objs)
|
self._prepare_for_bulk_create(objs)
|
||||||
with transaction.atomic(using=self.db, savepoint=False):
|
with transaction.atomic(using=self.db, savepoint=False):
|
||||||
objs_with_pk, objs_without_pk = partition(lambda o: o.pk is None, objs)
|
objs_without_pk, objs_with_pk = partition(lambda o: o._is_pk_set(), objs)
|
||||||
if objs_with_pk:
|
if objs_with_pk:
|
||||||
returned_columns = self._batched_insert(
|
returned_columns = self._batched_insert(
|
||||||
objs_with_pk,
|
objs_with_pk,
|
||||||
|
@ -862,7 +862,7 @@ class QuerySet(AltersData):
|
||||||
if not fields:
|
if not fields:
|
||||||
raise ValueError("Field names must be given to bulk_update().")
|
raise ValueError("Field names must be given to bulk_update().")
|
||||||
objs = tuple(objs)
|
objs = tuple(objs)
|
||||||
if any(obj.pk is None for obj in objs):
|
if not all(obj._is_pk_set() for obj in objs):
|
||||||
raise ValueError("All bulk_update() objects must have a primary key set.")
|
raise ValueError("All bulk_update() objects must have a primary key set.")
|
||||||
fields = [self.model._meta.get_field(name) for name in fields]
|
fields = [self.model._meta.get_field(name) for name in fields]
|
||||||
if any(not f.concrete or f.many_to_many for f in fields):
|
if any(not f.concrete or f.many_to_many for f in fields):
|
||||||
|
@ -1289,7 +1289,7 @@ class QuerySet(AltersData):
|
||||||
return False
|
return False
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
raise TypeError("'obj' must be a model instance.")
|
raise TypeError("'obj' must be a model instance.")
|
||||||
if obj.pk is None:
|
if not obj._is_pk_set():
|
||||||
raise ValueError("QuerySet.contains() cannot be used on unsaved objects.")
|
raise ValueError("QuerySet.contains() cannot be used on unsaved objects.")
|
||||||
if self._result_cache is not None:
|
if self._result_cache is not None:
|
||||||
return obj in self._result_cache
|
return obj in self._result_cache
|
||||||
|
|
|
@ -220,7 +220,7 @@ class DeferredAttribute:
|
||||||
# might be able to reuse the already loaded value. Refs #18343.
|
# might be able to reuse the already loaded value. Refs #18343.
|
||||||
val = self._check_parent_chain(instance)
|
val = self._check_parent_chain(instance)
|
||||||
if val is None:
|
if val is None:
|
||||||
if instance.pk is None and self.field.generated:
|
if not instance._is_pk_set() and self.field.generated:
|
||||||
raise AttributeError(
|
raise AttributeError(
|
||||||
"Cannot read a generated field from an unsaved model."
|
"Cannot read a generated field from an unsaved model."
|
||||||
)
|
)
|
||||||
|
|
|
@ -935,7 +935,7 @@ class BaseModelFormSet(BaseFormSet, AltersData):
|
||||||
# 1. The object is an unexpected empty model, created by invalid
|
# 1. The object is an unexpected empty model, created by invalid
|
||||||
# POST data such as an object outside the formset's queryset.
|
# POST data such as an object outside the formset's queryset.
|
||||||
# 2. The object was already deleted from the database.
|
# 2. The object was already deleted from the database.
|
||||||
if obj.pk is None:
|
if not obj._is_pk_set():
|
||||||
continue
|
continue
|
||||||
if form in forms_to_delete:
|
if form in forms_to_delete:
|
||||||
self.deleted_objects.append(obj)
|
self.deleted_objects.append(obj)
|
||||||
|
@ -1103,7 +1103,7 @@ class BaseInlineFormSet(BaseModelFormSet):
|
||||||
self.save_as_new = save_as_new
|
self.save_as_new = save_as_new
|
||||||
if queryset is None:
|
if queryset is None:
|
||||||
queryset = self.model._default_manager
|
queryset = self.model._default_manager
|
||||||
if self.instance.pk is not None:
|
if self.instance._is_pk_set():
|
||||||
qs = queryset.filter(**{self.fk.name: self.instance})
|
qs = queryset.filter(**{self.fk.name: self.instance})
|
||||||
else:
|
else:
|
||||||
qs = queryset.none()
|
qs = queryset.none()
|
||||||
|
|
|
@ -973,3 +973,14 @@ Other attributes
|
||||||
since they are yet to be saved. Instances fetched from a ``QuerySet``
|
since they are yet to be saved. Instances fetched from a ``QuerySet``
|
||||||
will have ``adding=False`` and ``db`` set to the alias of the associated
|
will have ``adding=False`` and ``db`` set to the alias of the associated
|
||||||
database.
|
database.
|
||||||
|
|
||||||
|
``_is_pk_set()``
|
||||||
|
----------------
|
||||||
|
|
||||||
|
.. method:: Model._is_pk_set()
|
||||||
|
|
||||||
|
.. versionadded:: 5.2
|
||||||
|
|
||||||
|
The ``_is_pk_set()`` method returns whether the model instance's ``pk`` is set.
|
||||||
|
It abstracts the model's primary key definition, ensuring consistent behavior
|
||||||
|
regardless of the specific ``pk`` configuration.
|
||||||
|
|
|
@ -289,7 +289,9 @@ Database backend API
|
||||||
This section describes changes that may be needed in third-party database
|
This section describes changes that may be needed in third-party database
|
||||||
backends.
|
backends.
|
||||||
|
|
||||||
* ...
|
* The new :meth:`Model._is_pk_set() <django.db.models.Model._is_pk_set>` method
|
||||||
|
allows checking if a Model instance's primary key is defined.
|
||||||
|
|
||||||
|
|
||||||
:mod:`django.contrib.gis`
|
:mod:`django.contrib.gis`
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
|
@ -661,6 +661,31 @@ class ModelTest(TestCase):
|
||||||
headline__startswith="Area",
|
headline__startswith="Area",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_is_pk_unset(self):
|
||||||
|
cases = [
|
||||||
|
Article(),
|
||||||
|
Article(id=None),
|
||||||
|
]
|
||||||
|
for case in cases:
|
||||||
|
with self.subTest(case=case):
|
||||||
|
self.assertIs(case._is_pk_set(), False)
|
||||||
|
|
||||||
|
def test_is_pk_set(self):
|
||||||
|
def new_instance():
|
||||||
|
a = Article(pub_date=datetime.today())
|
||||||
|
a.save()
|
||||||
|
return a
|
||||||
|
|
||||||
|
cases = [
|
||||||
|
Article(id=1),
|
||||||
|
Article(id=0),
|
||||||
|
Article.objects.create(pub_date=datetime.today()),
|
||||||
|
new_instance(),
|
||||||
|
]
|
||||||
|
for case in cases:
|
||||||
|
with self.subTest(case=case):
|
||||||
|
self.assertIs(case._is_pk_set(), True)
|
||||||
|
|
||||||
|
|
||||||
class ModelLookupTest(TestCase):
|
class ModelLookupTest(TestCase):
|
||||||
@classmethod
|
@classmethod
|
||||||
|
|
Loading…
Reference in New Issue