forked from openapi-generators/openapi-python-client
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathschemas.py
125 lines (96 loc) · 4.49 KB
/
schemas.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
__all__ = ["Class", "Schemas", "parse_reference_path", "update_schemas_with", "_ReferencePath"]
from typing import TYPE_CHECKING, Dict, Generic, List, NewType, Optional, Set, TypeVar, Union, cast
from urllib.parse import urlparse
import attr
from ... import Config
from ... import schema as oai
from ... import utils
from ..errors import ParseError, PropertyError, RecursiveReferenceInterupt
if TYPE_CHECKING: # pragma: no cover
from .property import Property
else:
Property = "Property"
T = TypeVar("T")
_ReferencePath = NewType("_ReferencePath", str)
_ClassName = NewType("_ClassName", str)
def parse_reference_path(ref_path_raw: str) -> Union[_ReferencePath, ParseError]:
parsed = urlparse(ref_path_raw)
if parsed.scheme or parsed.path:
return ParseError(detail=f"Remote references such as {ref_path_raw} are not supported yet.")
return cast(_ReferencePath, parsed.fragment)
@attr.s(auto_attribs=True)
class _Holder(Generic[T]):
data: Optional[T]
@attr.s(auto_attribs=True, frozen=True)
class Class:
"""Represents Python class which will be generated from an OpenAPI schema"""
name: _ClassName
module_name: str
@staticmethod
def from_string(*, string: str, config: Config) -> "Class":
"""Get a Class from an arbitrary string"""
class_name = string.split("/")[-1] # Get rid of ref path stuff
class_name = utils.pascal_case(class_name)
override = config.class_overrides.get(class_name)
if override is not None and override.class_name is not None:
class_name = override.class_name
if override is not None and override.module_name is not None:
module_name = override.module_name
else:
module_name = utils.snake_case(class_name)
return Class(name=cast(_ClassName, class_name), module_name=module_name)
@attr.s(auto_attribs=True, frozen=True)
class Schemas:
"""Structure for containing all defined, shareable, and reusable schemas (attr classes and Enums)"""
classes_by_reference: Dict[_ReferencePath, _Holder[Union[Property, RecursiveReferenceInterupt]]] = attr.ib(
factory=dict
)
classes_by_name: Dict[_ClassName, _Holder[Union[Property, RecursiveReferenceInterupt]]] = attr.ib(factory=dict)
errors: List[ParseError] = attr.ib(factory=list)
def update_schemas_with(
*,
ref_path: _ReferencePath,
data: Union[oai.Reference, oai.Schema],
schemas: Schemas,
visited: Set[_ReferencePath],
config: Config,
) -> Union[Schemas, PropertyError]:
if isinstance(data, oai.Reference):
return _update_schemas_with_reference(
ref_path=ref_path, data=data, schemas=schemas, visited=visited, config=config
)
else:
return _update_schemas_with_data(ref_path=ref_path, data=data, schemas=schemas, visited=visited, config=config)
def _update_schemas_with_reference(
*, ref_path: _ReferencePath, data: oai.Reference, schemas: Schemas, visited: Set[_ReferencePath], config: Config
) -> Union[Schemas, PropertyError]:
reference_pointer = parse_reference_path(data.ref)
if isinstance(reference_pointer, ParseError):
return PropertyError(detail=reference_pointer.detail, data=data)
resolved_reference = schemas.classes_by_reference.get(reference_pointer)
if resolved_reference:
return attr.evolve(schemas, classes_by_reference={ref_path: resolved_reference, **schemas.classes_by_reference})
else:
return PropertyError(f"Reference {ref_path} could not be resolved", data=data)
def _update_schemas_with_data(
*, ref_path: _ReferencePath, data: oai.Schema, schemas: Schemas, visited: Set[_ReferencePath], config: Config
) -> Union[Schemas, PropertyError]:
from . import property_from_data
prop: Union[PropertyError, Property]
prop, schemas = property_from_data(
data=data, name=ref_path, schemas=schemas, required=True, parent_name="", config=config
)
holder = schemas.classes_by_reference.get(ref_path)
if isinstance(prop, PropertyError):
if ref_path in visited and not holder:
holder = _Holder(data=RecursiveReferenceInterupt())
schemas = attr.evolve(schemas, classes_by_reference={ref_path: holder, **schemas.classes_by_reference})
return RecursiveReferenceInterupt(schemas=schemas)
return prop
if holder:
holder.data = prop
else:
schemas = attr.evolve(
schemas, classes_by_reference={ref_path: _Holder(data=prop), **schemas.classes_by_reference}
)
return schemas