""" Tests that should be applied to all interfaces """ import json from importlib.metadata import version import dask.array as da import numpy as np import pytest from pydantic import BaseModel from zarr.core import Array as ZarrArray from numpydantic.interface import Interface, InterfaceMark, MarkedJson from numpydantic.testing.helpers import ValidationCase def _test_roundtrip(source: BaseModel, target: BaseModel): """Test model equality for roundtrip tests""" assert type(target.array) is type(source.array) if isinstance(source.array, (np.ndarray, ZarrArray)): assert np.array_equal(target.array, np.array(source.array)) elif isinstance(source.array, da.Array): if target.array.dtype == object: # object equality doesn't really work well with dask # just check that the types match target_type = type(target.array.ravel()[0].compute()) source_type = type(source.array.ravel()[0].compute()) assert target_type is source_type else: assert np.all(da.equal(target.array, source.array)) else: assert target.array == source.array assert target.array.dtype == source.array.dtype def test_dunder_len(interface_cases, tmp_output_dir_func): """ Each interface or proxy type should support __len__ """ case = ValidationCase(interface=interface_cases) if interface_cases.interface.name == "video": case.shape = (10, 10, 2, 3) case.dtype = np.uint8 case.annotation_dtype = np.uint8 case.annotation_shape = (10, 10, "*", 3) array = case.array(path=tmp_output_dir_func) instance = case.model(array=array) assert len(instance.array) == case.shape[0] def test_interface_revalidate(all_passing_cases_instance): """ An interface should revalidate with the output of its initial validation See: https://github.com/p2p-ld/numpydantic/pull/14 """ _ = type(all_passing_cases_instance)(array=all_passing_cases_instance.array) @pytest.mark.json_schema def test_interface_jsonschema(all_passing_cases_instance): """ All interfaces should be able to generate json schema for all combinations of dtype and shape Note that this does not test for json schema correctness - see ndarray tests for that """ _ = all_passing_cases_instance.model_json_schema() @pytest.mark.xfail def test_interface_rematch(interface_cases, tmp_output_dir_func): """ All interfaces should match the results of the object they return after validation """ array = interface_cases.make_array(path=tmp_output_dir_func) assert ( Interface.match(interface_cases.interface.validate(array)) is interface_cases.interface ) def test_interface_to_numpy_array(dtype_by_interface_instance): """ All interfaces should be able to have the output of their validation stage coerced to a numpy array with np.array() """ _ = np.array(dtype_by_interface_instance.array) @pytest.mark.serialization def test_interface_dump_json(dtype_by_interface_instance): """ All interfaces should be able to dump to json """ dtype_by_interface_instance.model_dump_json() @pytest.mark.serialization def test_interface_roundtrip_json(dtype_by_interface, tmp_output_dir_func): """ All interfaces should be able to roundtrip to and from json """ if "subclass" in dtype_by_interface.id.lower(): pytest.xfail() array = dtype_by_interface.array(path=tmp_output_dir_func) case = dtype_by_interface.model(array=array) dumped_json = case.model_dump_json(round_trip=True) model = case.model_validate_json(dumped_json) _test_roundtrip(case, model) @pytest.mark.serialization @pytest.mark.parametrize("an_interface", Interface.interfaces()) def test_interface_mark_interface(an_interface): """ All interfaces should be able to mark the current version and interface info """ mark = an_interface.mark_interface() assert isinstance(mark, InterfaceMark) assert mark.name == an_interface.name assert mark.cls == an_interface.__name__ assert mark.module == an_interface.__module__ assert mark.version == version(mark.module.split(".")[0]) @pytest.mark.serialization @pytest.mark.parametrize("valid", [True, False]) @pytest.mark.filterwarnings("ignore:Mismatch between serialized mark") def test_interface_mark_roundtrip(dtype_by_interface, valid, tmp_output_dir_func): """ All interfaces should be able to roundtrip with the marked interface, and a mismatch should raise a warning and attempt to proceed """ if "subclass" in dtype_by_interface.id.lower(): pytest.xfail() array = dtype_by_interface.array(path=tmp_output_dir_func) case = dtype_by_interface.model(array=array) dumped_json = case.model_dump_json( round_trip=True, context={"mark_interface": True} ) data = json.loads(dumped_json) # ensure that we are a MarkedJson _ = MarkedJson.model_validate_json(json.dumps(data["array"])) if not valid: # ruin the version data["array"]["interface"]["version"] = "v99999999" dumped_json = json.dumps(data) with pytest.warns(match="Mismatch.*"): model = case.model_validate_json(dumped_json) else: model = case.model_validate_json(dumped_json) _test_roundtrip(case, model)