Skip to content

Commit 4498c22

Browse files
committed
fix #98, add 5.2 classifier, fix #95, fix #79, steps towards #72
1 parent 0c5b9e1 commit 4498c22

24 files changed

+307
-221
lines changed

doc/source/changelog.rst

+6-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,11 @@ Change Log
77
v2.2.0 (2025-03-07)
88
===================
99

10+
* Implemented `Use intersphinx for doc references <https://github.com/bckohan/django-enum/issues/98>`_
11+
* Implemented `Support Django 5.2 <https://github.com/bckohan/django-enum/issues/96>`_
12+
* Implemented `Upgrade to enum-properties >=2.2 <https://github.com/bckohan/django-enum/issues/95>`_
1013
* Implemented `Move form imports to locally scoped imports where needed in fields.py <https://github.com/bckohan/django-enum/issues/79>`_
14+
* Implemented `Reorganize documentation using diataxis <https://github.com/bckohan/django-enum/issues/72>`_
1115

1216
v2.1.0 (2025-02-24)
1317
===================
@@ -54,7 +58,8 @@ v2.0.0 (2024-09-09)
5458
Migration from 1.x -> 2.x
5559
-------------------------
5660

57-
* Imports of enum-properties_ extended ``TextChoices`` and ``IntegerChoices`` have been changed:
61+
* Imports of :doc:`enum-properties:index` extended ``TextChoices`` and ``IntegerChoices`` have been
62+
changed:
5863

5964
.. code-block:: python
6065

doc/source/conf.py

+15
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,14 @@
11
from datetime import datetime
2+
import os
23
import sys
34
from pathlib import Path
45
from sphinx.ext.autodoc import between
56

67
sys.path.append(str(Path(__file__).parent.parent.parent))
78
import django_enum
89

10+
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'tests.settings')
11+
912
# Configuration file for the Sphinx documentation builder.
1013
#
1114
# This file only contains a selection of the most common options. For a full
@@ -39,6 +42,8 @@
3942
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
4043
# ones.
4144
extensions = [
45+
'sphinxcontrib_django',
46+
'sphinx.ext.intersphinx',
4247
'sphinx.ext.autodoc',
4348
'sphinx.ext.todo'
4449
]
@@ -71,6 +76,16 @@
7176

7277
todo_include_todos = True
7378

79+
intersphinx_mapping = {
80+
"django": (
81+
"https://docs.djangoproject.com/en/stable",
82+
"https://docs.djangoproject.com/en/stable/_objects/",
83+
),
84+
"enum-properties": ("https://enum-properties.readthedocs.io/en/stable", None),
85+
"django-render-static": ("https://django-render-static.readthedocs.io/en/stable", None),
86+
"python": ('https://docs.python.org/3', None)
87+
}
88+
7489

7590
def setup(app):
7691
# Register a sphinx.ext.autodoc.between listener to ignore everything

doc/source/eccentric_enums.rst

+6-6
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,13 @@
66
Eccentric Enumerations
77
======================
88

9-
Python's Enum_ type is extremely lenient. Enumeration values may be any
9+
Python's :class:`enum.Enum` type is extremely lenient. Enumeration values may be any
1010
hashable type and values of the same enumeration may be of different types.
1111

1212
For use in databases it is recommended to use more strict enumeration types
1313
that only allow a single value type of either string or integer. If additional
1414
properties need to be associated with enumeration values, a library like
15-
enum-properties_ should be used to store them on the enumeration value classes.
15+
:doc:`enum-properties:index` should be used to store them on the enumeration value classes.
1616

1717
However, the goal of django-enum is to provide as complete a bridge as possible
1818
between Python and the database so eccentric enumerations are supported with
@@ -63,12 +63,12 @@ be a string.
6363

6464
If none of the supported primitive types are symmetrically coercible
6565
:class:`~django_enum.fields.EnumField` will not be able to determine an appropriate column
66-
type and a ``ValueError`` will be raised.
66+
type and a :exc:`ValueError` will be raised.
6767

6868
In these cases, or to override the primitive type selection made by
6969
:class:`~django_enum.fields.EnumField`, pass the ``primitive`` parameter. It may be necessary to
7070
extend one of the supported primitives to make it coercible. It may also be necessary
71-
to override the Enum_'s ``_missing_`` method:
71+
to override the :class:`enum.Enum`'s ``_missing_`` method:
7272

7373
.. code-block:: python
7474
@@ -87,8 +87,8 @@ Custom Enumeration Values
8787
.. warning::
8888
There is almost certainly a better way to do what you might be trying to do
8989
by writing a custom enumeration value - for example consider using
90-
enum-properties_ to make your enumeration types more robust by pushing more
91-
of this functionality on the Enum_ class itself.
90+
:doc:`enum-properties:index` to make your enumeration types more robust by pushing more
91+
of this functionality on the :class:`enum.Enum` class itself.
9292

9393
If you must use a custom value type, you can by specifying a symmetrically
9494
coercible primitive type. For example Path is already symmetrically coercible

doc/source/usage.rst doc/source/howto.rst

+56-53
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
.. include:: refs.rst
22

3-
=====
4-
Usage
5-
=====
3+
======
4+
How To
5+
======
66

77
:class:`~django_enum.fields.EnumField` inherits from the appropriate native Django_ field and sets
8-
the correct choice tuple set based on the enumeration type. This means ``EnumFields`` are
9-
compatible with all modules, utilities and libraries that fields defined with a choice tuple are.
10-
For example:
8+
the correct choice tuple set based on the enumeration type. This means
9+
:class:`~django_enum.fields.EnumField` are compatible with all modules, utilities and libraries
10+
that fields defined with a choice tuple are. For example:
1111

1212
.. code:: python
1313
@@ -65,13 +65,15 @@ simple declarative syntax are possible with :class:`~django_enum.fields.EnumFiel
6565
External Enum Types
6666
###################
6767

68-
Enum_ classes defined externally to your code base or enum classes that otherwise do not inherit
69-
from Django's Choices_ type, are supported. When no choices are present on an Enum_ type,
70-
:class:`~django_enum.fields.EnumField` will attempt to use the ``label`` member on each
71-
enumeration value if it is present, otherwise the labels will be based off the enumeration name.
72-
Choices can also be overridden at the :class:`~django_enum.fields.EnumField` declaration.
68+
:class:`enum.Enum` classes defined externally to your code base or enum classes that otherwise do
69+
not inherit from Django's :ref:`field-choices-enum-types`, are supported. When no choices are
70+
present on an :class:`enum.Enum` type, :class:`~django_enum.fields.EnumField` will attempt to use
71+
the ``label`` member on each enumeration value if it is present, otherwise the labels will be based
72+
off the enumeration name. Choices can also be overridden at the
73+
:class:`~django_enum.fields.EnumField` declaration.
7374

74-
In short, :class:`~django_enum.fields.EnumField` should work with any subclass of Enum_.
75+
In short, :class:`~django_enum.fields.EnumField` should work with any subclass of
76+
:class:`enum.Enum`.
7577

7678
.. code:: python
7779
@@ -93,8 +95,8 @@ The above code will produce a choices set like ``[('V0', 'VALUE0'), ...]``.
9395
9496
.. warning::
9597
96-
One nice feature of Django's Choices_ type is that it disables
97-
``auto()`` on Enum_ fields. ``auto()`` can be dangerous because the
98+
One nice feature of Django's :ref:`field-choices-enum-types` are that they disable
99+
``auto()`` on :class:`enum.Enum` fields. ``auto()`` can be dangerous because the
98100
values assigned depend on the order of declaration. This means that if the
99101
order changes existing database values will no longer align with the
100102
enumeration values. When using ``Enums`` where control over the values is
@@ -114,15 +116,16 @@ The following :class:`~django_enum.fields.EnumField` specific parameters are ava
114116
``strict``
115117
----------
116118

117-
By default all ``EnumFields`` are ``strict``. This means a ``ValidationError``
118-
will be thrown anytime full_clean is run on a model and a value is set for the
119-
field that can not be coerced to its native Enum_ type. To allow the field
120-
to store values that are not present in the fields Enum_ type we can pass
119+
By default all :class:`~django_enum.fields.EnumField` are ``strict``. This means a
120+
:exc:`~django.core.exceptions.ValidationError` will be thrown anytime
121+
:meth:`django.db.models.Model.full_clean` is run on a model and a value is set for the
122+
field that can not be coerced to its native :class:`enum.Enum` type. To allow the field
123+
to store values that are not present in the fields :class:`enum.Enum` type we can pass
121124
`strict=False`.
122125

123126
Non-strict fields that have values outside of the enumeration will be instances
124-
of the enumeration where a valid Enum_ value is present and the plain old
125-
data where no Enum_ type coercion is possible.
127+
of the enumeration where a valid :class:`enum.Enum` value is present and the plain old
128+
data where no :class:`enum.Enum` type coercion is possible.
126129

127130
.. code-block:: python
128131
@@ -157,11 +160,10 @@ data where no Enum_ type coercion is possible.
157160
``constrained``
158161
---------------
159162

160-
By default all strict ``EnumFields`` are ``constrained``. This means that
161-
`CheckConstraints <https://docs.djangoproject.com/en/stable/ref/models/constraints/>`_ will be
162-
generated at the database level to ensure that the column will reject any value that is not
163-
present in the enumeration. This is a good idea for most use cases, but it can be turned off
164-
by setting ``constrained`` to ``False``.
163+
By default all strict :class:`~django_enum.fields.EnumField` are ``constrained``. This means that
164+
:doc:`CheckConstraints <django:ref/models/constraints>` will be generated at the database level
165+
to ensure that the column will reject any value that is not present in the enumeration. This is a
166+
good idea for most use cases, but it can be turned off by setting ``constrained`` to ``False``.
165167

166168
.. note::
167169

@@ -171,19 +173,19 @@ by setting ``constrained`` to ``False``.
171173
``primitive``
172174
-------------
173175

174-
``EnumFields`` dynamically determine the database column type by determining the most appropriate
175-
primitive type for the enumeration based on the enumeration values. You may override the primitive
176-
determined by :class:`~django_enum.fields.EnumField` by passing a type to the ``primitive``
177-
parameter. You will likely not need to do this unless your enumeration is
178-
:ref:`eccentric <eccentric>` in some way.
176+
:class:`~django_enum.fields.EnumField` dynamically determines the database column type by
177+
determining the most appropriate primitive type for the enumeration based on the enumeration
178+
values. You may override the primitive determined by :class:`~django_enum.fields.EnumField` by
179+
passing a type to the ``primitive`` parameter. You will likely not need to do this unless your
180+
enumeration is :ref:`eccentric <eccentric>` in some way.
179181

180182
``coerce``
181183
----------
182184

183185
Setting this parameter to ``False`` will turn off the automatic conversion to
184-
the field's Enum_ type while leaving all validation checks in place. It will
185-
still be possible to set the field directly as an Enum_ instance and to
186-
filter by Enum_ instance or any symmetric value:
186+
the field's :class:`enum.Enum` type while leaving all validation checks in place. It will
187+
still be possible to set the field directly as an :class:`enum.Enum` instance and to
188+
filter by :class:`enum.Enum` instance or any symmetric value:
187189

188190
.. code-block:: python
189191
@@ -210,21 +212,22 @@ filter by Enum_ instance or any symmetric value:
210212
enum-properties
211213
###############
212214

213-
Almost any Enum_ type is supported, so you may make use of Enum_ extension libraries like
214-
enum-properties_ to define very rich enumeration fields:
215+
Almost any :class:`enum.Enum` type is supported, so you may make use of :class:`enum.Enum`
216+
extension libraries like :doc:`enum-properties:index` to define very rich enumeration fields:
215217

216218
.. code:: bash
217219
218220
pip install enum-properties
219221
220-
enum-properties_ is an extension to Enum_ that allows properties to be added to enumeration
221-
instances using a simple declarative syntax. This is a less awkward and more compatible alternative
222-
than dataclass enumerations.
222+
:doc:`enum-properties:index` is an extension to :class:`enum.Enum` that allows properties to be
223+
added to enumeration instances using a simple declarative syntax. This is a less awkward and more
224+
compatible alternative than dataclass enumerations.
223225

224-
If you find yourself considering a dataclass enumeration, consider using enum-properties_ instead.
225-
dataclass enumerations do not work with :class:`~django_enum.fields.EnumField` because their value
226-
type is a dataclass. Futher, most libraries that expect to be able to work with enumerations expect
227-
the ``value`` attribute to be a primitive serializable type.
226+
If you find yourself considering a dataclass enumeration, consider using
227+
:doc:`enum-properties:index` instead. dataclass enumerations do not work with
228+
:class:`~django_enum.fields.EnumField` because their value type is a dataclass. Futher, most
229+
libraries that expect to be able to work with enumerations expect the ``value`` attribute to be a
230+
primitive serializable type.
228231

229232
.. code-block:: python
230233
@@ -295,7 +298,7 @@ compatible with Django's
295298
(``TextChoices`` and ``IntegerChoices`` django-enum_ provides
296299
:class:`~django_enum.choices.TextChoices`, :class:`~django_enum.choices.IntegerChoices`,
297300
:class:`~django_enum.choices.FlagChoices` and :class:`~django_enum.choices.FloatChoices` types that
298-
derive from enum-properties_ and Django's ``Choices``. So the above enumeration could also be
301+
derive from :doc:`enum-properties:index` and Django's ``Choices``. So the above enumeration could also be
299302
written:
300303

301304
.. code-block:: python
@@ -316,7 +319,7 @@ written:
316319
317320
.. note::
318321

319-
To use these ``Choices`` extensions you will need to install enum-properties_ which is an
322+
To use these ``Choices`` extensions you will need to install :doc:`enum-properties:index` which is an
320323
optional dependency.
321324

322325
.. _forms:
@@ -327,7 +330,7 @@ Forms
327330
An ``EnumChoiceField`` type is provided that enables symmetric value resolution
328331
and will automatically coerce any set value to the underlying enumeration type.
329332
Django_'s ``ModelForms`` will use this form field type to represent
330-
``EnumFields`` by default. For most scenarios this is sufficient. The
333+
:class:`~django_enum.fields.EnumField` by default. For most scenarios this is sufficient. The
331334
``EnumChoiceField`` can also be explicitly used. For example, using our
332335
``TextChoicesExample`` from above - if ``color`` was declared with
333336
`strict=False`, we could add additional choices to our form field like so:
@@ -453,8 +456,8 @@ dependency on django-filter_ is optional, you must first install it:
453456
# now filtering by symmetric value in url parameters works:
454457
# e.g.: /?color=FF0000
455458
456-
An ``EnumFilterSet`` type is also provided that uses ``EnumFilter`` for ``EnumFields``
457-
by default. So the above is also equivalent to:
459+
An ``EnumFilterSet`` type is also provided that uses ``EnumFilter`` for
460+
:class:`~django_enum.fields.EnumField` by default. So the above is also equivalent to:
458461

459462
.. code-block::
460463
@@ -482,21 +485,21 @@ Migrations
482485
*Never reference or import your enumeration classes in a migration file,
483486
work with the primitive values instead*.
484487

485-
The deconstructed ``EnumFields`` only include the choices tuple in the
486-
migration files. This is because Enum_ classes may come and go or be
488+
The deconstructed :class:`~django_enum.fields.EnumField` only include the choices tuple in the
489+
migration files. This is because :class:`enum.Enum` classes may come and go or be
487490
altered but the earlier migration files must still work. Simply treat any
488491
custom migration routines as if they were operating on a normal model field
489492
with choices.
490493

491-
``EnumFields`` in migration files will not resolve the field values to
494+
:class:`~django_enum.fields.EnumField` in migration files will not resolve the field values to
492495
enumeration types. The fields will be the primitive enumeration values as they
493496
are with any field with choices.
494497

495498
Flag Enumerations
496499
#################
497500

498501
Python supports `bit masks <https://en.wikipedia.org/wiki/Mask_(computing)>`_ through the
499-
`Flag <https://docs.python.org/3/library/enum.html#enum.Flag>`_ extension to Enum_.
502+
`Flag <https://docs.python.org/3/library/enum.html#enum.Flag>`_ extension to :class:`enum.Enum`.
500503

501504
These enumerations are fully supported and will render as multi select form fields
502505
by default. For example:
@@ -572,8 +575,8 @@ Flags with more than 64 bits
572575
----------------------------
573576

574577
Flag enumerations of arbitrary size are supported, however if the enum has more
575-
than 64 flags it will be stored as a `BinaryField <https://docs.djangoproject.com/en/stable/ref/models/fields/#binaryfield>`_.
576-
It is therefore strongly recommended to keep your Flag_ enumerations at 64 bits or less.
578+
than 64 flags it will be stored as a :class:`django.db.models.BinaryField`. It is therefore
579+
strongly recommended to keep your :class:`enum.Flag` enumerations at 64 bits or less.
577580

578581
.. warning::
579582

0 commit comments

Comments
 (0)