Skip to content

Commit dd9ad5a

Browse files
barrybarretteBarry Barrette
and
Barry Barrette
authored
Adding support for named integer enums (#1214)
Adding support for named enums via an optional extension, `x-enum-varnames`. This extension is added to the schema inline with the `enum` definition: ``` "Siva.e_Install_State": { "enum": [ 0, 1, 2, 3, 4, 5, 6, 99 ], "type": "integer", "format": "int32", "x-enum-varnames": [ "Deinstalled", "Installed", "Upcoming_Site", "Lab_Site", "Pending_Deinstall", "Suspended", "Install_In_Progress", "Unknown" ] }, ``` The result: ![image](https://github.com/user-attachments/assets/780880b3-2f1f-49be-823b-f9abb713a3e1) --------- Co-authored-by: Barry Barrette <[email protected]>
1 parent d681971 commit dd9ad5a

File tree

4 files changed

+113
-3
lines changed

4 files changed

+113
-3
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
---
2+
default: minor
3+
---
4+
5+
# Adding support for named integer enums
6+
7+
#1214 by @barrybarrette
8+
9+
Adding support for named integer enums via an optional extension, `x-enum-varnames`.
10+
11+
This extension is added to the schema inline with the `enum` definition:
12+
```
13+
"MyEnum": {
14+
"enum": [
15+
0,
16+
1,
17+
2,
18+
3,
19+
4,
20+
5,
21+
6,
22+
99
23+
],
24+
"type": "integer",
25+
"format": "int32",
26+
"x-enum-varnames": [
27+
"Deinstalled",
28+
"Installed",
29+
"Upcoming_Site",
30+
"Lab_Site",
31+
"Pending_Deinstall",
32+
"Suspended",
33+
"Install_In_Progress",
34+
"Unknown"
35+
]
36+
}
37+
```
38+
39+
The result:
40+
![image](https://github.com/user-attachments/assets/780880b3-2f1f-49be-823b-f9abb713a3e1)

README.md

+33
Original file line numberDiff line numberDiff line change
@@ -192,6 +192,39 @@ content_type_overrides:
192192
application/zip: application/octet-stream
193193
```
194194

195+
## Supported Extensions
196+
197+
### x-enum-varnames
198+
199+
This extension has been adopted by similar projects such as [OpenAPI Tools](https://github.com/OpenAPITools/openapi-generator/pull/917).
200+
It is intended to provide user-friendly names for integer Enum members that get generated.
201+
It is critical that the length of the array matches that of the enum values.
202+
203+
```
204+
"Colors": {
205+
"type": "integer",
206+
"format": "int32",
207+
"enum": [
208+
0,
209+
1,
210+
2
211+
],
212+
"x-enum-varnames": [
213+
"Red",
214+
"Green",
215+
"Blue"
216+
]
217+
}
218+
```
219+
220+
Results in:
221+
```
222+
class Color(IntEnum):
223+
RED = 0
224+
GREEN = 1
225+
BLUE = 2
226+
```
227+
195228
[changelog.md]: CHANGELOG.md
196229
[poetry]: https://python-poetry.org/
197230
[PDM]: https://pdm-project.org/latest/

end_to_end_tests/functional_tests/generated_code_execution/test_enums_and_consts.py

+29
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,35 @@ def test_invalid_values(self, MyModel):
129129
MyModel.from_dict({"enumProp": "a"})
130130

131131

132+
@with_generated_client_fixture(
133+
"""
134+
components:
135+
schemas:
136+
MyEnum:
137+
type: integer
138+
enum: [2, 3, -4]
139+
x-enum-varnames: [
140+
"Two",
141+
"Three",
142+
"Negative Four"
143+
]
144+
""")
145+
@with_generated_code_imports(
146+
".models.MyEnum",
147+
)
148+
class TestIntEnumVarNameExtensions:
149+
@pytest.mark.parametrize(
150+
"expected_name,expected_value",
151+
[
152+
("TWO", 2),
153+
("THREE", 3),
154+
("NEGATIVE_FOUR", -4),
155+
],
156+
)
157+
def test_enum_values(self, MyEnum, expected_name, expected_value):
158+
assert getattr(MyEnum, expected_name) == MyEnum(expected_value)
159+
160+
132161
@with_generated_client_fixture(
133162
"""
134163
components:

openapi_python_client/parser/properties/enum_property.py

+11-3
Original file line numberDiff line numberDiff line change
@@ -121,7 +121,8 @@ def build( # noqa: PLR0911
121121
if parent_name:
122122
class_name = f"{utils.pascal_case(parent_name)}{utils.pascal_case(class_name)}"
123123
class_info = Class.from_string(string=class_name, config=config)
124-
values = EnumProperty.values_from_list(value_list, class_info)
124+
var_names = data.model_extra.get("x-enum-varnames", []) if data.model_extra else []
125+
values = EnumProperty.values_from_list(value_list, class_info, var_names)
125126

126127
if class_info.name in schemas.classes_by_name:
127128
existing = schemas.classes_by_name[class_info.name]
@@ -183,14 +184,21 @@ def get_imports(self, *, prefix: str) -> set[str]:
183184
return imports
184185

185186
@staticmethod
186-
def values_from_list(values: list[str] | list[int], class_info: Class) -> dict[str, ValueType]:
187+
def values_from_list(
188+
values: list[str] | list[int], class_info: Class, var_names: list[str]
189+
) -> dict[str, ValueType]:
187190
"""Convert a list of values into dict of {name: value}, where value can sometimes be None"""
188191
output: dict[str, ValueType] = {}
192+
use_var_names = len(var_names) == len(values)
189193

190194
for i, value in enumerate(values):
191195
value = cast(Union[str, int], value)
192196
if isinstance(value, int):
193-
if value < 0:
197+
if use_var_names:
198+
key = var_names[i]
199+
sanitized_key = utils.snake_case(key).upper()
200+
output[sanitized_key] = value
201+
elif value < 0:
194202
output[f"VALUE_NEGATIVE_{-value}"] = value
195203
else:
196204
output[f"VALUE_{value}"] = value

0 commit comments

Comments
 (0)