Skip to content

Commit f98f9a7

Browse files
committed
use django-stubs during static type checking, fixes #60
1 parent db1f571 commit f98f9a7

File tree

7 files changed

+121
-62
lines changed

7 files changed

+121
-62
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -133,3 +133,4 @@ dmypy.json
133133
test.db
134134
django_enum/tests/edit_tests/migrations/00*.py
135135
benchmark.db
136+
type_check.py

django_enum/choices.py

+6-8
Original file line numberDiff line numberDiff line change
@@ -9,21 +9,19 @@
99
from django.db.models import Choices
1010
from django.db.models import IntegerChoices as DjangoIntegerChoices
1111
from django.db.models import TextChoices as DjangoTextChoices
12-
13-
try:
14-
from django.db.models.enums import ChoicesType
15-
except ImportError: # pragma: no cover
16-
from django.db.models.enums import ChoicesMeta as ChoicesType
12+
from django.db.models import enums as model_enums
1713

1814
from django_enum.utils import choices, names
1915

16+
ChoicesType = getattr(model_enums, "ChoicesType", getattr(model_enums, "ChoicesMeta"))
17+
2018
DEFAULT_BOUNDARY = getattr(enum, "KEEP", None)
2119

2220

2321
try:
2422
from enum_properties import DecomposeMixin, EnumPropertiesMeta, SymmetricMixin
2523

26-
class DjangoEnumPropertiesMeta(EnumPropertiesMeta, ChoicesType):
24+
class DjangoEnumPropertiesMeta(EnumPropertiesMeta, ChoicesType): # type: ignore
2725
"""
2826
A composite meta class that combines Django's Choices metaclass with
2927
enum-properties metaclass. This metaclass will add Django's expected
@@ -37,15 +35,15 @@ def names(cls):
3735
For some eccentric enums list(Enum) is empty, so we override names
3836
if empty
3937
"""
40-
return ChoicesType.names.fget(cls) or names(cls, override=True)
38+
return super().names or names(cls, override=True)
4139

4240
@property
4341
def choices(cls):
4442
"""
4543
For some eccentric enums list(Enum) is empty, so we override
4644
choices if empty
4745
"""
48-
return ChoicesType.choices.fget(cls) or choices(cls, override=True)
46+
return super().choices or choices(cls, override=True)
4947

5048
class DjangoSymmetricMixin(SymmetricMixin):
5149
"""

django_enum/fields.py

+40-26
Original file line numberDiff line numberDiff line change
@@ -31,22 +31,16 @@
3131
Q,
3232
SmallIntegerField,
3333
TimeField,
34+
expressions,
3435
)
3536
from django.db.models.constraints import CheckConstraint
37+
from django.db.models.fields import BLANK_CHOICE_DASH
3638
from django.db.models.query_utils import DeferredAttribute
3739
from django.utils.deconstruct import deconstructible
3840
from django.utils.duration import duration_string
3941
from django.utils.functional import cached_property
4042
from django.utils.translation import gettext_lazy as _
4143

42-
try:
43-
from django.db.models.expressions import DatabaseDefault
44-
except ImportError: # pragma: no cover
45-
46-
class DatabaseDefault: # type: ignore
47-
"""Spoof DatabaseDefault for Django < 5.0"""
48-
49-
5044
from django_enum.forms import (
5145
EnumChoiceField,
5246
EnumFlagField,
@@ -67,14 +61,21 @@ class DatabaseDefault: # type: ignore
6761
with_typehint,
6862
)
6963

64+
65+
class _DatabaseDefault:
66+
"""Spoof DatabaseDefault for Django < 5.0"""
67+
68+
69+
DatabaseDefault = getattr(expressions, "DatabaseDefault", _DatabaseDefault)
70+
7071
CONFORM: Union[Enum, Type[NOT_PROVIDED]]
7172
EJECT: Union[Enum, Type[NOT_PROVIDED]]
7273
STRICT: Union[Enum, Type[NOT_PROVIDED]]
7374

74-
if sys.version_info >= (3, 11): # pragma: no cover
75+
if sys.version_info >= (3, 11):
7576
from enum import CONFORM, EJECT, STRICT
7677
else:
77-
CONFORM = EJECT = STRICT = NOT_PROVIDED # pragma: no cover
78+
CONFORM = EJECT = STRICT = NOT_PROVIDED
7879

7980

8081
MAX_CONSTRAINT_NAME_LENGTH = 64
@@ -624,7 +625,7 @@ def get_default(self) -> Any:
624625
return 0
625626
return super().get_default()
626627

627-
def validate(self, value: Any, model_instance: Model):
628+
def validate(self, value: Any, model_instance: Optional[Model]):
628629
"""
629630
Validates the field as part of model clean routines. Runs super class
630631
validation routines then tries to convert the value to a valid
@@ -683,14 +684,23 @@ def formfield(self, form_class=None, choices_form_class=None, **kwargs):
683684
form_field.primitive = self.primitive
684685
return form_field
685686

686-
def get_choices(self, **kwargs): # pylint: disable=W0221
687+
def get_choices(
688+
self,
689+
include_blank=True,
690+
blank_choice=tuple(BLANK_CHOICE_DASH),
691+
limit_choices_to=None,
692+
ordering=(),
693+
): # pylint: disable=W0221
687694
if self.enum and issubclass(self.enum, Flag):
688-
kwargs["blank_choice"] = [
689-
(self.enum(0), "---------") # pylint: disable=E1102
690-
]
695+
blank_choice = [(self.enum(0), "---------")] # pylint: disable=E1102
691696
return [
692697
(getattr(choice, "value", choice), label)
693-
for choice, label in super().get_choices(**kwargs)
698+
for choice, label in super().get_choices(
699+
include_blank=include_blank,
700+
blank_choice=list(blank_choice),
701+
limit_choices_to=limit_choices_to,
702+
ordering=ordering,
703+
)
694704
]
695705

696706
@staticmethod
@@ -716,15 +726,17 @@ def constraint_name(
716726
return name[len(name) - MAX_CONSTRAINT_NAME_LENGTH :]
717727
return name
718728

719-
def contribute_to_class(self, cls, name, **kwargs): # pylint: disable=W0221
720-
super().contribute_to_class(cls, name, **kwargs)
729+
def contribute_to_class(
730+
self, cls: Type[Model], name: str, private_only: bool = False
731+
): # pylint: disable=W0221
732+
super().contribute_to_class(cls, name, private_only=private_only)
721733
if self.constrained and self.enum and issubclass(self.enum, IntFlag):
722734
# It's possible to declare an IntFlag field with negative values -
723735
# these enums do not behave has expected and flag-like DB
724736
# operations are not supported, so they are treated as normal
725737
# IntEnum fields, but the check constraints are flag-like range
726738
# constraints, so we bring those in here
727-
FlagField.contribute_to_class(self, cls, name, call_base=False, **kwargs)
739+
FlagField.contribute_to_class(self, cls, name, private_only=private_only)
728740
elif self.constrained and self.enum:
729741
constraint = Q(
730742
**{
@@ -1113,7 +1125,7 @@ class FlagField(with_typehint(IntEnumField)): # type: ignore
11131125
enum: Type[Flag]
11141126

11151127
def contribute_to_class(
1116-
self, cls: Type[EnumField], name: str, call_base: bool = True, **kwargs
1128+
self, cls: Type[Model], name: str, private_only: bool = False
11171129
) -> None:
11181130
"""
11191131
Add check constraints that honor flag fields range and boundary
@@ -1175,8 +1187,10 @@ def contribute_to_class(
11751187
cls._meta.original_attrs.setdefault( # pylint: disable=W0212
11761188
"constraints", cls._meta.constraints # pylint: disable=W0212
11771189
)
1178-
if call_base:
1179-
IntegerField.contribute_to_class(self, cls, name, **kwargs)
1190+
if isinstance(self, FlagField):
1191+
# this may have been called by a normal EnumField to bring in flag-like constraints
1192+
# for non flag fields
1193+
IntegerField.contribute_to_class(self, cls, name, private_only=private_only)
11801194

11811195

11821196
class SmallIntegerFlagField(FlagField, EnumPositiveSmallIntegerField):
@@ -1275,17 +1289,17 @@ def from_db_value(
12751289
connection,
12761290
)
12771291

1278-
def contribute_to_class(self, cls, name, **kwargs):
1279-
BinaryField.contribute_to_class(self, cls, name, **kwargs)
1292+
def contribute_to_class(self, cls, name, private_only: bool = False):
1293+
BinaryField.contribute_to_class(self, cls, name, private_only=private_only)
12801294

12811295

12821296
class ExtraBigIntegerFlagField(FlagField, EnumExtraBigIntegerField):
12831297
"""
12841298
Flag fields that require more than 64 bits.
12851299
"""
12861300

1287-
def contribute_to_class(self, cls, name, call_base=True, **kwargs):
1288-
BinaryField.contribute_to_class(self, cls, name, **kwargs)
1301+
def contribute_to_class(self, cls, name, private_only: bool = False):
1302+
BinaryField.contribute_to_class(self, cls, name, private_only=private_only)
12891303

12901304

12911305
# ExtraBigIntegerFlagField.register_lookup(HasAnyFlagsExtraBigLookup)

django_enum/filters.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
from typing import Tuple, Type
44

55
from django.db.models import Field as ModelField
6-
from django.forms.fields import Field as FormField
76

87
from django_enum.forms import EnumChoiceField
98
from django_enum.utils import choices
@@ -40,7 +39,7 @@ class Color(TextChoices, s('rgb'), s('hex', case_fold=True)):
4039
:param kwargs: Any additional arguments for base classes
4140
"""
4241

43-
field_class: FormField = EnumChoiceField
42+
field_class = EnumChoiceField
4443

4544
def __init__(self, *, enum, strict=False, **kwargs):
4645
self.enum = enum

0 commit comments

Comments
 (0)