31
31
Q ,
32
32
SmallIntegerField ,
33
33
TimeField ,
34
+ expressions ,
34
35
)
35
36
from django .db .models .constraints import CheckConstraint
37
+ from django .db .models .fields import BLANK_CHOICE_DASH
36
38
from django .db .models .query_utils import DeferredAttribute
37
39
from django .utils .deconstruct import deconstructible
38
40
from django .utils .duration import duration_string
39
41
from django .utils .functional import cached_property
40
42
from django .utils .translation import gettext_lazy as _
41
43
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
-
50
44
from django_enum .forms import (
51
45
EnumChoiceField ,
52
46
EnumFlagField ,
@@ -67,14 +61,21 @@ class DatabaseDefault: # type: ignore
67
61
with_typehint ,
68
62
)
69
63
64
+
65
+ class _DatabaseDefault :
66
+ """Spoof DatabaseDefault for Django < 5.0"""
67
+
68
+
69
+ DatabaseDefault = getattr (expressions , "DatabaseDefault" , _DatabaseDefault )
70
+
70
71
CONFORM : Union [Enum , Type [NOT_PROVIDED ]]
71
72
EJECT : Union [Enum , Type [NOT_PROVIDED ]]
72
73
STRICT : Union [Enum , Type [NOT_PROVIDED ]]
73
74
74
- if sys .version_info >= (3 , 11 ): # pragma: no cover
75
+ if sys .version_info >= (3 , 11 ):
75
76
from enum import CONFORM , EJECT , STRICT
76
77
else :
77
- CONFORM = EJECT = STRICT = NOT_PROVIDED # pragma: no cover
78
+ CONFORM = EJECT = STRICT = NOT_PROVIDED
78
79
79
80
80
81
MAX_CONSTRAINT_NAME_LENGTH = 64
@@ -624,7 +625,7 @@ def get_default(self) -> Any:
624
625
return 0
625
626
return super ().get_default ()
626
627
627
- def validate (self , value : Any , model_instance : Model ):
628
+ def validate (self , value : Any , model_instance : Optional [ Model ] ):
628
629
"""
629
630
Validates the field as part of model clean routines. Runs super class
630
631
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):
683
684
form_field .primitive = self .primitive
684
685
return form_field
685
686
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
687
694
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
691
696
return [
692
697
(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
+ )
694
704
]
695
705
696
706
@staticmethod
@@ -716,15 +726,17 @@ def constraint_name(
716
726
return name [len (name ) - MAX_CONSTRAINT_NAME_LENGTH :]
717
727
return name
718
728
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 )
721
733
if self .constrained and self .enum and issubclass (self .enum , IntFlag ):
722
734
# It's possible to declare an IntFlag field with negative values -
723
735
# these enums do not behave has expected and flag-like DB
724
736
# operations are not supported, so they are treated as normal
725
737
# IntEnum fields, but the check constraints are flag-like range
726
738
# 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 )
728
740
elif self .constrained and self .enum :
729
741
constraint = Q (
730
742
** {
@@ -1113,7 +1125,7 @@ class FlagField(with_typehint(IntEnumField)): # type: ignore
1113
1125
enum : Type [Flag ]
1114
1126
1115
1127
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
1117
1129
) -> None :
1118
1130
"""
1119
1131
Add check constraints that honor flag fields range and boundary
@@ -1175,8 +1187,10 @@ def contribute_to_class(
1175
1187
cls ._meta .original_attrs .setdefault ( # pylint: disable=W0212
1176
1188
"constraints" , cls ._meta .constraints # pylint: disable=W0212
1177
1189
)
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 )
1180
1194
1181
1195
1182
1196
class SmallIntegerFlagField (FlagField , EnumPositiveSmallIntegerField ):
@@ -1275,17 +1289,17 @@ def from_db_value(
1275
1289
connection ,
1276
1290
)
1277
1291
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 )
1280
1294
1281
1295
1282
1296
class ExtraBigIntegerFlagField (FlagField , EnumExtraBigIntegerField ):
1283
1297
"""
1284
1298
Flag fields that require more than 64 bits.
1285
1299
"""
1286
1300
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 )
1289
1303
1290
1304
1291
1305
# ExtraBigIntegerFlagField.register_lookup(HasAnyFlagsExtraBigLookup)
0 commit comments