@@ -453,6 +453,10 @@ class to ensure the same object layout and then use the same "weird"
453
453
obj .__dict__ = self .__dict__ .copy ()
454
454
return obj
455
455
456
+ def _fallback (self , value : Any ) -> Any :
457
+ """Allow deriving classes to implement a final fallback coercion attempt."""
458
+ return value
459
+
456
460
def _try_coerce (self , value : Any , force : bool = False ) -> Union [Enum , Any ]:
457
461
"""
458
462
Attempt coercion of value to enumeration type instance, if unsuccessful
@@ -482,7 +486,10 @@ def _try_coerce(self, value: Any, force: bool = False) -> Union[Enum, Any]:
482
486
)
483
487
except Exception : # pylint: disable=W0703
484
488
pass
485
- if self .strict or not isinstance (value , self .primitive ):
489
+ value = self ._fallback (value )
490
+ if not isinstance (value , self .enum ) and (
491
+ self .strict or not isinstance (value , self .primitive )
492
+ ):
486
493
raise ValueError (
487
494
f"'{ value } ' is not a valid "
488
495
f"{ self .enum .__name__ } required by field "
@@ -774,10 +781,52 @@ def __init__(
774
781
class EnumFloatField (EnumField [Type [float ]], FloatField ):
775
782
"""A database field supporting enumerations with floating point values"""
776
783
784
+ _tolerance_ : float
785
+ _value_primitives_ : List [Tuple [float , Enum ]]
786
+
777
787
@property
778
788
def primitive (self ):
779
789
return EnumField .primitive .fget (self ) or float # type: ignore
780
790
791
+ def _fallback (self , value : Any ) -> Any :
792
+ if value and isinstance (value , float ):
793
+ for en_value , en in self ._value_primitives_ :
794
+ if abs (en_value - value ) < self ._tolerance_ :
795
+ return en
796
+ return value
797
+
798
+ def __init__ (
799
+ self ,
800
+ enum : Optional [Type [Enum ]] = None ,
801
+ primitive : Optional [Type [float ]] = None ,
802
+ strict : bool = EnumField ._strict_ ,
803
+ coerce : bool = EnumField ._coerce_ ,
804
+ constrained : Optional [bool ] = None ,
805
+ ** kwargs ,
806
+ ):
807
+ super ().__init__ (
808
+ enum = enum ,
809
+ primitive = primitive ,
810
+ strict = strict ,
811
+ coerce = coerce ,
812
+ constrained = constrained ,
813
+ ** kwargs ,
814
+ )
815
+ # some database backends (earlier supported versions of Postgres)
816
+ # can't rely on straight equality because of floating point imprecision
817
+ if self .enum :
818
+ self ._value_primitives_ = []
819
+ for en in self .enum :
820
+ if en .value is not None :
821
+ self ._value_primitives_ .append (
822
+ (self ._coerce_to_value_type (en .value ), en )
823
+ )
824
+ self ._tolerance_ = (
825
+ min ((prim [0 ] * 1e-6 for prim in self ._value_primitives_ ))
826
+ if self ._value_primitives_
827
+ else 0.0
828
+ )
829
+
781
830
782
831
class IntEnumField (EnumField [Type [int ]]):
783
832
"""
0 commit comments