Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

INTPYTHON-393 Remove ObjectIdAutoField acceptance of integer values #256

Merged
merged 1 commit into from
Feb 25, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions django_mongodb_backend/features.py
Original file line number Diff line number Diff line change
Expand Up @@ -205,6 +205,8 @@ def django_test_expected_failures(self):
"lookup.tests.LookupTests.test_in_ignore_none_with_unhashable_items",
"m2m_through_regress.tests.ThroughLoadDataTestCase.test_sequence_creation",
"many_to_many.tests.ManyToManyTests.test_add_remove_invalid_type",
"many_to_one.tests.ManyToOneTests.test_fk_to_smallautofield",
"many_to_one.tests.ManyToOneTests.test_fk_to_bigautofield",
"migrations.test_operations.OperationTests.test_autofield__bigautofield_foreignfield_growth",
"migrations.test_operations.OperationTests.test_model_with_bigautofield",
"migrations.test_operations.OperationTests.test_smallfield_autofield_foreignfield_growth",
Expand All @@ -213,6 +215,8 @@ def django_test_expected_failures(self):
"model_fields.test_autofield.BigAutoFieldTests",
"model_fields.test_autofield.SmallAutoFieldTests",
"queries.tests.TestInvalidValuesRelation.test_invalid_values",
"schema.tests.SchemaTests.test_alter_autofield_pk_to_bigautofield_pk",
"schema.tests.SchemaTests.test_alter_autofield_pk_to_smallautofield_pk",
},
"Converters aren't run on returning fields from insert.": {
# Unsure this is needed for this backend. Can implement by request.
Expand All @@ -231,6 +235,7 @@ def django_test_expected_failures(self):
"queries.test_qs_combinators.QuerySetSetOperationTests.test_order_raises_on_non_selected_column",
"queries.tests.RelatedLookupTypeTests.test_values_queryset_lookup",
"queries.tests.ValuesSubqueryTests.test_values_in_subquery",
"sites_tests.tests.CreateDefaultSiteTests.test_no_site_id",
},
"Cannot use QuerySet.delete() when querying across multiple collections on MongoDB.": {
"admin_changelist.tests.ChangeListTests.test_distinct_for_many_to_many_at_second_level_in_search_fields",
Expand Down
34 changes: 3 additions & 31 deletions django_mongodb_backend/fields/auto.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
from bson import ObjectId, errors
from django.core import exceptions
from django.db.models.fields import AutoField
from django.utils.functional import cached_property

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

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

def get_internal_type(self):
return "ObjectIdAutoField"

def to_python(self, value):
if value is None or isinstance(value, int):
return value
try:
return ObjectId(value)
except errors.InvalidId:
try:
return int(value)
except ValueError:
raise exceptions.ValidationError(
self.error_messages["invalid"],
code="invalid",
params={"value": value},
) from None

@cached_property
def validators(self):
# Avoid IntegerField validators inherited from AutoField.
Expand Down
1 change: 1 addition & 0 deletions docs/source/contents.rst
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ Table of contents
intro/index
topics/index
ref/index
howto/index
faq
releases/index
internals
Expand Down
26 changes: 26 additions & 0 deletions docs/source/howto/contrib-apps.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
=================================
Configuring Django's contrib apps
=================================

Generally, Django's contribs app work out of the box, but here are some
required adjustments.

``contrib.sites``
=================

Usually the :doc:`sites framework <django:ref/contrib/sites>` requires the
:setting:`SITE_ID` setting to be an integer corresponding to the primary key of
the :class:`~django.contrib.sites.models.Site` object. For MongoDB, however,
all primary keys are :class:`~bson.objectid.ObjectId`\s, and so
:setting:`SITE_ID` must be set accordingly::

from bson import ObjectId

SITE_ID = ObjectId("000000000000000000000001")

You must also use the :setting:`SILENCED_SYSTEM_CHECKS` setting to suppress
Django's system check requiring :setting:`SITE_ID` to be an integer::

SILENCED_SYSTEM_CHECKS = [
"sites.E101", # SITE_ID must be an ObjectId for MongoDB.
]
13 changes: 13 additions & 0 deletions docs/source/howto/index.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
=============
How-to guides
=============

Practical guides covering common tasks and problems.

Project configuration
=====================

.. toctree::
:maxdepth: 1

contrib-apps
3 changes: 2 additions & 1 deletion docs/source/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ First steps
**Getting started:**

- :doc:`Installation <intro/install>`
- :doc:`Configuration <intro/configure>`
- :doc:`Configuring a project <intro/configure>`
- :doc:`howto/contrib-apps`

Getting help
============
Expand Down
4 changes: 4 additions & 0 deletions docs/source/intro/configure.rst
Original file line number Diff line number Diff line change
Expand Up @@ -155,3 +155,7 @@ it into the format above, you can use
This constructs a :setting:`DATABASES` setting equivalent to the first example.

Congratulations, your project is ready to go!

.. seealso::

:doc:`/howto/contrib-apps`
4 changes: 4 additions & 0 deletions docs/source/releases/5.0.x.rst
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,10 @@ Django MongoDB Backend 5.0.x

*Unreleased*

- Backward-incompatible:
:class:`~django_mongodb_backend.fields.ObjectIdAutoField` no longer accepts
integer values. The undocumented behavior eased testing with Django's test
suite which hardcodes many integer primary key values.
- Fixed the inability to save nested embedded model forms.
- Fixed :ref:`persistent database connections
<django:persistent-database-connections>`.
Expand Down
4 changes: 4 additions & 0 deletions docs/source/releases/5.1.x.rst
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,10 @@ Django MongoDB Backend 5.1.x

*Unreleased*

- Backward-incompatible:
:class:`~django_mongodb_backend.fields.ObjectIdAutoField` no longer accepts
integer values. The undocumented behavior eased testing with Django's test
suite which hardcodes many integer primary key values.
- Fixed the inability to save nested embedded model forms.
- Fixed :ref:`persistent database connections
<django:persistent-database-connections>`.
Expand Down
4 changes: 2 additions & 2 deletions tests/indexes_/test_condition.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ def test_negated_not_supported(self):
Index(
name="test",
fields=["headline"],
condition=~Q(pk=True),
condition=~Q(pk__isnull=True),
)._get_condition_mql(Article, schema_editor=editor)

def test_xor_not_supported(self):
Expand All @@ -43,7 +43,7 @@ def test_xor_not_supported(self):
Index(
name="test",
fields=["headline"],
condition=Q(pk=True) ^ Q(pk=False),
condition=Q(pk__isnull=True) ^ Q(pk__isnull=False),
)._get_condition_mql(Article, schema_editor=editor)

def test_operations(self):
Expand Down
7 changes: 5 additions & 2 deletions tests/model_fields_/test_autofield.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
from django.core.exceptions import ValidationError
from django.test import SimpleTestCase

from django_mongodb_backend.fields import ObjectIdAutoField
Expand All @@ -15,6 +16,8 @@ def test_get_internal_type(self):
f = ObjectIdAutoField()
self.assertEqual(f.get_internal_type(), "ObjectIdAutoField")

def test_to_python(self):
def test_to_python_invalid_value(self):
f = ObjectIdAutoField()
self.assertEqual(f.to_python("1"), 1)
msg = "“1” is not a valid Object Id."
with self.assertRaisesMessage(ValidationError, msg):
f.to_python("1")
Loading