Skip to content

Commit 1142ddc

Browse files
committed
readme updates, fix form tests
1 parent 9c14812 commit 1142ddc

File tree

3 files changed

+100
-95
lines changed

3 files changed

+100
-95
lines changed

README.md

+84-88
Original file line numberDiff line numberDiff line change
@@ -43,71 +43,68 @@ Many packages aim to ease usage of Python enumerations as model fields. Most wer
4343
[django-enum](https://pypi.org/project/enum-properties) provides a new model field type, [EnumField](https://django-enum.rtfd.io/en/stable/reference/fields.html#django_enum.fields.EnumField), that allows you to treat almost any [PEP435](https://peps.python.org/pep-0435) enumeration as a database column. [EnumField](https://django-enum.rtfd.io/en/stable/reference/fields.html#django_enum.fields.EnumField) resolves the correct native [Django](https://www.djangoproject.com) field type for the given enumeration based on its value type and range. For example, [IntegerChoices](https://docs.djangoproject.com/en/stable/ref/models/fields/#field-choices-enum-types) that contain values between 0 and 32767 become [PositiveSmallIntegerField](https://docs.djangoproject.com/en/stable/ref/models/fields/#positivesmallintegerfield).
4444

4545
```python
46+
from django.db import models
47+
from django_enum import EnumField
4648

47-
from django.db import models
48-
from django_enum import EnumField
49+
class MyModel(models.Model):
4950

50-
class MyModel(models.Model):
51+
class TextEnum(models.TextChoices):
5152

52-
class TextEnum(models.TextChoices):
53+
VALUE0 = 'V0', 'Value 0'
54+
VALUE1 = 'V1', 'Value 1'
55+
VALUE2 = 'V2', 'Value 2'
5356

54-
VALUE0 = 'V0', 'Value 0'
55-
VALUE1 = 'V1', 'Value 1'
56-
VALUE2 = 'V2', 'Value 2'
57+
class IntEnum(models.IntegerChoices):
5758

58-
class IntEnum(models.IntegerChoices):
59+
ONE = 1, 'One'
60+
TWO = 2, 'Two',
61+
THREE = 3, 'Three'
5962

60-
ONE = 1, 'One'
61-
TWO = 2, 'Two',
62-
THREE = 3, 'Three'
63+
# this is equivalent to:
64+
# CharField(max_length=2, choices=TextEnum.choices, null=True, blank=True)
65+
txt_enum = EnumField(TextEnum, null=True, blank=True)
6366

64-
# this is equivalent to:
65-
# CharField(max_length=2, choices=TextEnum.choices, null=True, blank=True)
66-
txt_enum = EnumField(TextEnum, null=True, blank=True)
67-
68-
# this is equivalent to
69-
# PositiveSmallIntegerField(choices=IntEnum.choices, default=IntEnum.ONE.value)
70-
int_enum = EnumField(IntEnum, default=IntEnum.ONE)
67+
# this is equivalent to
68+
# PositiveSmallIntegerField(choices=IntEnum.choices, default=IntEnum.ONE.value)
69+
int_enum = EnumField(IntEnum, default=IntEnum.ONE)
7170
```
7271

7372
[EnumField](https://django-enum.rtfd.io/en/stable/reference/fields.html#django_enum.fields.EnumField) **is more than just an alias. The fields are now assignable and accessible as their enumeration type rather than by-value:**
7473

7574
```python
75+
instance = MyModel.objects.create(
76+
txt_enum=MyModel.TextEnum.VALUE1,
77+
int_enum=3 # by-value assignment also works
78+
)
7679

77-
instance = MyModel.objects.create(
78-
txt_enum=MyModel.TextEnum.VALUE1,
79-
int_enum=3 # by-value assignment also works
80-
)
81-
82-
assert instance.txt_enum == MyModel.TextEnum('V1')
83-
assert instance.txt_enum.label == 'Value 1'
80+
assert instance.txt_enum == MyModel.TextEnum('V1')
81+
assert instance.txt_enum.label == 'Value 1'
8482

85-
assert instance.int_enum == MyModel.IntEnum['THREE']
86-
assert instance.int_enum.value == 3
83+
assert instance.int_enum == MyModel.IntEnum['THREE']
84+
assert instance.int_enum.value == 3
8785
```
8886

8987
## Flag Support (BitFields)
9088

9189
[Flag](https://docs.python.org/3/library/enum.html#enum.Flag) types are also seamlessly supported! This allows a database column to behave like a bit field and is an alternative to having multiple boolean columns. There are positive performance implications for using a bit field instead of booleans proportional on the size of the bit field and the types of queries you will run against it. For bit fields more than a few bits long the size reduction both speeds up queries and reduces the required storage space. See the documentation for [discussion and benchmarks](https://django-enum.readthedocs.io/en/latest/performance.html#flags).
9290

9391
```python
92+
class Permissions(IntFlag):
9493

95-
class Permissions(IntFlag):
96-
97-
READ = 1 << 0
98-
WRITE = 1 << 1
99-
EXECUTE = 1 << 2
94+
READ = 1 << 0
95+
WRITE = 1 << 1
96+
EXECUTE = 1 << 2
10097

10198

102-
class FlagExample(models.Model):
99+
class FlagExample(models.Model):
103100

104-
permissions = EnumField(Permissions)
101+
permissions = EnumField(Permissions)
105102

106103

107-
FlagExample.objects.create(permissions=Permissions.READ | Permissions.WRITE)
104+
FlagExample.objects.create(permissions=Permissions.READ | Permissions.WRITE)
108105

109-
# get all models with RW:
110-
FlagExample.objects.filter(permissions__has_all=Permissions.READ | Permissions.WRITE)
106+
# get all models with RW:
107+
FlagExample.objects.filter(permissions__has_all=Permissions.READ | Permissions.WRITE)
111108
```
112109

113110
## Complex Enumerations
@@ -117,82 +114,81 @@ Many packages aim to ease usage of Python enumerations as model fields. Most wer
117114
``?> pip install enum-properties``
118115

119116
```python
117+
from enum_properties import StrEnumProperties
118+
from django.db import models
120119

121-
from enum_properties import StrEnumProperties
122-
from django.db import models
120+
class TextChoicesExample(models.Model):
123121

124-
class TextChoicesExample(models.Model):
122+
class Color(StrEnumProperties):
125123

126-
class Color(StrEnumProperties):
124+
# attribute type hints become properties on each value,
125+
# and the enumeration may be instantiated from any symmetric
126+
# property's value
127127

128-
label: Annotated[str, Symmetric()]
129-
rgb: Annotated[t.Tuple[int, int, int], Symmetric()]
130-
hex: Annotated[str, Symmetric(case_fold=True)]
131-
132-
# name value label rgb hex
133-
RED = "R", "Red", (1, 0, 0), "ff0000"
134-
GREEN = "G", "Green", (0, 1, 0), "00ff00"
135-
BLUE = "B", "Blue", (0, 0, 1), "0000ff"
128+
label: Annotated[str, Symmetric()]
129+
rgb: Annotated[t.Tuple[int, int, int], Symmetric()]
130+
hex: Annotated[str, Symmetric(case_fold=True)]
136131

137-
# any named s() values in the Enum's inheritance become properties on
138-
# each value, and the enumeration value may be instantiated from the
139-
# property's value
132+
# properties specified in type hint order after the value
133+
# name value label rgb hex
134+
RED = "R", "Red", (1, 0, 0), "ff0000"
135+
GREEN = "G", "Green", (0, 1, 0), "00ff00"
136+
BLUE = "B", "Blue", (0, 0, 1), "0000ff"
140137

141-
color = EnumField(Color)
138+
color = EnumField(Color)
142139

143-
instance = TextChoicesExample.objects.create(
144-
color=TextChoicesExample.Color('FF0000')
145-
)
146-
assert instance.color == TextChoicesExample.Color('Red')
147-
assert instance.color == TextChoicesExample.Color('R')
148-
assert instance.color == TextChoicesExample.Color((1, 0, 0))
140+
instance = TextChoicesExample.objects.create(
141+
color=TextChoicesExample.Color('FF0000')
142+
)
143+
assert instance.color == TextChoicesExample.Color('Red')
144+
assert instance.color == TextChoicesExample.Color('R')
145+
assert instance.color == TextChoicesExample.Color((1, 0, 0))
149146

150-
# direct comparison to any symmetric value also works
151-
assert instance.color == 'Red'
152-
assert instance.color == 'R'
153-
assert instance.color == (1, 0, 0)
147+
# direct comparison to any symmetric value also works
148+
assert instance.color == 'Red'
149+
assert instance.color == 'R'
150+
assert instance.color == (1, 0, 0)
154151

155-
# save by any symmetric value
156-
instance.color = 'FF0000'
152+
# save by any symmetric value
153+
instance.color = 'FF0000'
157154

158-
# access any enum property right from the model field
159-
assert instance.color.hex == 'ff0000'
155+
# access any enum property right from the model field
156+
assert instance.color.hex == 'ff0000'
160157

161-
# this also works!
162-
assert instance.color == 'ff0000'
158+
# this also works!
159+
assert instance.color == 'ff0000'
163160

164-
# and so does this!
165-
assert instance.color == 'FF0000'
161+
# and so does this!
162+
assert instance.color == 'FF0000'
166163

167-
instance.save()
164+
instance.save()
168165

169-
# filtering works by any symmetric value or enum type instance
170-
assert TextChoicesExample.objects.filter(
171-
color=TextChoicesExample.Color.RED
172-
).first() == instance
166+
# filtering works by any symmetric value or enum type instance
167+
assert TextChoicesExample.objects.filter(
168+
color=TextChoicesExample.Color.RED
169+
).first() == instance
173170

174-
assert TextChoicesExample.objects.filter(color=(1, 0, 0)).first() == instance
171+
assert TextChoicesExample.objects.filter(color=(1, 0, 0)).first() == instance
175172

176-
assert TextChoicesExample.objects.filter(color='FF0000').first() == instance
173+
assert TextChoicesExample.objects.filter(color='FF0000').first() == instance
177174
```
178175

179176
While they should be unnecessary if you need to integrate with code that expects an interface fully compatible with Django's [TextChoices](https://docs.djangoproject.com/en/stable/ref/models/fields/#field-choices-enum-types) and [IntegerChoices](https://docs.djangoproject.com/en/stable/ref/models/fields/#field-choices-enum-types) [django-enum](https://pypi.org/project/django-enum) provides [TextChoices](https://django-enum.rtfd.io/en/stable/reference/choices.html#django_enum.choices.TextChoices), [IntegerChoices](https://django-enum.rtfd.io/en/stable/reference/choices.html#django_enum.choices.IntegerChoices), [FlagChoices](https://django-enum.rtfd.io/en/stable/reference/choices.html#django_enum.choices.FlagChoices) and [FloatChoices](https://django-enum.rtfd.io/en/stable/reference/choices.html#django_enum.choices.FloatChoices) types that derive from enum-properties and Django's ``Choices``. So the above enumeration could also be written:
180177

181178
```python
179+
from django_enum.choices import TextChoices
182180

183-
from django_enum.choices import TextChoices
184-
185-
class Color(TextChoices):
181+
class Color(TextChoices):
186182

187-
# label is added as a symmetric property by the base class
183+
# label is added as a symmetric property by the base class
188184

189-
rgb: Annotated[t.Tuple[int, int, int], Symmetric()]
190-
hex: Annotated[str, Symmetric(case_fold=True)]
185+
rgb: Annotated[t.Tuple[int, int, int], Symmetric()]
186+
hex: Annotated[str, Symmetric(case_fold=True)]
191187

192-
# name value label rgb hex
193-
RED = "R", "Red", (1, 0, 0), "ff0000"
194-
GREEN = "G", "Green", (0, 1, 0), "00ff00"
195-
BLUE = "B", "Blue", (0, 0, 1), "0000ff"
188+
# name value label rgb hex
189+
RED = "R", "Red", (1, 0, 0), "ff0000"
190+
GREEN = "G", "Green", (0, 1, 0), "00ff00"
191+
BLUE = "B", "Blue", (0, 0, 1), "0000ff"
196192

197193
```
198194

tests/test_admin.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -19,13 +19,12 @@
1919
StrTestEnum,
2020
NullableStrEnum,
2121
)
22-
from playwright.sync_api import sync_playwright, Page, expect
22+
from playwright.sync_api import sync_playwright, expect
2323
from django.urls import reverse
2424
from django.contrib.auth import get_user_model
2525
from django.db.models import Model
2626
from django.db.models.fields import NOT_PROVIDED
2727
from django_enum.utils import decompose
28-
from asgiref.sync import sync_to_async
2928

3029

3130
class TestAdmin(EnumTypeMixin, LiveServerTestCase):
@@ -394,7 +393,7 @@ def changes(self) -> t.Dict[str, Enum]:
394393
{
395394
"required": NullableStrEnum.STR2,
396395
"required_default": NullableStrEnum.STR2,
397-
"blank": None,
396+
"blank": NullableStrEnum.STR1,
398397
"blank_nullable": "",
399398
"blank_nullable_default": "",
400399
},

tests/test_forms.py

+14-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
from django.test import TestCase
2+
import django
23
from tests.utils import EnumTypeMixin
34
from tests.djenum.models import EnumTester, Bug53Tester, NullableStrEnum
45
from tests.djenum.forms import EnumTesterForm
@@ -150,7 +151,11 @@ class Meta:
150151
model = NullBlankFormTester
151152

152153
form = NullBlankFormTesterForm(
153-
data={"required": ExternEnum.TWO, "required_default": ExternEnum.ONE}
154+
data={
155+
"required": ExternEnum.TWO,
156+
"required_default": ExternEnum.ONE,
157+
"blank": None,
158+
}
154159
)
155160

156161
# null=False, blank=false
@@ -179,10 +184,15 @@ class Meta:
179184
[("", "---------"), (1, "ONE"), (2, "TWO"), (3, "THREE")],
180185
)
181186

182-
# with self.assertRaises(ValueError):
183-
# because blank will error out on save
184-
form.full_clean()
187+
with self.assertRaises(ValueError):
188+
# because blank will error out on save - this is correct behavior
189+
# because the form allows blank, but the field does not - this is an
190+
# issue with how the user specifies their field (null=False, blank=True)
191+
# with no blank value conversion
192+
form.full_clean()
193+
form.save()
185194

195+
# the form is valid because the error happens on model save!
186196
self.assertTrue(form.is_valid(), form.errors)
187197
self.assertEqual(form.cleaned_data["required"], ExternEnum.TWO)
188198
self.assertEqual(form.cleaned_data["required_default"], ExternEnum.ONE)

0 commit comments

Comments
 (0)