Skip to content
This repository has been archived by the owner on Nov 21, 2024. It is now read-only.

Commit

Permalink
Merge pull request #628 from cisco-open/fr/optional-response-validation
Browse files Browse the repository at this point in the history
allow disabling of response validation against pydantic model
  • Loading branch information
jpkrajewski authored May 9, 2024
2 parents f8e64e8 + ee2352c commit f481748
Show file tree
Hide file tree
Showing 5 changed files with 70 additions and 11 deletions.
8 changes: 6 additions & 2 deletions catalystwan/abstractions.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,10 @@ def text(self) -> str:
def content(self) -> bytes:
...

def dataobj(self, cls: Type[T], sourcekey: Optional[str]) -> T:
def dataobj(self, cls: Type[T], sourcekey: Optional[str], validate: bool) -> T:
...

def dataseq(self, cls: Type[T], sourcekey: Optional[str]) -> DataSequence[T]:
def dataseq(self, cls: Type[T], sourcekey: Optional[str], validate: bool) -> DataSequence[T]:
...

def json(self) -> dict:
Expand All @@ -52,3 +52,7 @@ def api_version(self) -> Optional[Version]:
@property
def session_type(self) -> Optional[SessionType]:
...

@property
def validate_responses(self) -> bool:
...
10 changes: 8 additions & 2 deletions catalystwan/endpoints/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -565,9 +565,15 @@ def wrapper(*args, **kwargs):
pass
elif issubclass(self.return_spec.payload_type, BaseModel):
if self.return_spec.sequence_type == DataSequence:
return response.dataseq(self.return_spec.payload_type, self.resp_json_key)
return response.dataseq(
cls=self.return_spec.payload_type,
sourcekey=self.resp_json_key,
validate=_self._client.validate_responses,
)
else:
return response.dataobj(self.return_spec.payload_type, self.resp_json_key)
return response.dataobj(
self.return_spec.payload_type, self.resp_json_key, validate=_self._client.validate_responses
)
elif issubclass(self.return_spec.payload_type, str):
return response.text
elif issubclass(self.return_spec.payload_type, bytes):
Expand Down
12 changes: 8 additions & 4 deletions catalystwan/response.py
Original file line number Diff line number Diff line change
Expand Up @@ -174,7 +174,7 @@ def info(self, history: bool = False) -> str:
return response_history_debug(self, None)
return response_debug(self, None)

def dataseq(self, cls: Type[T], sourcekey: Optional[str] = "data") -> DataSequence[T]:
def dataseq(self, cls: Type[T], sourcekey: Optional[str] = "data", validate: bool = True) -> DataSequence[T]:
"""Returns data contents from JSON payload parsed as DataSequence of Dataclass/BaseModel instances
Args:
cls: Dataclass/BaseModel subtype (eg. Devices)
Expand All @@ -195,10 +195,12 @@ def dataseq(self, cls: Type[T], sourcekey: Optional[str] = "data") -> DataSequen
sequence = [cast(dict, data)]

if issubclass(cls, BaseModel):
return DataSequence(cls, [cls.model_validate(item) for item in sequence]) # type: ignore
if validate:
return DataSequence(cls, [cls.model_validate(item) for item in sequence]) # type: ignore
return DataSequence(cls, [cls.model_construct(**item) for item in sequence]) # type: ignore
return DataSequence(cls, [create_dataclass(cls, item) for item in sequence])

def dataobj(self, cls: Type[T], sourcekey: Optional[str] = "data") -> T:
def dataobj(self, cls: Type[T], sourcekey: Optional[str] = "data", validate: bool = True) -> T:
"""Returns data contents from JSON payload parsed as Dataclass/BaseModel instance
Args:
cls: Dataclass/BaseModel subtype (eg. Devices)
Expand All @@ -214,7 +216,9 @@ def dataobj(self, cls: Type[T], sourcekey: Optional[str] = "data") -> T:
data = self.payload.json.get(sourcekey)

if issubclass(cls, BaseModel):
return cls.model_validate(data) # type: ignore[return-value]
if validate:
return cls.model_validate(data) # type: ignore[return-value]
return cls.model_construct(**data) # type: ignore[return-value]
return create_dataclass(cls, data)

def get_error_info(self) -> ManagerErrorInfo:
Expand Down
14 changes: 12 additions & 2 deletions catalystwan/session.py
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,7 @@ def __init__(
port: Optional[int] = None,
subdomain: Optional[str] = None,
auth: Optional[AuthBase] = None,
validate_responses: bool = True,
):
self.url = url
self.port = port
Expand All @@ -177,6 +178,7 @@ def __init__(
self._state: ManagerSessionState = ManagerSessionState.OPERATIVE
self.restart_timeout: int = 1200
self.polling_requests_timeout: int = 10
self._validate_responses = validate_responses

@property
def state(self) -> ManagerSessionState:
Expand Down Expand Up @@ -236,7 +238,7 @@ def login(self) -> ManagerSession:
try:
server_info = self.server()
except DefaultPasswordError:
server_info = ServerInfo.parse_obj({})
server_info = ServerInfo.model_construct(**{})

self.server_name = server_info.server

Expand Down Expand Up @@ -404,7 +406,7 @@ def get_tenant_id(self) -> str:
Returns:
Tenant UUID.
"""
tenants = self.get("dataservice/tenant").dataseq(Tenant)
tenants = self.get("dataservice/tenant").dataseq(Tenant, validate=False)
tenant = tenants.filter(subdomain=self.subdomain).single_or_default()

if not tenant or not tenant.tenant_id:
Expand Down Expand Up @@ -480,6 +482,14 @@ def platform_version(self, version: str):
def api_version(self) -> Version:
return self._api_version

@property
def validate_responses(self) -> bool:
return self._validate_responses

@validate_responses.setter
def validate_responses(self, value: bool):
self._validate_responses = value

def __str__(self) -> str:
return f"{self.username}@{self.base_url}"

Expand Down
37 changes: 36 additions & 1 deletion catalystwan/tests/test_response.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@

from attr import define, field # type: ignore
from parameterized import parameterized # type: ignore
from pydantic import BaseModel, Field
from pydantic import BaseModel, Field, ValidationError

from catalystwan.dataclasses import DataclassBase
from catalystwan.response import ManagerErrorInfo, ManagerResponse
Expand All @@ -26,6 +26,11 @@ class ParsedDataTypePydanticV2(BaseModel):
key3: Optional[float] = Field(default=None)


class DataForValidateTest(BaseModel):
important: str
not_important: ParsedDataTypePydanticV2


PARSE_DATASEQ_TEST_DATA: List = [
(True, None, 0, "data"),
(True, "string", 0, "data"),
Expand Down Expand Up @@ -56,6 +61,11 @@ class ParsedDataTypePydanticV2(BaseModel):
(True, {"data": [{"key1": "string", "key2": 66}, {"key1": "required", "key2": 18, "key3": 0.1}]}, "data"),
]

VALIDATE_DATASEQ_TEST_DATA = [
{"important": "correct1", "not_important": {"invalid-key1": "string", "key2": "not int"}},
{"important": "correct2", "not_important": {}},
]


class TestResponse(unittest.TestCase):
@patch("requests.Response")
Expand Down Expand Up @@ -136,3 +146,28 @@ def test_get_error(self, empty_error: bool, json: Any):
assert error_info.message is None
assert error_info.details is None
assert error_info.code is None

def test_dataobj_optional_validate(self):
# Arrange
self.response_mock.json.return_value = VALIDATE_DATASEQ_TEST_DATA[0]
vmng_response = ManagerResponse(self.response_mock)
# Act
data = vmng_response.dataobj(DataForValidateTest, sourcekey=None, validate=False)
# Assert
assert isinstance(data, DataForValidateTest)
assert data.important == VALIDATE_DATASEQ_TEST_DATA[0]["important"]
with self.assertRaises(ValidationError):
vmng_response.dataobj(DataForValidateTest, sourcekey=None, validate=True)

def test_dataseq_optional_validate(self):
self.response_mock.json.return_value = VALIDATE_DATASEQ_TEST_DATA
vmng_response = ManagerResponse(self.response_mock)
# Act
dataseq = vmng_response.dataseq(DataForValidateTest, sourcekey=None, validate=False)
# Assert
assert isinstance(dataseq, DataSequence)
for i, data in enumerate(dataseq):
assert isinstance(data, DataForValidateTest)
assert data.important == VALIDATE_DATASEQ_TEST_DATA[i]["important"]
with self.assertRaises(ValidationError):
vmng_response.dataseq(DataForValidateTest, sourcekey=None, validate=True)

0 comments on commit f481748

Please sign in to comment.