1
1
from __future__ import annotations
2
2
3
- from dataclasses import dataclass
4
3
from itertools import chain
5
- from typing import Any , ClassVar , Iterable , NamedTuple
4
+ from typing import Any , ClassVar , Iterable
6
5
7
- from attrs import define , evolve
6
+ from attrs import define , evolve , field
8
7
9
8
from ... import Config , utils
10
9
from ... import schema as oai
15
14
from .schemas import Class , ReferencePath , Schemas , parse_reference_path
16
15
17
16
18
- @dataclass
17
+ @define
19
18
class ModelDetails :
19
+ """Container for basic attributes of a model schema that can be computed separately"""
20
+
20
21
required_properties : list [Property ] | None = None
21
22
optional_properties : list [Property ] | None = None
22
23
additional_properties : Property | None = None
23
- relative_imports : set [str ] | None = None
24
- lazy_imports : set [str ] | None = None
24
+ relative_imports : set [str ] = field ( factory = set )
25
+ lazy_imports : set [str ] = field ( factory = set )
25
26
26
27
27
28
@define
@@ -88,11 +89,7 @@ def build(
88
89
)
89
90
if isinstance (data_or_err , PropertyError ):
90
91
return data_or_err , schemas
91
- property_data , details .additional_properties = data_or_err
92
- details .required_properties = property_data .required_props
93
- details .optional_properties = property_data .optional_props
94
- details .relative_imports = property_data .relative_imports
95
- details .lazy_imports = property_data .lazy_imports
92
+ details = data_or_err
96
93
for root in roots :
97
94
if isinstance (root , utils .ClassName ):
98
95
continue
@@ -142,11 +139,11 @@ def additional_properties(self) -> Property | None:
142
139
143
140
@property
144
141
def relative_imports (self ) -> set [str ]:
145
- return self .details .relative_imports or set ()
142
+ return self .details .relative_imports
146
143
147
144
@property
148
145
def lazy_imports (self ) -> set [str ] | None :
149
- return self .details .lazy_imports or set ()
146
+ return self .details .lazy_imports
150
147
151
148
@classmethod
152
149
def convert_value (cls , value : Any ) -> Value | None | PropertyError :
@@ -253,22 +250,14 @@ def _resolve_naming_conflict(first: Property, second: Property, config: Config)
253
250
return None
254
251
255
252
256
- class _PropertyData (NamedTuple ):
257
- optional_props : list [Property ]
258
- required_props : list [Property ]
259
- relative_imports : set [str ]
260
- lazy_imports : set [str ]
261
- schemas : Schemas
262
-
263
-
264
253
def _process_properties ( # noqa: PLR0911
265
254
* ,
266
255
data : oai .Schema ,
267
256
schemas : Schemas ,
268
257
class_name : utils .ClassName ,
269
258
config : Config ,
270
259
roots : set [ReferencePath | utils .ClassName ],
271
- ) -> _PropertyData | PropertyError :
260
+ ) -> tuple [ ModelDetails | PropertyError , Schemas ] :
272
261
from . import property_from_data
273
262
from .merge_properties import merge_properties
274
263
@@ -303,19 +292,19 @@ def _add_if_no_conflict(new_prop: Property) -> PropertyError | None:
303
292
if isinstance (sub_prop , oai .Reference ):
304
293
ref_path = parse_reference_path (sub_prop .ref )
305
294
if isinstance (ref_path , ParseError ):
306
- return PropertyError (detail = ref_path .detail , data = sub_prop )
295
+ return PropertyError (detail = ref_path .detail , data = sub_prop ), schemas
307
296
sub_model = schemas .classes_by_reference .get (ref_path )
308
297
if sub_model is None :
309
- return PropertyError (f"Reference { sub_prop .ref } not found" )
298
+ return PropertyError (f"Reference { sub_prop .ref } not found" ), schemas
310
299
if not isinstance (sub_model , ModelProperty ):
311
- return PropertyError ("Cannot take allOf a non-object" )
300
+ return PropertyError ("Cannot take allOf a non-object" ), schemas
312
301
# Properties of allOf references first should be processed first
313
302
if sub_model .needs_post_processing ():
314
- return PropertyError (f"Reference { sub_model .name } in allOf was not processed" , data = sub_prop )
303
+ return PropertyError (f"Reference { sub_model .name } in allOf was not processed" , data = sub_prop ), schemas
315
304
for prop in chain (sub_model .required_properties , sub_model .optional_properties ):
316
305
err = _add_if_no_conflict (prop )
317
306
if err is not None :
318
- return err
307
+ return err , schemas
319
308
schemas .add_dependencies (ref_path = ref_path , roots = roots )
320
309
else :
321
310
unprocessed_props .extend (sub_prop .properties .items () if sub_prop .properties else [])
@@ -336,12 +325,12 @@ def _add_if_no_conflict(new_prop: Property) -> PropertyError | None:
336
325
if not isinstance (prop_or_error , PropertyError ):
337
326
prop_or_error = _add_if_no_conflict (prop_or_error )
338
327
if isinstance (prop_or_error , PropertyError ):
339
- return prop_or_error
328
+ return prop_or_error , schemas
340
329
341
- return _gather_property_data (properties .values (), schemas )
330
+ return _gather_property_data (properties .values ()) , schemas
342
331
343
332
344
- def _gather_property_data (properties : Iterable [Property ], schemas : Schemas ) -> _PropertyData :
333
+ def _gather_property_data (properties : Iterable [Property ]) -> ModelDetails :
345
334
required_properties : list [Property ] = []
346
335
optional_properties : list [Property ] = []
347
336
relative_imports : set [str ] = set ()
@@ -350,12 +339,12 @@ def _gather_property_data(properties: Iterable[Property], schemas: Schemas) -> _
350
339
(required_properties if prop .required else optional_properties ).append (prop )
351
340
lazy_imports .update (prop .get_lazy_imports (prefix = ".." ))
352
341
relative_imports .update (prop .get_imports (prefix = ".." ))
353
- return _PropertyData (
354
- optional_props = optional_properties ,
355
- required_props = required_properties ,
342
+ return ModelDetails (
343
+ optional_properties = optional_properties ,
344
+ required_properties = required_properties ,
356
345
relative_imports = relative_imports ,
357
346
lazy_imports = lazy_imports ,
358
- schemas = schemas ,
347
+ additional_properties = None ,
359
348
)
360
349
361
350
@@ -410,13 +399,12 @@ def _process_property_data(
410
399
class_info : Class ,
411
400
config : Config ,
412
401
roots : set [ReferencePath | utils .ClassName ],
413
- ) -> tuple [tuple [ _PropertyData , Property | None ] | PropertyError , Schemas ]:
414
- property_data = _process_properties (
402
+ ) -> tuple [ModelDetails | PropertyError , Schemas ]:
403
+ model_details , schemas = _process_properties (
415
404
data = data , schemas = schemas , class_name = class_info .name , config = config , roots = roots
416
405
)
417
- if isinstance (property_data , PropertyError ):
418
- return property_data , schemas
419
- schemas = property_data .schemas
406
+ if isinstance (model_details , PropertyError ):
407
+ return model_details , schemas
420
408
421
409
additional_properties , schemas = _get_additional_properties (
422
410
schema_additional = data .additionalProperties ,
@@ -430,10 +418,11 @@ def _process_property_data(
430
418
elif additional_properties is None :
431
419
pass
432
420
else :
433
- property_data .relative_imports .update (additional_properties .get_imports (prefix = ".." ))
434
- property_data .lazy_imports .update (additional_properties .get_lazy_imports (prefix = ".." ))
421
+ model_details = evolve (model_details , additional_properties = additional_properties )
422
+ model_details .relative_imports .update (additional_properties .get_imports (prefix = ".." ))
423
+ model_details .lazy_imports .update (additional_properties .get_lazy_imports (prefix = ".." ))
435
424
436
- return ( property_data , additional_properties ) , schemas
425
+ return model_details , schemas
437
426
438
427
439
428
def process_model (model_prop : ModelProperty , * , schemas : Schemas , config : Config ) -> Schemas | PropertyError :
@@ -455,12 +444,8 @@ def process_model(model_prop: ModelProperty, *, schemas: Schemas, config: Config
455
444
if isinstance (data_or_err , PropertyError ):
456
445
return data_or_err
457
446
458
- property_data , additional_properties = data_or_err
459
-
460
- model_prop .details .required_properties = property_data .required_props
461
- model_prop .details .optional_properties = property_data .optional_props
462
- model_prop .details .additional_properties = additional_properties
463
- model_prop .set_relative_imports (property_data .relative_imports )
464
- model_prop .set_lazy_imports (property_data .lazy_imports )
447
+ model_prop .details = data_or_err
448
+ model_prop .set_relative_imports (data_or_err .relative_imports )
449
+ model_prop .set_lazy_imports (data_or_err .lazy_imports )
465
450
466
451
return schemas
0 commit comments