1
0
mirror of https://github.com/openSUSE/osc.git synced 2025-01-07 15:06:22 +01:00

Add support for List[BaseModel] type to Field class

This commit is contained in:
Daniel Mach 2024-01-23 20:05:38 +01:00
parent ea0bf1bb60
commit 7903ade2b4
2 changed files with 86 additions and 0 deletions

View File

@ -130,6 +130,23 @@ class Field(property):
return get_origin(types[0]) or types[0] return get_origin(types[0]) or types[0]
return origin_type return origin_type
@property
def inner_type(self):
if self.is_optional:
types = [i for i in self.type.__args__ if i != type(None)]
type_ = types[0]
else:
type_ = self.type
if get_origin(type_) != list:
return None
if not hasattr(type_, "__args__"):
return None
inner_type = [i for i in type_.__args__ if i != type(None)][0]
return inner_type
@property @property
def is_optional(self): def is_optional(self):
origin_type = get_origin(self.type) or self.type origin_type = get_origin(self.type) or self.type
@ -139,6 +156,10 @@ class Field(property):
def is_model(self): def is_model(self):
return inspect.isclass(self.origin_type) and issubclass(self.origin_type, BaseModel) return inspect.isclass(self.origin_type) and issubclass(self.origin_type, BaseModel)
@property
def is_model_list(self):
return inspect.isclass(self.inner_type) and issubclass(self.inner_type, BaseModel)
def validate_type(self, value, expected_types=None): def validate_type(self, value, expected_types=None):
if not expected_types and self.is_optional and value is None: if not expected_types and self.is_optional and value is None:
return True return True
@ -255,6 +276,15 @@ class Field(property):
# initialize a model instance from a dictionary # initialize a model instance from a dictionary
klass = self.origin_type klass = self.origin_type
value = klass(**value) # pylint: disable=not-callable value = klass(**value) # pylint: disable=not-callable
elif self.is_model_list and isinstance(value, list):
new_value = []
for i in value:
if isinstance(i, dict):
klass = self.inner_type
new_value.append(klass(**i))
else:
new_value.append(i)
value = new_value
self.validate_type(value) self.validate_type(value)
obj._values[self.name] = value obj._values[self.name] = value
@ -342,6 +372,8 @@ class BaseModel(metaclass=ModelMeta):
value = getattr(self, name) value = getattr(self, name)
if value is not None and field.is_model: if value is not None and field.is_model:
result[name] = value.dict() result[name] = value.dict()
if value is not None and field.is_model_list:
result[name] = [i.dict() for i in value]
else: else:
result[name] = value result[name] = value

View File

@ -198,6 +198,60 @@ class Test(unittest.TestCase):
self.assertEqual(m.field.text, "text") self.assertEqual(m.field.text, "text")
m.dict() m.dict()
def test_list_submodels(self):
class TestSubmodel(BaseModel):
text: str = Field(default="default")
class TestModel(BaseModel):
field: List[TestSubmodel] = Field(default=[])
m = TestModel()
field = m.__fields__["field"]
self.assertEqual(field.is_model, False)
self.assertEqual(field.is_model_list, True)
self.assertEqual(field.is_optional, False)
self.assertEqual(field.origin_type, list)
m.dict()
m = TestModel(field=[TestSubmodel()])
self.assertEqual(m.field[0].text, "default")
m.dict()
m = TestModel(field=[{"text": "text"}])
self.assertEqual(m.field[0].text, "text")
m.dict()
self.assertRaises(TypeError, getattr(m, "field"))
def test_optional_list_submodels(self):
class TestSubmodel(BaseModel):
text: str = Field(default="default")
class TestModel(BaseModel):
field: Optional[List[TestSubmodel]] = Field(default=[])
m = TestModel()
field = m.__fields__["field"]
self.assertEqual(field.is_model, False)
self.assertEqual(field.is_model_list, True)
self.assertEqual(field.is_optional, True)
self.assertEqual(field.origin_type, list)
m.dict()
m = TestModel(field=[TestSubmodel()])
self.assertEqual(m.field[0].text, "default")
m.dict()
m = TestModel(field=[{"text": "text"}])
self.assertEqual(m.field[0].text, "text")
m.dict()
m.field = None
self.assertEqual(m.field, None)
m.dict()
def test_enum(self): def test_enum(self):
class Numbers(Enum): class Numbers(Enum):
one = "one" one = "one"