Skip to content

Commit 4e5e6e5

Browse files
committed
Change the style of property names for API class accessors
1 parent 8b28d8b commit 4e5e6e5

File tree

14 files changed

+57
-57
lines changed

14 files changed

+57
-57
lines changed

README.md

+20-20
Original file line numberDiff line numberDiff line change
@@ -174,12 +174,12 @@ Alternatively, you can instantiate your client directly from the parent `OpenAPI
174174

175175
### Make an API request
176176

177-
To make an API request with your API client, call an API function as `client.<API_TAG>.<api_function_name>()`.
177+
To make an API request with your API client, call an API function as `client.<ApiTag>.<api_function_name>()`.
178178
The function will take all parameters documented in the OpenAPI specs as keyword arguments.
179179

180180
eg. To call the login API defined under the Auth tag:
181181
```pycon
182-
>>> r = client.AUTH.login(username='foo', password='bar')
182+
>>> r = client.Auth.login(username='foo', password='bar')
183183
2024-01-01T00:00:00.863-0800 - request: POST http://127.0.0.1:5000/v1/auth/login
184184
2024-01-01T00:00:00.877-0800 - response: 201 (Created)
185185
- request_id: a2b20acf-22d5-4131-ac0d-6796bf19d2af
@@ -330,11 +330,11 @@ class DemoAppAPIClient(OpenAPIClient):
330330
super().__init__("demo_app", env=env, doc="openapi.json")
331331

332332
@cached_property
333-
def AUTH(self):
333+
def Auth(self):
334334
return AuthAPI(self)
335335

336336
@cached_property
337-
def USERS(self):
337+
def Users(self):
338338
return UsersAPI(self)
339339
```
340340

@@ -441,10 +441,10 @@ parameters and/or with raw `requests` library options (eg. `timeout`, `headers`,
441441
Some attributes available from the API class:
442442
```pycon
443443
>>> # Get tag data
444-
>>> client.AUTH.TAGs
444+
>>> client.Auth.TAGs
445445
('Auth',)
446446
>>> # Get available endpoints under this API class
447-
>>> pprint(client.AUTH.endpoints)
447+
>>> pprint(client.Auth.endpoints)
448448
[Endpoint(tags=('Auth',),
449449
api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
450450
method='post',
@@ -492,11 +492,11 @@ DELETE /v1/users/{user_id}
492492

493493
Each API class function is decorated with a `@endpoint.<method>(<path>)` endpoint decorator. This decorator converts the original function into an instance of the `EndpointHandler` class at runtime. The `EndpointHandler` object acts as a proxy to a per-endpoint `EndpointFunc` object, which is also created at runtime and is responsible for handling the actual API calls via its base class's `__call__()` method. Additionally, the `EndpointFunc` object provides various capabilities and attributes related to the endpoint.
494494

495-
eg. The Login API is accessible via `client.AUTH.login()` API function, which is actually an instance of
495+
eg. The Login API is accessible via `client.Auth.login()` API function, which is actually an instance of
496496
`AuthAPILoginEndpointFunc` class returned by its associated `EndpointHandler` obj.
497497

498498
```pycon
499-
>>> client.AUTH.login
499+
>>> client.Auth.login
500500
<openapi_test_client.libraries.api.api_functions.endpoint.AuthAPILoginEndpointFunc object at 0x1074abf10>
501501
(mapped to: <function AuthAPI.login at 0x10751c360>)
502502
```
@@ -511,9 +511,9 @@ The endpoint function is also accessible directly from the API class:
511511

512512
Various endpoint data is available from the endpoint function via `endpoint` property:
513513
```pycon
514-
>>> print(client.AUTH.login.endpoint)
514+
>>> print(client.Auth.login.endpoint)
515515
POST /v1/auth/login
516-
>>> pprint(client.AUTH.login.endpoint)
516+
>>> pprint(client.Auth.login.endpoint)
517517
Endpoint(tags=('Auth',),
518518
api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
519519
method='post',
@@ -525,18 +525,18 @@ Endpoint(tags=('Auth',),
525525
is_public=True,
526526
is_documented=True,
527527
is_deprecated=False)
528-
>>> client.AUTH.login.endpoint.method
528+
>>> client.Auth.login.endpoint.method
529529
'post'
530-
>>> client.AUTH.login.endpoint.path
530+
>>> client.Auth.login.endpoint.path
531531
'/v1/auth/login'
532-
>>> client.AUTH.login.endpoint.url
532+
>>> client.Auth.login.endpoint.url
533533
'http://127.0.0.1:5000/v1/auth/login'
534534
```
535535

536536
Note that the same endpoint data is also available directly from the API class, except for `url` will always be `None`.
537537
```pycon
538538
>>> from openapi_test_client.clients.demo_app.api.auth import AuthAPI
539-
>>> client.AUTH.login.endpoint == AuthAPI.login.endpoint
539+
>>> client.Auth.login.endpoint == AuthAPI.login.endpoint
540540
True
541541
>>> print(AuthAPI.login.endpoint)
542542
POST /v1/auth/login
@@ -557,7 +557,7 @@ Endpoint(tags=('Auth',),
557557
An example of the additional capability the `EndpointFunc` obj provides - Automatic retry:
558558
```pycon
559559
# Call the endpiont with the automatic retry (you can specify a retry condition if needed)
560-
>>> r = client.AUTH.login.with_retry(username='foo', password='bar')
560+
>>> r = client.Auth.login.with_retry(username='foo', password='bar')
561561
2024-01-01T00:00:00.153-0000 - request: POST http://127.0.0.1:5000/v1/auth/login
562562
2024-01-01T00:00:00.158-0000 - response: 429 (Too Many Requests)
563563
- request_id: 1b028ff7-0880-430c-b5a3-12aa057892cf
@@ -590,7 +590,7 @@ An example of the additional capability the `EndpointFunc` obj provides - Automa
590590
Each endpoint is represented as an `EndpointModel` dataclass model, which holds various context around each
591591
parameter (eg. type annotation).
592592
```pycon
593-
>>> model = client.AUTH.login.endpoint.model
593+
>>> model = client.Auth.login.endpoint.model
594594
>>> print(model)
595595
<class 'types.AuthAPILoginEndpointModel'>
596596
>>> pprint(model.__dataclass_fields__, sort_dicts=False)
@@ -774,7 +774,7 @@ Here are some comparisons between regular models and pydantic models:
774774
- Regular dataclass model (Validation will be done on the server-side)
775775
```pycon
776776
>>> # Model definition
777-
>>> model = client.USERS.create_user.endpoint.model
777+
>>> model = client.Users.create_user.endpoint.model
778778
>>> print(model)
779779
<class 'types.UsersAPICreateUserEndpointModel'>
780780
>>> pprint(model.__dataclass_fields__, sort_dicts=False)
@@ -785,7 +785,7 @@ Here are some comparisons between regular models and pydantic models:
785785
'metadata': Field(name='metadata',type=typing.Optional[openapi_test_client.clients.demo_app.models.users.Metadata],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD)}
786786
>>>
787787
>>> # Make an API request with the invalid parameter values
788-
>>> r = client.USERS.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123)
788+
>>> r = client.Users.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123)
789789
2024-01-01T00:00:00.741-0800 - The request contains one or more parameters UsersAPI.create_user() does not expect:
790790
- extra
791791
2024-01-01T00:00:00.742-0800 - request: POST http://127.0.0.1:5000/v1/users
@@ -874,7 +874,7 @@ Here are some comparisons between regular models and pydantic models:
874874
- Pydantic model (Validation will be done on the client-side)
875875
```pycon
876876
>>> # Model definition
877-
>>> pydantic_model = client.USERS.create_user.endpoint.model.to_pydantic()
877+
>>> pydantic_model = client.Users.create_user.endpoint.model.to_pydantic()
878878
>>> print(pydantic_model)
879879
<class 'types.UsersAPICreateUserEndpointModel'>
880880
>>> pprint(pydantic_model.model_fields, sort_dicts=False)
@@ -885,7 +885,7 @@ Here are some comparisons between regular models and pydantic models:
885885
'metadata': FieldInfo(annotation=Union[Metadata, NoneType], required=False, default=None)}
886886
>>>
887887
>>> # Make an API request with the same invalid parmeter values, but with validate=True option
888-
>>> r = client.USERS.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123, validate=True)
888+
>>> r = client.Users.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123, validate=True)
889889
2024-01-01T00:00:00.830-0800 - The request contains one or more parameters UsersAPI.create_user() does not expect:
890890
- extra
891891
Traceback (most recent call last):

images/generate.gif

-2.03 KB
Loading

src/openapi_test_client/clients/demo_app/api/base/demo_app_api.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,5 +29,5 @@ def post_request_hook(
2929
):
3030
super().post_request_hook(endpoint, response, request_exception, *path_params, **params)
3131
if response and response.ok:
32-
if endpoint in self.api_client.AUTH.endpoints:
32+
if endpoint in self.api_client.Auth.endpoints:
3333
manage_auth_session(self.api_client, endpoint, response)

src/openapi_test_client/clients/demo_app/api/request_hooks/post_request.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -41,8 +41,8 @@ def manage_auth_session(api_client: DemoAppAPIClient, endpoint: Endpoint, r: Res
4141
:param endpoint: The Endpoint object of the API endpoint
4242
:param r: RestResponse object returned from the request
4343
"""
44-
if endpoint == api_client.AUTH.login.endpoint:
44+
if endpoint == api_client.Auth.login.endpoint:
4545
token = r.response["token"]
4646
api_client.rest_client.set_bearer_token(token)
47-
elif endpoint == api_client.AUTH.logout.endpoint:
47+
elif endpoint == api_client.Auth.logout.endpoint:
4848
api_client.rest_client.unset_bear_token()

src/openapi_test_client/clients/demo_app/demo_app_client.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ class DemoAppAPIClient(OpenAPIClient):
1111
1212
Usage:
1313
>>> client = DemoAppAPIClient()
14-
>>> r = client.AUTH.login(username="foo", password="bar")
14+
>>> r = client.Auth.login(username="foo", password="bar")
1515
>>> assert r.status_code == 200
1616
>>> token = r.response["token"]
1717
"""
@@ -20,9 +20,9 @@ def __init__(self, env: str = "dev"):
2020
super().__init__("demo_app", env=env, doc="openapi.json")
2121

2222
@cached_property
23-
def AUTH(self):
23+
def Auth(self):
2424
return AuthAPI(self)
2525

2626
@cached_property
27-
def USERS(self):
27+
def Users(self):
2828
return UsersAPI(self)

src/openapi_test_client/libraries/api/api_classes/__init__.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -22,12 +22,12 @@ def init_api_classes(base_api_class: type[APIClassType]) -> list[type[APIClassTy
2222
Note: This function must be called from the __init__.py of a directory that contains API class files
2323
2424
Example:
25-
`AuthAPI.endpoints` or `<DemoAppAPIClient>.AUTH.endpoints` will return the following `Endpoint` objects
25+
`AuthAPI.endpoints` or `<DemoAppAPIClient>.Auth.endpoints` will return the following `Endpoint` objects
2626
2727
>>> from openapi_test_client.clients.demo_app import DemoAppAPIClient
2828
>>>
2929
>>> client = DemoAppAPIClient()
30-
>>> client.AUTH.endpoints
30+
>>> client.Auth.endpoints
3131
[
3232
Endpoint(tag='Auth', api_class=<class 'test_client.clients.demo_app.api.auth.AuthAPI'>, method='post', path='/v1/auth/login', func_name='login', model=<class 'types.LoginEndpointModel'>),
3333
Endpoint(tag='Auth', api_class=<class 'test_client.clients.demo_app.api.auth.AuthAPI'>, method='get', path='/v1/auth/logout', func_name='logout', model=<class 'types.LogoutEndpointModel'>)

src/openapi_test_client/libraries/api/api_client_generator.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -621,7 +621,7 @@ def generate_api_client(temp_api_client: OpenAPIClient, show_generated_code: boo
621621
):
622622
mod = inspect.getmodule(api_class)
623623
imports_code += f"from .{API_CLASS_DIR_NAME}.{Path(mod.__file__).stem} import {api_class.__name__}\n"
624-
property_name = camel_to_snake(api_class.__name__.removesuffix("API")).upper()
624+
property_name = api_class.__name__.removesuffix("API")
625625
api_client_code += (
626626
f"{TAB}@cached_property\n{TAB}def {property_name}(self):\n{TAB}{TAB}return {api_class.__name__}(self)\n\n"
627627
)

src/openapi_test_client/libraries/api/api_functions/endpoints.py

+8-8
Original file line numberDiff line numberDiff line change
@@ -82,9 +82,9 @@ def __call__(
8282
>>> from openapi_test_client.clients.demo_app import DemoAppAPIClient
8383
>>>
8484
>>> client = DemoAppAPIClient()
85-
>>> r = client.AUTH.login(username="foo", password="bar")
85+
>>> r = client.Auth.login(username="foo", password="bar")
8686
>>> # Above API call can be also done directly from the endpoint object, if you need to:
87-
>>> endpoint = client.AUTH.login.endpoint
87+
>>> endpoint = client.Auth.login.endpoint
8888
>>> r2 = endpoint(client, username="foo", password="bar")
8989
"""
9090
endpoint_func: EndpointFunc = getattr(self.api_class(api_client), self.func_name)
@@ -109,23 +109,23 @@ class endpoint:
109109
>>> ...
110110
>>>
111111
>>> client = DemoAppAPIClient()
112-
>>> type(client.AUTH.login)
112+
>>> type(client.Auth.login)
113113
<class 'openapi_test_client.libraries.api.api_functions.endpoint.AuthAPILoginEndpointFunc'>
114114
>>> type(AuthAPI.login)
115115
<class 'openapi_test_client.libraries.api.api_functions.endpoint.AuthAPILoginEndpointFunc'>
116-
>>> isinstance(client.AUTH.login, EndpointFunc) and isinstance(AuthAPI.login, EndpointFunc)
116+
>>> isinstance(client.Auth.login, EndpointFunc) and isinstance(AuthAPI.login, EndpointFunc)
117117
True
118-
>>> client.AUTH.login.endpoint
118+
>>> client.Auth.login.endpoint
119119
Endpoint(tags=('Auth',), api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>, method='post', path='/v1/auth/login', func_name='login', model=<class 'types.AuthAPILoginEndpointModel'>, url='http://127.0.0.1:5000/v1/auth/login', content_type=None, is_public=False, is_documented=True, is_deprecated=False)
120120
>>> AuthAPI.login.endpoint
121121
Endpoint(tags=('Auth',), api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>, method='post', path='/v1/auth/login', func_name='login', model=<class 'types.AuthAPILoginEndpointModel'>, url=None, content_type=None, is_public=False, is_documented=True, is_deprecated=False)
122-
>>> str(client.AUTH.login.endpoint)
122+
>>> str(client.Auth.login.endpoint)
123123
'POST /v1/auth/login'
124124
>>> str(AuthAPI.login.endpoint)
125125
'POST /v1/auth/login'
126-
>>> client.AUTH.login.endpoint.path
126+
>>> client.Auth.login.endpoint.path
127127
'/v1/auth/login'
128-
>>> client.AUTH.login.endpoint.url
128+
>>> client.Auth.login.endpoint.url
129129
'http://127.0.0.1:5000/v1/auth/login'
130130
131131
""" # noqa: E501

tests/integration/conftest.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -65,10 +65,10 @@ def unauthenticated_api_client() -> DemoAppAPIClient:
6565
@pytest.fixture(scope="session")
6666
def api_client() -> DemoAppAPIClient:
6767
client = DemoAppAPIClient()
68-
r = client.AUTH.login(username="foo", password="bar")
68+
r = client.Auth.login(username="foo", password="bar")
6969
assert r.ok
7070
yield client
71-
client.AUTH.logout()
71+
client.Auth.logout()
7272

7373

7474
@pytest.fixture

tests/integration/test_api_auth.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,11 @@
77
@pytest.mark.parametrize("validation_mode", [False, True])
88
def test_user_login_logout(unauthenticated_api_client: DemoAppAPIClient, validation_mode: bool):
99
"""Check basic client/server functionality of Auth login/logout APIs"""
10-
r = unauthenticated_api_client.AUTH.login(username="foo", password="bar", validate=validation_mode)
10+
r = unauthenticated_api_client.Auth.login(username="foo", password="bar", validate=validation_mode)
1111
assert r.ok
1212
assert set(r.response.keys()) == {"token"}
1313

14-
r = unauthenticated_api_client.AUTH.logout()
14+
r = unauthenticated_api_client.Auth.logout()
1515
assert r.ok
1616
assert r.response["message"] == "logged out"
1717

@@ -25,7 +25,7 @@ def test_user_login_with_invalid_params(unauthenticated_api_client: DemoAppAPICl
2525
- password: missing required parameter
2626
"""
2727
helper.do_test_invalid_params(
28-
endpoint_func=unauthenticated_api_client.AUTH.login,
28+
endpoint_func=unauthenticated_api_client.Auth.login,
2929
validation_mode=validation_mode,
3030
invalid_params=dict(username=123),
3131
num_expected_errors=2,

tests/integration/test_api_users.py

+8-8
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
@pytest.mark.parametrize("validation_mode", [False, True])
1111
def test_create_user(api_client: DemoAppAPIClient, validation_mode: bool):
1212
"""Check basic client/server functionality of create user API"""
13-
r = api_client.USERS.create_user(
13+
r = api_client.Users.create_user(
1414
first_name="test",
1515
last_name="test",
1616
@@ -29,7 +29,7 @@ def test_create_user(api_client: DemoAppAPIClient, validation_mode: bool):
2929
def test_get_user(api_client: DemoAppAPIClient, validation_mode: bool):
3030
"""Check basic client/server functionality of get user API"""
3131
user_id = 5
32-
r = api_client.USERS.get_user(user_id, validate=validation_mode)
32+
r = api_client.Users.get_user(user_id, validate=validation_mode)
3333
assert r.status_code == 200
3434
assert r.response["id"] == user_id
3535

@@ -38,7 +38,7 @@ def test_get_user(api_client: DemoAppAPIClient, validation_mode: bool):
3838
def test_get_users(api_client: DemoAppAPIClient, validation_mode: bool):
3939
"""Check basic client/server functionality of get users API"""
4040
role = "support"
41-
r = api_client.USERS.get_users(role=role, validate=validation_mode)
41+
r = api_client.Users.get_users(role=role, validate=validation_mode)
4242
assert r.status_code == 200
4343
assert len(r.response) == len([x for x in USERS if x.role.value == role])
4444

@@ -53,7 +53,7 @@ def test_upload_image(api_client: DemoAppAPIClient, validation_mode: bool):
5353
b"\x01c`\x00\x00\x00\x02\x00\x01su\x01\x18\x00\x00\x00\x00IEND\xaeB`\x82"
5454
)
5555
file = File(filename="test_image.png", content=image_data, content_type="image/png")
56-
r = api_client.USERS.upload_image(file=file, description="test image", validate=validation_mode)
56+
r = api_client.Users.upload_image(file=file, description="test image", validate=validation_mode)
5757
assert r.status_code == 201
5858
assert r.response["message"] == f"Image '{file.filename}' uploaded"
5959

@@ -62,7 +62,7 @@ def test_upload_image(api_client: DemoAppAPIClient, validation_mode: bool):
6262
def test_delete_user(api_client: DemoAppAPIClient, validation_mode: bool):
6363
"""Check basic client/server functionality of delete user API"""
6464
user_id = 1 + bool(validation_mode)
65-
r = api_client.USERS.delete_user(user_id, validate=validation_mode)
65+
r = api_client.Users.delete_user(user_id, validate=validation_mode)
6666
assert r.status_code == 200
6767
assert r.response["message"] == f"Deleted user {user_id}"
6868

@@ -82,7 +82,7 @@ def test_create_user_with_invalid_params(api_client: DemoAppAPIClient, validatio
8282
8383
"""
8484
helper.do_test_invalid_params(
85-
endpoint_func=api_client.USERS.create_user,
85+
endpoint_func=api_client.Users.create_user,
8686
validation_mode=validation_mode,
8787
invalid_params=dict(
8888
first_name=123,
@@ -106,7 +106,7 @@ def test_get_users_with_invalid_params(api_client: DemoAppAPIClient, validation_
106106
- role: Invalid enum value
107107
"""
108108
helper.do_test_invalid_params(
109-
endpoint_func=api_client.USERS.get_users,
109+
endpoint_func=api_client.Users.get_users,
110110
validation_mode=validation_mode,
111111
invalid_params=dict(id="test", role="test"),
112112
num_expected_errors=2,
@@ -126,7 +126,7 @@ def test_upload_image_with_invalid_params(api_client: DemoAppAPIClient, validati
126126
the content is properly uploaded with the multipart/form-data Content-Type header
127127
"""
128128
helper.do_test_invalid_params(
129-
endpoint_func=api_client.USERS.upload_image,
129+
endpoint_func=api_client.Users.upload_image,
130130
validation_mode=validation_mode,
131131
invalid_params=dict(file="test"),
132132
num_expected_errors=1,

tests/integration/test_script.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ def test_update_client(temp_app_client: OpenAPIClient, dry_run: bool, option: st
9191
NOTE: temp_app_client is a temporary client generated for this test against the demo_app app.
9292
API class and model file code should be identical to DemoAppAPIClient's, except for API function names
9393
"""
94-
users_api = getattr(temp_app_client, "USERS")
94+
users_api = getattr(temp_app_client, "Users")
9595
assert users_api._unnamed_endpoint_1.endpoint == UsersAPI.create_user.endpoint
9696
users_api_class_file = Path(inspect.getabsfile(type(users_api)))
9797
assert users_api_class_file.exists()

tests/unit/conftest.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ def api_client() -> DemoAppAPIClient:
3131
def api_class_or_instance(request: FixtureRequest, api_client: DemoAppAPIClient) -> AuthAPI | type[AuthAPI]:
3232
"""Parametrize fixture that returns the demo API client's AuthAPI class or an isntance of the class"""
3333
if request.param == "instance":
34-
return api_client.AUTH
34+
return api_client.Auth
3535
else:
3636
return AuthAPI
3737

0 commit comments

Comments
 (0)