Skip to content

Commit a67ba32

Browse files
committed
INTPYTHON-393 Remove ObjectIdAutoField acceptance of integer values
1 parent 18e7177 commit a67ba32

File tree

11 files changed

+68
-35
lines changed

11 files changed

+68
-35
lines changed

django_mongodb_backend/features.py

+5
Original file line numberDiff line numberDiff line change
@@ -205,6 +205,8 @@ def django_test_expected_failures(self):
205205
"lookup.tests.LookupTests.test_in_ignore_none_with_unhashable_items",
206206
"m2m_through_regress.tests.ThroughLoadDataTestCase.test_sequence_creation",
207207
"many_to_many.tests.ManyToManyTests.test_add_remove_invalid_type",
208+
"many_to_one.tests.ManyToOneTests.test_fk_to_smallautofield",
209+
"many_to_one.tests.ManyToOneTests.test_fk_to_bigautofield",
208210
"migrations.test_operations.OperationTests.test_autofield__bigautofield_foreignfield_growth",
209211
"migrations.test_operations.OperationTests.test_model_with_bigautofield",
210212
"migrations.test_operations.OperationTests.test_smallfield_autofield_foreignfield_growth",
@@ -213,6 +215,8 @@ def django_test_expected_failures(self):
213215
"model_fields.test_autofield.BigAutoFieldTests",
214216
"model_fields.test_autofield.SmallAutoFieldTests",
215217
"queries.tests.TestInvalidValuesRelation.test_invalid_values",
218+
"schema.tests.SchemaTests.test_alter_autofield_pk_to_bigautofield_pk",
219+
"schema.tests.SchemaTests.test_alter_autofield_pk_to_smallautofield_pk",
216220
},
217221
"Converters aren't run on returning fields from insert.": {
218222
# Unsure this is needed for this backend. Can implement by request.
@@ -231,6 +235,7 @@ def django_test_expected_failures(self):
231235
"queries.test_qs_combinators.QuerySetSetOperationTests.test_order_raises_on_non_selected_column",
232236
"queries.tests.RelatedLookupTypeTests.test_values_queryset_lookup",
233237
"queries.tests.ValuesSubqueryTests.test_values_in_subquery",
238+
"sites_tests.tests.CreateDefaultSiteTests.test_no_site_id",
234239
},
235240
"Cannot use QuerySet.delete() when querying across multiple collections on MongoDB.": {
236241
"admin_changelist.tests.ChangeListTests.test_distinct_for_many_to_many_at_second_level_in_search_fields",

django_mongodb_backend/fields/auto.py

+3-31
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
from bson import ObjectId, errors
2-
from django.core import exceptions
31
from django.db.models.fields import AutoField
42
from django.utils.functional import cached_property
53

@@ -22,39 +20,13 @@ def deconstruct(self):
2220
return name, path, args, kwargs
2321

2422
def get_prep_value(self, value):
25-
if value is None:
26-
return None
27-
# Accept int for compatibility with Django's test suite which has many
28-
# instances of manually assigned integer IDs, as well as for things
29-
# like settings.SITE_ID which has a system check requiring an integer.
30-
if isinstance(value, (ObjectId | int)):
31-
return value
32-
try:
33-
return ObjectId(value)
34-
except errors.InvalidId as e:
35-
# A manually assigned integer ID?
36-
if isinstance(value, str) and value.isdigit():
37-
return int(value)
38-
raise ValueError(f"Field '{self.name}' expected an ObjectId but got {value!r}.") from e
23+
# Override to omit super() which would call AutoField/IntegerField's
24+
# implementation that requires value to be an integer.
25+
return self.to_python(value)
3926

4027
def get_internal_type(self):
4128
return "ObjectIdAutoField"
4229

43-
def to_python(self, value):
44-
if value is None or isinstance(value, int):
45-
return value
46-
try:
47-
return ObjectId(value)
48-
except errors.InvalidId:
49-
try:
50-
return int(value)
51-
except ValueError:
52-
raise exceptions.ValidationError(
53-
self.error_messages["invalid"],
54-
code="invalid",
55-
params={"value": value},
56-
) from None
57-
5830
@cached_property
5931
def validators(self):
6032
# Avoid IntegerField validators inherited from AutoField.

docs/source/contents.rst

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ Table of contents
1313
intro/index
1414
topics/index
1515
ref/index
16+
howto/index
1617
faq
1718
releases/index
1819
internals

docs/source/howto/contrib-apps.rst

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
=================================
2+
Configuring Django's contrib apps
3+
=================================
4+
5+
Generally, Django's contribs app work out of the box, but here are some
6+
required adjustments.
7+
8+
``contrib.sites``
9+
=================
10+
11+
Usually the :doc:`sites framework <django:ref/contrib/sites>` requires the
12+
:setting:`SITE_ID` setting to be an integer corresponding to the primary key of
13+
the :class:`~django.contrib.sites.models.Site` object. For MongoDB, however,
14+
all primary keys are :class:`~bson.objectid.ObjectId`\s, and so
15+
:setting:`SITE_ID` must be set accordingly::
16+
17+
from bson import ObjectId
18+
19+
SITE_ID = ObjectId("000000000000000000000001")
20+
21+
You must also use the :setting:`SILENCED_SYSTEM_CHECKS` setting to suppress
22+
Django's system check requiring :setting:`SITE_ID` to be an integer::
23+
24+
SILENCED_SYSTEM_CHECKS = [
25+
"sites.E101", # SITE_ID must be an ObjectId for MongoDB.
26+
]

docs/source/howto/index.rst

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
=============
2+
How-to guides
3+
=============
4+
5+
Practical guides covering common tasks and problems.
6+
7+
Project configuration
8+
=====================
9+
10+
.. toctree::
11+
:maxdepth: 1
12+
13+
contrib-apps

docs/source/index.rst

+2-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,8 @@ First steps
1212
**Getting started:**
1313

1414
- :doc:`Installation <intro/install>`
15-
- :doc:`Configuration <intro/configure>`
15+
- :doc:`Configuring a project <intro/configure>`
16+
- :doc:`howto/contrib-apps`
1617

1718
Getting help
1819
============

docs/source/intro/configure.rst

+4
Original file line numberDiff line numberDiff line change
@@ -155,3 +155,7 @@ it into the format above, you can use
155155
This constructs a :setting:`DATABASES` setting equivalent to the first example.
156156

157157
Congratulations, your project is ready to go!
158+
159+
.. seealso::
160+
161+
:doc:`/howto/contrib-apps`

docs/source/releases/5.0.x.rst

+4
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,10 @@ Django MongoDB Backend 5.0.x
77

88
*Unreleased*
99

10+
- Backward-incompatible:
11+
:class:`~django_mongodb_backend.fields.ObjectIdAutoField` no longer accepts
12+
integer values. The undocumented behavior eased testing with Django's test
13+
suite which hardcodes many integer primary key values.
1014
- Fixed the inability to save nested embedded model forms.
1115
- Fixed :ref:`persistent database connections
1216
<django:persistent-database-connections>`.

docs/source/releases/5.1.x.rst

+4
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,10 @@ Django MongoDB Backend 5.1.x
77

88
*Unreleased*
99

10+
- Backward-incompatible:
11+
:class:`~django_mongodb_backend.fields.ObjectIdAutoField` no longer accepts
12+
integer values. The undocumented behavior eased testing with Django's test
13+
suite which hardcodes many integer primary key values.
1014
- Fixed the inability to save nested embedded model forms.
1115
- Fixed :ref:`persistent database connections
1216
<django:persistent-database-connections>`.

tests/indexes_/test_condition.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ def test_negated_not_supported(self):
3434
Index(
3535
name="test",
3636
fields=["headline"],
37-
condition=~Q(pk=True),
37+
condition=~Q(pk__isnull=True),
3838
)._get_condition_mql(Article, schema_editor=editor)
3939

4040
def test_xor_not_supported(self):
@@ -43,7 +43,7 @@ def test_xor_not_supported(self):
4343
Index(
4444
name="test",
4545
fields=["headline"],
46-
condition=Q(pk=True) ^ Q(pk=False),
46+
condition=Q(pk__isnull=True) ^ Q(pk__isnull=False),
4747
)._get_condition_mql(Article, schema_editor=editor)
4848

4949
def test_operations(self):

tests/model_fields_/test_autofield.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
from django.core.exceptions import ValidationError
12
from django.test import SimpleTestCase
23

34
from django_mongodb_backend.fields import ObjectIdAutoField
@@ -17,4 +18,6 @@ def test_get_internal_type(self):
1718

1819
def test_to_python(self):
1920
f = ObjectIdAutoField()
20-
self.assertEqual(f.to_python("1"), 1)
21+
msg = "“1” is not a valid Object Id."
22+
with self.assertRaisesMessage(ValidationError, msg):
23+
f.to_python("1")

0 commit comments

Comments
 (0)