|
2 | 2 | import os
|
3 | 3 | import shutil
|
4 | 4 | from collections.abc import Generator
|
5 |
| -from dataclasses import dataclass |
| 5 | +from dataclasses import dataclass, make_dataclass |
6 | 6 | from pathlib import Path
|
7 |
| -from typing import Any |
| 7 | +from typing import Annotated, Any, cast |
8 | 8 |
|
9 | 9 | import pytest
|
| 10 | +from _pytest.fixtures import SubRequest |
10 | 11 | from pytest import FixtureRequest
|
11 | 12 | from pytest_mock import MockerFixture
|
12 | 13 |
|
| 14 | +import openapi_test_client.libraries.api.api_functions.utils.param_type as param_type_util |
13 | 15 | from openapi_test_client import ENV_VAR_PACKAGE_DIR
|
14 | 16 | from openapi_test_client.clients.base import OpenAPIClient
|
15 | 17 | from openapi_test_client.clients.demo_app import DemoAppAPIClient
|
16 | 18 | from openapi_test_client.clients.demo_app.api.auth import AuthAPI
|
17 |
| -from openapi_test_client.libraries.api.api_client_generator import ( |
18 |
| - setup_external_directory, |
19 |
| -) |
| 19 | +from openapi_test_client.libraries.api.api_client_generator import setup_external_directory |
| 20 | +from openapi_test_client.libraries.api.api_functions.utils.pydantic_model import PARAM_FORMAT_AND_TYPE_MAP |
20 | 21 | from openapi_test_client.libraries.api.api_spec import OpenAPISpec
|
21 |
| -from openapi_test_client.libraries.api.types import ParamModel, Unset |
| 22 | +from openapi_test_client.libraries.api.types import File, Format, Optional, ParamModel, Unset |
22 | 23 | from tests.unit import helper
|
23 | 24 |
|
24 | 25 |
|
@@ -96,3 +97,74 @@ class Model(ParamModel):
|
96 | 97 | inner_param2: str = Unset
|
97 | 98 |
|
98 | 99 | return Model
|
| 100 | + |
| 101 | + |
| 102 | +@pytest.fixture |
| 103 | +def NewParamModel(request: SubRequest) -> type[ParamModel]: |
| 104 | + """A new dataclass param model generated with requested field data via indirect parametrization |
| 105 | +
|
| 106 | + The fixture can be take the field data in various shapes as follows: |
| 107 | + - Just one field: |
| 108 | + - Only field type (field name and the default value will be automatically set) |
| 109 | + - As tuple (field name, field type) or (field name, field type, default value) |
| 110 | + - Multiple fields: List of above |
| 111 | + """ |
| 112 | + if not hasattr(request, "param"): |
| 113 | + raise ValueError(f"{NewParamModel.__name__} fixture must be used as an indirect parametrization") |
| 114 | + |
| 115 | + def add_field(field_data: Any | tuple[str, Any] | tuple[str, Any, Any], idx: int = 1) -> None: |
| 116 | + if isinstance(field_data, tuple): |
| 117 | + assert len(field_data) <= 3, f"Invalid field: {field_data}. Each field must be given as 2 or 3 items" |
| 118 | + if len(field_data) == 1: |
| 119 | + fields.append((f"field_{idx}", field_data, Unset)) |
| 120 | + elif len(field_data) >= 2: |
| 121 | + fields.append(field_data) |
| 122 | + else: |
| 123 | + fields.append((f"field{idx}", field_data, Unset)) |
| 124 | + |
| 125 | + requested_field_data = request.param |
| 126 | + fields: list[Any | tuple[str, Any] | tuple[str, Any, Any]] = [] |
| 127 | + if isinstance(requested_field_data, list): |
| 128 | + for i, requested_field in enumerate(requested_field_data, start=1): |
| 129 | + add_field(requested_field, idx=i) |
| 130 | + else: |
| 131 | + add_field(requested_field_data) |
| 132 | + |
| 133 | + param_model = cast(type[ParamModel], make_dataclass("Model", fields, bases=(ParamModel,))) |
| 134 | + return param_model |
| 135 | + |
| 136 | + |
| 137 | +@pytest.fixture(scope="session") |
| 138 | +def ParamModelWithParamFormats() -> type[ParamModel]: |
| 139 | + """A a dataclass param model that has fields with various param formats we support""" |
| 140 | + fields = [ |
| 141 | + ("uuid", Optional[Annotated[str, Format("uuid")]], Unset), |
| 142 | + ("date_time", Optional[Annotated[str, Format("date-time")]], Unset), |
| 143 | + ("date", Optional[Annotated[str, Format("date")]], Unset), |
| 144 | + ("time", Optional[Annotated[str, Format("time")]], Unset), |
| 145 | + ("duration", Optional[Annotated[str, Format("duration")]], Unset), |
| 146 | + ("binary", Optional[Annotated[str, Format("binary")]], Unset), |
| 147 | + ("file", Optional[Annotated[File, Format("binary")]], Unset), |
| 148 | + ("byte", Optional[Annotated[str, Format("byte")]], Unset), |
| 149 | + ("path", Optional[Annotated[str, Format("path")]], Unset), |
| 150 | + ("base64", Optional[Annotated[str, Format("base64")]], Unset), |
| 151 | + ("base64url", Optional[Annotated[str, Format("base64url")]], Unset), |
| 152 | + ("email", Optional[Annotated[str, Format("email")]], Unset), |
| 153 | + ("name_email", Optional[Annotated[str, Format("name-email")]], Unset), |
| 154 | + ("uri", Optional[Annotated[str, Format("uri")]], Unset), |
| 155 | + ("ipv4", Optional[Annotated[str, Format("ipv4")]], Unset), |
| 156 | + ("ipv6", Optional[Annotated[str, Format("ipv6")]], Unset), |
| 157 | + ("ipvanyaddress", Optional[Annotated[str, Format("ipvanyaddress")]], Unset), |
| 158 | + ("ipvanyinterface", Optional[Annotated[str, Format("ipvanyinterface")]], Unset), |
| 159 | + ("ipvanynetwork", Optional[Annotated[str, Format("ipvanynetwork")]], Unset), |
| 160 | + ("phone", Optional[Annotated[str, Format("phone")]], Unset), |
| 161 | + ] |
| 162 | + param_model = cast(type[ParamModel], make_dataclass("Model", fields, bases=(ParamModel,))) |
| 163 | + |
| 164 | + # Make sure the model covers all Pydantic specific types we support |
| 165 | + annotated_types = [param_type_util.get_annotated_type(t) for _, t, _ in fields] |
| 166 | + param_formats = [x.__metadata__[0].value for x in annotated_types] |
| 167 | + undefined_formats = set(PARAM_FORMAT_AND_TYPE_MAP.keys()).difference(set(param_formats)) |
| 168 | + assert not undefined_formats, f"Missing test coverage for these formats: {undefined_formats}" |
| 169 | + |
| 170 | + return param_model |
0 commit comments