diff --git a/apps/schematic/api/.gitignore b/apps/schematic/api/.gitignore index c78610579f..a78f16f2cf 100644 --- a/apps/schematic/api/.gitignore +++ b/apps/schematic/api/.gitignore @@ -71,6 +71,8 @@ target/ *secrets* synapse_config.yaml schematic_service_account_creds.json +private_localhost_certificate.crt +private_localhost.key #schematic downloaded files manifests diff --git a/apps/schematic/api/.openapi-generator/FILES b/apps/schematic/api/.openapi-generator/FILES index 2e57b51219..67091db1d9 100644 --- a/apps/schematic/api/.openapi-generator/FILES +++ b/apps/schematic/api/.openapi-generator/FILES @@ -13,6 +13,9 @@ schematic_api/models/__init__.py schematic_api/models/asset_type.py schematic_api/models/base_model_.py schematic_api/models/basic_error.py +schematic_api/models/component_requirement_array.py +schematic_api/models/component_requirement_graph.py +schematic_api/models/component_requirement_subgraph.py schematic_api/models/connected_node_pair.py schematic_api/models/connected_node_pair_array.py schematic_api/models/connected_node_pair_page.py diff --git a/apps/schematic/api/schematic_api/controllers/manifest_generation_controller.py b/apps/schematic/api/schematic_api/controllers/manifest_generation_controller.py index 578f94e16a..f6b76007af 100644 --- a/apps/schematic/api/schematic_api/controllers/manifest_generation_controller.py +++ b/apps/schematic/api/schematic_api/controllers/manifest_generation_controller.py @@ -17,8 +17,8 @@ def generate_google_sheet_manifests( manifest_title=None, data_type_array=None, display_label_type=None, - asset_view_id=None, use_strict_validation=None, + asset_view_id=None, generate_all_manifests=None, ): # noqa: E501 """Generates a list of google sheet links @@ -37,10 +37,10 @@ def generate_google_sheet_manifests( :type data_type_array: List[str] :param display_label_type: The type of label to display :type display_label_type: str - :param asset_view_id: ID of view listing all project data assets. E.g. for Synapse this would be the Synapse ID of the fileview listing all data assets for a given project - :type asset_view_id: str :param use_strict_validation: If true, users are blocked from entering incorrect values. If false, users will get a warning when using incorrect values. :type use_strict_validation: bool + :param asset_view_id: ID of view listing all project data assets. E.g. for Synapse this would be the Synapse ID of the fileview listing all data assets for a given project + :type asset_view_id: str :param generate_all_manifests: If true, a manifest for all components will be generated, datasetIds will be ignored. If false, manifests for each id in datasetIds will be generated. :type generate_all_manifests: bool @@ -53,7 +53,7 @@ def generate_google_sheet_manifests( manifest_title, data_type_array, display_label_type, - asset_view_id, use_strict_validation, + asset_view_id, generate_all_manifests, ) diff --git a/apps/schematic/api/schematic_api/controllers/manifest_generation_controller_impl.py b/apps/schematic/api/schematic_api/controllers/manifest_generation_controller_impl.py index 2b93f64fb0..fdd6c7bd02 100644 --- a/apps/schematic/api/schematic_api/controllers/manifest_generation_controller_impl.py +++ b/apps/schematic/api/schematic_api/controllers/manifest_generation_controller_impl.py @@ -11,7 +11,6 @@ handle_exceptions, get_access_token, download_schema_file_as_jsonld, - InvalidValueError, ) @@ -22,10 +21,10 @@ def generate_google_sheet_manifests( dataset_id_array: list[str] | None, manifest_title: str | None, data_type_array: list[str] | None, - display_label_type: DisplayLabelType = "class_label", - asset_view_id: str | None = None, - use_strict_validation: bool = True, - generate_all_manifests: bool = False, + display_label_type: DisplayLabelType, + use_strict_validation: bool, + asset_view_id: str | None, + generate_all_manifests: bool, ) -> tuple[GoogleSheetLinks | BasicError, int]: """Generates a list of links to manifests in google sheet form @@ -57,38 +56,9 @@ def generate_google_sheet_manifests( """ if generate_all_manifests: - if dataset_id_array: - raise InvalidValueError( - "When generate_all_manifests is True dataset_id_array must be None", - {"dataset_id_array": dataset_id_array}, - ) - if data_type_array: - raise InvalidValueError( - "When generate_all_manifests is True data_type_array must be None", - {"data_type_array": data_type_array}, - ) data_type_array = ["all manifests"] - - else: - if not data_type_array: - raise InvalidValueError( - ( - "When generate_all_manifests is False data_type_array must be a list with " - "at least one item" - ), - {"data_type_array": data_type_array}, - ) - if dataset_id_array and len(dataset_id_array) != len(data_type_array): - raise InvalidValueError( - ( - "When generate_all_manifests is False data_type_array and dataset_id_array " - "must both lists with the same length" - ), - { - "data_type_array": data_type_array, - "dataset_id_array": dataset_id_array, - }, - ) + if not data_type_array: + data_type_array = [] access_token = get_access_token() if asset_view_id: diff --git a/apps/schematic/api/schematic_api/controllers/schema_controller.py b/apps/schematic/api/schematic_api/controllers/schema_controller.py index 80fbed68bd..9cc701b0d9 100644 --- a/apps/schematic/api/schematic_api/controllers/schema_controller.py +++ b/apps/schematic/api/schematic_api/controllers/schema_controller.py @@ -5,6 +5,12 @@ from typing import Union from schematic_api.models.basic_error import BasicError # noqa: E501 +from schematic_api.models.component_requirement_array import ( + ComponentRequirementArray, +) # noqa: E501 +from schematic_api.models.component_requirement_graph import ( + ComponentRequirementGraph, +) # noqa: E501 from schematic_api.models.connected_node_pair_array import ( ConnectedNodePairArray, ) # noqa: E501 @@ -42,6 +48,48 @@ def get_component( ) +def get_component_requirements_array( + component_label, schema_url, display_label_type=None +): # noqa: E501 + """Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. + + Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. # noqa: E501 + + :param component_label: The label of a component in a schema + :type component_label: str + :param schema_url: The URL of a schema in jsonld or csv form + :type schema_url: str + :param display_label_type: The type of label to display + :type display_label_type: str + + :rtype: Union[ComponentRequirementArray, Tuple[ComponentRequirementArray, int], Tuple[ComponentRequirementArray, int, Dict[str, str]] + """ + return schema_controller_impl.get_component_requirements_array( + component_label, schema_url, display_label_type + ) + + +def get_component_requirements_graph( + component_label, schema_url, display_label_type=None +): # noqa: E501 + """Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. + + Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. # noqa: E501 + + :param component_label: The label of a component in a schema + :type component_label: str + :param schema_url: The URL of a schema in jsonld or csv form + :type schema_url: str + :param display_label_type: The type of label to display + :type display_label_type: str + + :rtype: Union[ComponentRequirementGraph, Tuple[ComponentRequirementGraph, int], Tuple[ComponentRequirementGraph, int, Dict[str, str]] + """ + return schema_controller_impl.get_component_requirements_graph( + component_label, schema_url, display_label_type + ) + + def get_connected_node_pair_array( schema_url, relationship_type, display_label_type=None ): # noqa: E501 diff --git a/apps/schematic/api/schematic_api/controllers/schema_controller_impl.py b/apps/schematic/api/schematic_api/controllers/schema_controller_impl.py index 72e8699b6b..d028a4b31c 100644 --- a/apps/schematic/api/schematic_api/controllers/schema_controller_impl.py +++ b/apps/schematic/api/schematic_api/controllers/schema_controller_impl.py @@ -9,8 +9,12 @@ from schematic.visualization.attributes_explorer import AttributesExplorer # type: ignore from schematic.utils.schema_utils import get_property_label_from_display_name # type: ignore from schematic.utils.schema_utils import DisplayLabelType # type: ignore +from schematic.models.metadata import MetadataModel # type: ignore from schematic_api.models.basic_error import BasicError +from schematic_api.models.component_requirement_subgraph import ( + ComponentRequirementSubgraph, +) from schematic_api.models.node_property_array import NodePropertyArray from schematic_api.models.validation_rule import ValidationRule from schematic_api.models.validation_rule_array import ValidationRuleArray @@ -62,6 +66,72 @@ def get_component( return result, status +@handle_exceptions +def get_component_requirements_array( + component_label: str, + schema_url: str, + display_label_type: DisplayLabelType, +) -> tuple[Union[list[str], BasicError], int]: + """Gets the input components required components + + Args: + component_label (str): The label of the component + schema_url (str): The URL of the schema in json form + display_label_type (DisplayLabelType): + The type of label to use as display + Defaults to "class_label" + Returns: + tuple[Union[ComponentRequirementArray, BasicError], int]: A tuple + item 1 is either the required coponents or an error + item 2 is the status + """ + schema_path = download_schema_file_as_jsonld(schema_url) + metadata_model = MetadataModel( + inputMModelLocation=schema_path, + inputMModelLocationType="local", + data_model_labels=display_label_type, + ) + result = metadata_model.get_component_requirements( + source_component=component_label, as_graph=False + ) + status = 200 + return result, status + + +@handle_exceptions +def get_component_requirements_graph( + component_label: str, + schema_url: str, + display_label_type: DisplayLabelType, +) -> tuple[Union[list[ComponentRequirementSubgraph], BasicError], int]: + """Gets the input components required components + + Args: + component_label (str): The label of the component + schema_url (str): The URL of the schema in json form + display_label_type (DisplayLabelType): + The type of label to use as display + Defaults to "class_label" + Returns: + tuple[Union[ComponentRequirementGrpah, BasicError], int]: A tuple + item 1 is either the required coponents or an error + item 2 is the status + """ + schema_path = download_schema_file_as_jsonld(schema_url) + metadata_model = MetadataModel( + inputMModelLocation=schema_path, + inputMModelLocationType="local", + data_model_labels=display_label_type, + ) + results = metadata_model.get_component_requirements( + source_component=component_label, as_graph=True + ) + result = [ComponentRequirementSubgraph(result[0], result[1]) for result in results] + + status = 200 + return result, status + + def get_connected_node_pairs_from_schematic( relationship_type: str, schema_url: str, diff --git a/apps/schematic/api/schematic_api/models/__init__.py b/apps/schematic/api/schematic_api/models/__init__.py index 20d7f19caf..bc2c4edd3d 100644 --- a/apps/schematic/api/schematic_api/models/__init__.py +++ b/apps/schematic/api/schematic_api/models/__init__.py @@ -5,6 +5,9 @@ # import models into model package from schematic_api.models.asset_type import AssetType from schematic_api.models.basic_error import BasicError +from schematic_api.models.component_requirement_array import ComponentRequirementArray +from schematic_api.models.component_requirement_graph import ComponentRequirementGraph +from schematic_api.models.component_requirement_subgraph import ComponentRequirementSubgraph from schematic_api.models.connected_node_pair import ConnectedNodePair from schematic_api.models.connected_node_pair_array import ConnectedNodePairArray from schematic_api.models.connected_node_pair_page import ConnectedNodePairPage diff --git a/apps/schematic/api/schematic_api/models/component_requirement_array.py b/apps/schematic/api/schematic_api/models/component_requirement_array.py new file mode 100644 index 0000000000..d3100f376e --- /dev/null +++ b/apps/schematic/api/schematic_api/models/component_requirement_array.py @@ -0,0 +1,64 @@ +# coding: utf-8 + +from __future__ import absolute_import +from datetime import date, datetime # noqa: F401 + +from typing import List, Dict # noqa: F401 + +from schematic_api.models.base_model_ import Model +from schematic_api import util + + +class ComponentRequirementArray(Model): + """NOTE: This class is auto generated by OpenAPI Generator (https://openapi-generator.tech). + + Do not edit the class manually. + """ + + def __init__(self, component_requirements_list=None): # noqa: E501 + """ComponentRequirementArray - a model defined in OpenAPI + + :param component_requirements_list: The component_requirements_list of this ComponentRequirementArray. # noqa: E501 + :type component_requirements_list: List[str] + """ + self.openapi_types = { + 'component_requirements_list': List[str] + } + + self.attribute_map = { + 'component_requirements_list': 'componentRequirementsList' + } + + self._component_requirements_list = component_requirements_list + + @classmethod + def from_dict(cls, dikt) -> 'ComponentRequirementArray': + """Returns the dict as a model + + :param dikt: A dict. + :type: dict + :return: The ComponentRequirementArray of this ComponentRequirementArray. # noqa: E501 + :rtype: ComponentRequirementArray + """ + return util.deserialize_model(dikt, cls) + + @property + def component_requirements_list(self): + """Gets the component_requirements_list of this ComponentRequirementArray. + + + :return: The component_requirements_list of this ComponentRequirementArray. + :rtype: List[str] + """ + return self._component_requirements_list + + @component_requirements_list.setter + def component_requirements_list(self, component_requirements_list): + """Sets the component_requirements_list of this ComponentRequirementArray. + + + :param component_requirements_list: The component_requirements_list of this ComponentRequirementArray. + :type component_requirements_list: List[str] + """ + + self._component_requirements_list = component_requirements_list diff --git a/apps/schematic/api/schematic_api/models/component_requirement_graph.py b/apps/schematic/api/schematic_api/models/component_requirement_graph.py new file mode 100644 index 0000000000..a70a73f2c9 --- /dev/null +++ b/apps/schematic/api/schematic_api/models/component_requirement_graph.py @@ -0,0 +1,66 @@ +# coding: utf-8 + +from __future__ import absolute_import +from datetime import date, datetime # noqa: F401 + +from typing import List, Dict # noqa: F401 + +from schematic_api.models.base_model_ import Model +from schematic_api.models.component_requirement_subgraph import ComponentRequirementSubgraph +from schematic_api import util + +from schematic_api.models.component_requirement_subgraph import ComponentRequirementSubgraph # noqa: E501 + +class ComponentRequirementGraph(Model): + """NOTE: This class is auto generated by OpenAPI Generator (https://openapi-generator.tech). + + Do not edit the class manually. + """ + + def __init__(self, component_requirements_graph=None): # noqa: E501 + """ComponentRequirementGraph - a model defined in OpenAPI + + :param component_requirements_graph: The component_requirements_graph of this ComponentRequirementGraph. # noqa: E501 + :type component_requirements_graph: List[ComponentRequirementSubgraph] + """ + self.openapi_types = { + 'component_requirements_graph': List[ComponentRequirementSubgraph] + } + + self.attribute_map = { + 'component_requirements_graph': 'componentRequirementsGraph' + } + + self._component_requirements_graph = component_requirements_graph + + @classmethod + def from_dict(cls, dikt) -> 'ComponentRequirementGraph': + """Returns the dict as a model + + :param dikt: A dict. + :type: dict + :return: The ComponentRequirementGraph of this ComponentRequirementGraph. # noqa: E501 + :rtype: ComponentRequirementGraph + """ + return util.deserialize_model(dikt, cls) + + @property + def component_requirements_graph(self): + """Gets the component_requirements_graph of this ComponentRequirementGraph. + + + :return: The component_requirements_graph of this ComponentRequirementGraph. + :rtype: List[ComponentRequirementSubgraph] + """ + return self._component_requirements_graph + + @component_requirements_graph.setter + def component_requirements_graph(self, component_requirements_graph): + """Sets the component_requirements_graph of this ComponentRequirementGraph. + + + :param component_requirements_graph: The component_requirements_graph of this ComponentRequirementGraph. + :type component_requirements_graph: List[ComponentRequirementSubgraph] + """ + + self._component_requirements_graph = component_requirements_graph diff --git a/apps/schematic/api/schematic_api/models/component_requirement_subgraph.py b/apps/schematic/api/schematic_api/models/component_requirement_subgraph.py new file mode 100644 index 0000000000..9abeee3540 --- /dev/null +++ b/apps/schematic/api/schematic_api/models/component_requirement_subgraph.py @@ -0,0 +1,98 @@ +# coding: utf-8 + +from __future__ import absolute_import +from datetime import date, datetime # noqa: F401 + +from typing import List, Dict # noqa: F401 + +from schematic_api.models.base_model_ import Model +from schematic_api import util + + +class ComponentRequirementSubgraph(Model): + """NOTE: This class is auto generated by OpenAPI Generator (https://openapi-generator.tech). + + Do not edit the class manually. + """ + + def __init__(self, component1=None, component2=None): # noqa: E501 + """ComponentRequirementSubgraph - a model defined in OpenAPI + + :param component1: The component1 of this ComponentRequirementSubgraph. # noqa: E501 + :type component1: str + :param component2: The component2 of this ComponentRequirementSubgraph. # noqa: E501 + :type component2: str + """ + self.openapi_types = { + 'component1': str, + 'component2': str + } + + self.attribute_map = { + 'component1': 'component1', + 'component2': 'component2' + } + + self._component1 = component1 + self._component2 = component2 + + @classmethod + def from_dict(cls, dikt) -> 'ComponentRequirementSubgraph': + """Returns the dict as a model + + :param dikt: A dict. + :type: dict + :return: The ComponentRequirementSubgraph of this ComponentRequirementSubgraph. # noqa: E501 + :rtype: ComponentRequirementSubgraph + """ + return util.deserialize_model(dikt, cls) + + @property + def component1(self): + """Gets the component1 of this ComponentRequirementSubgraph. + + The display name of the first component in the graph # noqa: E501 + + :return: The component1 of this ComponentRequirementSubgraph. + :rtype: str + """ + return self._component1 + + @component1.setter + def component1(self, component1): + """Sets the component1 of this ComponentRequirementSubgraph. + + The display name of the first component in the graph # noqa: E501 + + :param component1: The component1 of this ComponentRequirementSubgraph. + :type component1: str + """ + if component1 is None: + raise ValueError("Invalid value for `component1`, must not be `None`") # noqa: E501 + + self._component1 = component1 + + @property + def component2(self): + """Gets the component2 of this ComponentRequirementSubgraph. + + The display name of the second component in the graph # noqa: E501 + + :return: The component2 of this ComponentRequirementSubgraph. + :rtype: str + """ + return self._component2 + + @component2.setter + def component2(self, component2): + """Sets the component2 of this ComponentRequirementSubgraph. + + The display name of the second component in the graph # noqa: E501 + + :param component2: The component2 of this ComponentRequirementSubgraph. + :type component2: str + """ + if component2 is None: + raise ValueError("Invalid value for `component2`, must not be `None`") # noqa: E501 + + self._component2 = component2 diff --git a/apps/schematic/api/schematic_api/openapi/openapi.yaml b/apps/schematic/api/schematic_api/openapi/openapi.yaml index 1b172b056c..d40becdc49 100644 --- a/apps/schematic/api/schematic_api/openapi/openapi.yaml +++ b/apps/schematic/api/schematic_api/openapi/openapi.yaml @@ -1202,6 +1202,116 @@ paths: tags: - Schema x-openapi-router-controller: schematic_api.controllers.schema_controller + /components/{componentLabel}/requirementsArray: + get: + description: "Given a source model component (see https://w3id.org/biolink/vocab/category\ + \ for definnition of component), return all components required by it in array\ + \ form." + operationId: get_component_requirements_array + parameters: + - description: The label of a component in a schema + explode: false + in: path + name: componentLabel + required: true + schema: + $ref: '#/components/schemas/ComponentLabel' + style: simple + - description: The URL of a schema in jsonld or csv form + explode: true + in: query + name: schemaUrl + required: true + schema: + $ref: '#/components/schemas/SchemaUrl' + style: form + - description: The type of label to display + explode: true + in: query + name: displayLabelType + required: false + schema: + default: class_label + enum: + - class_label + - display_label + type: string + style: form + responses: + "200": + content: + application/json: + schema: + $ref: '#/components/schemas/ComponentRequirementArray' + description: Success + "500": + content: + application/problem+json: + schema: + $ref: '#/components/schemas/BasicError' + description: The request cannot be fulfilled due to an unexpected server + error + summary: "Given a source model component (see https://w3id.org/biolink/vocab/category\ + \ for definnition of component), return all components required by it in array\ + \ form." + tags: + - Schema + x-openapi-router-controller: schematic_api.controllers.schema_controller + /components/{componentLabel}/requirementsGraph: + get: + description: "Given a source model component (see https://w3id.org/biolink/vocab/category\ + \ for definnition of component), return all components required by it in graph\ + \ form." + operationId: get_component_requirements_graph + parameters: + - description: The label of a component in a schema + explode: false + in: path + name: componentLabel + required: true + schema: + $ref: '#/components/schemas/ComponentLabel' + style: simple + - description: The URL of a schema in jsonld or csv form + explode: true + in: query + name: schemaUrl + required: true + schema: + $ref: '#/components/schemas/SchemaUrl' + style: form + - description: The type of label to display + explode: true + in: query + name: displayLabelType + required: false + schema: + default: class_label + enum: + - class_label + - display_label + type: string + style: form + responses: + "200": + content: + application/json: + schema: + $ref: '#/components/schemas/ComponentRequirementGraph' + description: Success + "500": + content: + application/problem+json: + schema: + $ref: '#/components/schemas/BasicError' + description: The request cannot be fulfilled due to an unexpected server + error + summary: "Given a source model component (see https://w3id.org/biolink/vocab/category\ + \ for definnition of component), return all components required by it in graph\ + \ form." + tags: + - Schema + x-openapi-router-controller: schematic_api.controllers.schema_controller /connectedNodePairArray: get: description: Gets a array of connected node pairs @@ -1384,16 +1494,6 @@ paths: - display_label type: string style: form - - description: ID of view listing all project data assets. E.g. for Synapse - this would be the Synapse ID of the fileview listing all data assets for - a given project - explode: true - in: query - name: assetViewId - required: false - schema: - $ref: '#/components/schemas/AssetViewId' - style: form - description: "If true, users are blocked from entering incorrect values. If\ \ false, users will get a warning when using incorrect values." explode: true @@ -1404,6 +1504,16 @@ paths: default: true type: boolean style: form + - description: ID of view listing all project data assets. E.g. for Synapse + this would be the Synapse ID of the fileview listing all data assets for + a given project + explode: true + in: query + name: assetViewId + required: false + schema: + $ref: '#/components/schemas/AssetViewId' + style: form - description: "If true, a manifest for all components will be generated, datasetIds\ \ will be ignored. If false, manifests for each id in datasetIds will be\ \ generated." @@ -2823,6 +2933,17 @@ components: schema: $ref: '#/components/schemas/DataTypeArray' style: form + useStrictValidation: + description: "If true, users are blocked from entering incorrect values. If\ + \ false, users will get a warning when using incorrect values." + explode: true + in: query + name: useStrictValidation + required: false + schema: + default: true + type: boolean + style: form responses: InternalServerError: content: @@ -3302,6 +3423,59 @@ components: description: The label of a component in a schema example: MolecularEntity type: string + ComponentRequirementArray: + description: An array of components + example: + componentRequirementsList: + - componentRequirementsList + - componentRequirementsList + properties: + componentRequirementsList: + items: + type: string + title: componentRequirementsList + type: array + title: ComponentRequirementArray + type: object + ComponentRequirementSubgraph: + description: A pair of components + example: + component1: component1 + component2: component2 + properties: + component1: + description: The display name of the first component in the graph + example: component1 + title: component1 + type: string + component2: + description: The display name of the second component in the graph + example: component2 + title: component2 + type: string + required: + - component1 + - component2 + title: ComponentRequirementSubgraph + type: object + x-java-class-annotations: + - '@lombok.Builder' + ComponentRequirementGraph: + description: A graph of components + example: + componentRequirementsGraph: + - component1: component1 + component2: component2 + - component1: component1 + component2: component2 + properties: + componentRequirementsGraph: + items: + $ref: '#/components/schemas/ComponentRequirementSubgraph' + title: componentRequirementsGraph + type: array + title: ComponentRequirementGraph + type: object RelationshipType: description: A type of schema relationship example: requiresDependency @@ -3397,11 +3571,15 @@ components: items: $ref: '#/components/schemas/DatasetId' type: array + DataType: + description: A data type + example: Patient + title: DataType + type: string DataTypeArray: description: An array of data types items: - example: Patient - type: string + $ref: '#/components/schemas/DataType' type: array GoogleSheetLinks: description: An array of google sheet links diff --git a/apps/schematic/api/schematic_api/test/conftest.py b/apps/schematic/api/schematic_api/test/conftest.py index 992c70eb7b..41e254fd26 100644 --- a/apps/schematic/api/schematic_api/test/conftest.py +++ b/apps/schematic/api/schematic_api/test/conftest.py @@ -27,9 +27,6 @@ def csv_to_json_str(path: str) -> str: GET_ACCESS_TOKEN_MOCK = ( "schematic_api.controllers.manifest_generation_controller_impl.get_access_token" ) -CREATE_SINGLE_MANIFEST_MOCK = ( - "schematic.manifest.generator.ManifestGenerator.create_single_manifest" -) CREATE_MANIFESTS_MOCK = ( "schematic.manifest.generator.ManifestGenerator.create_manifests" ) diff --git a/apps/schematic/api/schematic_api/test/test_manifest_generation_controller_impl.py b/apps/schematic/api/schematic_api/test/test_manifest_generation_controller_impl.py index 91f8c5b7ef..1f596c3361 100644 --- a/apps/schematic/api/schematic_api/test/test_manifest_generation_controller_impl.py +++ b/apps/schematic/api/schematic_api/test/test_manifest_generation_controller_impl.py @@ -27,12 +27,13 @@ def test_success(self, test_schema_url: str) -> None: manifest_title="title", use_strict_validation=True, generate_all_manifests=False, + display_label_type="class_label", ) assert status == 200 assert isinstance(result, GoogleSheetLinks) assert result.links == ["link1", "link2"] - def test_error_statuses(self, test_schema_url: str) -> None: + def test_error_statuses(self) -> None: """Test for error statuses""" with patch(GET_ACCESS_TOKEN_MOCK): with patch(CREATE_MANIFESTS_MOCK): @@ -45,92 +46,7 @@ def test_error_statuses(self, test_schema_url: str) -> None: manifest_title="title", use_strict_validation=True, generate_all_manifests=False, + display_label_type="class_label", ) assert status == 404 assert isinstance(result, BasicError) - - result, status = generate_google_sheet_manifests( - schema_url=test_schema_url, - dataset_id_array=["syn2", "syn3"], - asset_view_id="syn1", - data_type_array=["syn4", "syn5"], - add_annotations=False, - manifest_title="title", - use_strict_validation=True, - generate_all_manifests=True, - ) - assert status == 422 - assert isinstance(result, BasicError) - assert result.detail == ( - "When generate_all_manifests is True dataset_id_array must be None: " - "{'dataset_id_array': ['syn2', 'syn3']}" - ) - - result, status = generate_google_sheet_manifests( - schema_url=test_schema_url, - dataset_id_array=None, - asset_view_id="syn1", - data_type_array=["syn4", "syn5"], - add_annotations=False, - manifest_title="title", - use_strict_validation=True, - generate_all_manifests=True, - ) - assert status == 422 - assert isinstance(result, BasicError) - assert result.detail == ( - "When generate_all_manifests is True data_type_array must be None: " - "{'data_type_array': ['syn4', 'syn5']}" - ) - - result, status = generate_google_sheet_manifests( - schema_url=test_schema_url, - dataset_id_array=None, - asset_view_id="syn1", - data_type_array=None, - add_annotations=False, - manifest_title="title", - use_strict_validation=True, - generate_all_manifests=False, - ) - assert status == 422 - assert isinstance(result, BasicError) - assert result.detail == ( - "When generate_all_manifests is False data_type_array must be a list with " - "at least one item: {'data_type_array': None}" - ) - - result, status = generate_google_sheet_manifests( - schema_url=test_schema_url, - dataset_id_array=None, - asset_view_id="syn1", - data_type_array=[], - add_annotations=False, - manifest_title="title", - use_strict_validation=True, - generate_all_manifests=False, - ) - assert status == 422 - assert isinstance(result, BasicError) - assert result.detail == ( - "When generate_all_manifests is False data_type_array must be a list with " - "at least one item: {'data_type_array': []}" - ) - - result, status = generate_google_sheet_manifests( - schema_url=test_schema_url, - dataset_id_array=["syn4"], - asset_view_id="syn1", - data_type_array=["syn2", "syn3"], - add_annotations=False, - manifest_title="title", - use_strict_validation=True, - generate_all_manifests=False, - ) - assert status == 422 - assert isinstance(result, BasicError) - assert result.detail == ( - "When generate_all_manifests is False data_type_array and dataset_id_array " - "must both lists with the same length: " - "{'data_type_array': ['syn2', 'syn3'], 'dataset_id_array': ['syn4']}" - ) diff --git a/apps/schematic/api/schematic_api/test/test_manifest_generation_endpoints.py b/apps/schematic/api/schematic_api/test/test_manifest_generation_endpoints.py index 82853fa35e..dc845a6ad4 100644 --- a/apps/schematic/api/schematic_api/test/test_manifest_generation_endpoints.py +++ b/apps/schematic/api/schematic_api/test/test_manifest_generation_endpoints.py @@ -19,18 +19,6 @@ "Authorization": "Bearer xxx", } -CREATE_MANIFESTS_ARGS = [ - "path_to_data_model", - "output_format", - "data_types", - "title", - "access_token", - "dataset_ids", - "strict", - "use_annotations", - "data_model_labels", -] - class TestGenerateGoogleSheetManifests(BaseTestCase): """Tests google sheet manifest endpoint""" @@ -53,13 +41,13 @@ def test_success(self) -> None: assert list(result.keys()) == ["links"] assert result["links"] == ["l1"] call_args = mock_method.call_args.kwargs - assert list(call_args.keys()) == CREATE_MANIFESTS_ARGS assert call_args["output_format"] == "google_sheet" assert call_args["data_types"] == ["node_label"] assert not call_args["title"] assert call_args["dataset_ids"] == ["syn2"] assert call_args["strict"] assert not call_args["use_annotations"] + assert call_args["data_model_labels"] == "class_label" def test_arguments(self) -> None: """Test for correct arguments""" @@ -73,6 +61,7 @@ def test_arguments(self) -> None: "&manifestTitle=title" "&useStrictValidation=false" "&addAnnotations=true" + "&displayLabelType=display_label" ) response = self.client.open(url, method="GET", headers=HEADERS) self.assert200( @@ -83,15 +72,15 @@ def test_arguments(self) -> None: assert list(result.keys()) == ["links"] assert result["links"] == ["l1"] call_args = mock_method.call_args.kwargs - assert list(call_args.keys()) == CREATE_MANIFESTS_ARGS assert call_args["output_format"] == "google_sheet" assert call_args["data_types"] == ["data_type1", "data_type2"] assert call_args["dataset_ids"] == ["syn2", "syn3"] assert call_args["title"] == "title" assert not call_args["strict"] assert call_args["use_annotations"] + assert call_args["data_model_labels"] == "display_label" - url = f"{GENERATE_GOOGLE_SHEET_MANIFESTS_URL}" "&generateAllManifests=true" + url = f"{GENERATE_GOOGLE_SHEET_MANIFESTS_URL}&generateAllManifests=true" response = self.client.open(url, method="GET", headers=HEADERS) self.assert200( response, f"Response body is : {response.data.decode('utf-8')}" @@ -101,10 +90,10 @@ def test_arguments(self) -> None: assert list(result.keys()) == ["links"] assert result["links"] == ["l1"] call_args = mock_method.call_args.kwargs - assert list(call_args.keys()) == CREATE_MANIFESTS_ARGS assert call_args["output_format"] == "google_sheet" assert call_args["data_types"] == ["all manifests"] assert not call_args["title"] assert not call_args["dataset_ids"] assert call_args["strict"] assert not call_args["use_annotations"] + assert call_args["data_model_labels"] == "class_label" diff --git a/apps/schematic/api/schematic_api/test/test_schema_controller_endpoints.py b/apps/schematic/api/schematic_api/test/test_schema_controller_endpoints.py index 1e517e6340..5172de6804 100644 --- a/apps/schematic/api/schematic_api/test/test_schema_controller_endpoints.py +++ b/apps/schematic/api/schematic_api/test/test_schema_controller_endpoints.py @@ -2,6 +2,7 @@ # pylint: disable=duplicate-code import unittest +from unittest.mock import patch, Mock, MagicMock from schematic_api.test import BaseTestCase from .conftest import TEST_SCHEMA_URL, PAGING_KEYS @@ -12,6 +13,12 @@ } COMPONENT_URL = "/api/v1/components/Patient/?schemaUrl=" +COMPONENT_REQUIREMENTS_ARRAY_URL = ( + "/api/v1/components/Biospecimen/requirementsArray?schemaUrl=" +) +COMPONENT_REQUIREMENTS_GRAPH_URL = ( + "/api/v1/components/Biospecimen/requirementsGraph?schemaUrl=" +) CONNECTED_NODE_PAIR_ARRAY_URL = "/api/v1/connectedNodePairArray?schemaUrl=" CONNECTED_NODE_PAIR_PAGE_URL = "/api/v1/connectedNodePairPage?schemaUrl=" NODE_IS_REQUIRED_URL = "/api/v1/nodes/FamilyHistory/isRequired?schemaUrl=" @@ -47,6 +54,228 @@ def test_404(self) -> None: self.assert404(response, f"Response body is : {response.data.decode('utf-8')}") +class TestGetComponentRequirementsArray(BaseTestCase): + """Test case for component requirements array endpoint""" + + def test_success(self) -> None: + """Test for successful result""" + url = f"{COMPONENT_REQUIREMENTS_ARRAY_URL}{TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + assert isinstance(response.json, list) + + @patch("schematic.models.metadata.MetadataModel.get_component_requirements") + def test_get_component_requirements_default_parameters( + self, mock_method: Mock + ) -> None: + """Tests that MetadataModel.get_component_requirements gets default parameters""" + mock_method.return_value = ["component"] + + url = f"{COMPONENT_REQUIREMENTS_ARRAY_URL}{TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + + mock_method.assert_called_once_with( + source_component="Biospecimen", as_graph=False + ) + + @patch("schematic.models.metadata.MetadataModel.get_component_requirements") + def test_get_component_requirements_parameters(self, mock_method: Mock) -> None: + """Tests that MetadataModel.get_component_requirements gets non default parameters""" + mock_method.return_value = ["component"] + + url = f"/api/v1/components/Component/requirementsArray?schemaUrl={TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + + mock_method.assert_called_once_with( + source_component="Component", as_graph=False + ) + + def test_metadata_model_default_parameters(self) -> None: + "Tests that MetadataModel is getting default parameters" + + # Creates to mocked object and method to be used + mock_object = MagicMock() + mock_object.get_component_requirements = Mock(return_value=["component"]) + + # Patches in the mocked MetadataModel object and method into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.MetadataModel", + return_value=mock_object, + autospec=True, + ) as patched_object: + # Patches the download_schema_file_as_jsonld function into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.download_schema_file_as_jsonld", + return_value="temp_file_path.csv", + autospec=True, + ) as patched_function: + url = f"{COMPONENT_REQUIREMENTS_ARRAY_URL}{TEST_SCHEMA_URL}" + self.client.open(url, method="GET", headers=HEADERS) + + # Asserts the patched object was called once with correct arguments and parameters + patched_object.assert_called_once() + call_args = patched_object.call_args.kwargs + assert call_args["inputMModelLocation"] == "temp_file_path.csv" + assert call_args["inputMModelLocationType"] == "local" + assert call_args["data_model_labels"] == "class_label" + + # Asserts the patched function was called once with correct arguments and parameters + patched_function.assert_called_once_with(schema_url=TEST_SCHEMA_URL) + + def test_metadata_model_parameters(self) -> None: + "Tests that MetadataModel is getting non default parameters" + + # Creates to mocked object and method to be used + mock_object = MagicMock() + mock_object.get_component_requirements = Mock(return_value=["component"]) + + # Patches in the mocked MetadataModel object and method into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.MetadataModel", + return_value=mock_object, + autospec=True, + ) as patched_object: + # Patches the download_schema_file_as_jsonld function into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.download_schema_file_as_jsonld", + return_value="temp_file_path.csv", + autospec=True, + ) as patched_function: + url = ( + f"{COMPONENT_REQUIREMENTS_ARRAY_URL}url.jsonld" + "&displayLabelType=display_label" + ) + self.client.open(url, method="GET", headers=HEADERS) + + # Asserts the pacthed object was called once with correct arguments and parameters + patched_object.assert_called_once() + call_args = patched_object.call_args.kwargs + assert call_args["inputMModelLocation"] + assert call_args["inputMModelLocationType"] == "local" + assert call_args["data_model_labels"] == "display_label" + + # Asserts the patched function was called once with correct arguments and parameters + patched_function.assert_called_once_with(schema_url="url.jsonld") + + def test_404(self) -> None: + """Test for 404 result""" + url = f"{COMPONENT_REQUIREMENTS_ARRAY_URL}not_a_url" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert404(response, f"Response body is : {response.data.decode('utf-8')}") + + +class TestGetComponentRequirementsGraph(BaseTestCase): + """Test case for component requirements graph endpoint""" + + def test_success(self) -> None: + """Test for successful result""" + url = f"{COMPONENT_REQUIREMENTS_GRAPH_URL}{TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + assert isinstance(response.json, list) + + @patch("schematic.models.metadata.MetadataModel.get_component_requirements") + def test_get_component_requirements_default_parameters( + self, mock_method: Mock + ) -> None: + """Tests that MetadataModel.get_component_requirements gets default parameters""" + mock_method.return_value = ["component"] + + url = f"{COMPONENT_REQUIREMENTS_GRAPH_URL}{TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + + mock_method.assert_called_once_with( + source_component="Biospecimen", as_graph=True + ) + + @patch("schematic.models.metadata.MetadataModel.get_component_requirements") + def test_get_component_requirements_parameters(self, mock_method: Mock) -> None: + """Tests that MetadataModel.get_component_requirements gets non default parameters""" + mock_method.return_value = ["component"] + + url = f"/api/v1/components/Component/requirementsGraph?schemaUrl={TEST_SCHEMA_URL}" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert200(response, f"Response body is : {response.data.decode('utf-8')}") + + mock_method.assert_called_once_with(source_component="Component", as_graph=True) + + def test_metadata_model_default_parameters(self) -> None: + "Tests that MetadataModel is getting default parameters" + + # Creates to mocked object and method to be used + mock_object = MagicMock() + mock_object.get_component_requirements = Mock(return_value=["component"]) + + # Patches in the mocked MetadataModel object and method into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.MetadataModel", + return_value=mock_object, + autospec=True, + ) as patched_object: + # Patches the download_schema_file_as_jsonld function into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.download_schema_file_as_jsonld", + return_value="temp_file_path.csv", + autospec=True, + ) as patched_function: + url = f"{COMPONENT_REQUIREMENTS_GRAPH_URL}{TEST_SCHEMA_URL}" + self.client.open(url, method="GET", headers=HEADERS) + + # Asserts the patched object was called once with correct arguments and parameters + patched_object.assert_called_once() + call_args = patched_object.call_args.kwargs + assert call_args["inputMModelLocation"] == "temp_file_path.csv" + assert call_args["inputMModelLocationType"] == "local" + assert call_args["data_model_labels"] == "class_label" + + # Asserts the patched function was called once with correct arguments and parameters + patched_function.assert_called_once_with(schema_url=TEST_SCHEMA_URL) + + def test_metadata_model_parameters(self) -> None: + "Tests that MetadataModel is getting non default parameters" + + # Creates to mocked object and method to be used + mock_object = MagicMock() + mock_object.get_component_requirements = Mock(return_value=["component"]) + + # Patches in the mocked MetadataModel object and method into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.MetadataModel", + return_value=mock_object, + autospec=True, + ) as patched_object: + # Patches the download_schema_file_as_jsonld function into the API call + with patch( + "schematic_api.controllers.schema_controller_impl.download_schema_file_as_jsonld", + return_value="temp_file_path.csv", + autospec=True, + ) as patched_function: + url = ( + f"{COMPONENT_REQUIREMENTS_GRAPH_URL}url.jsonld" + "&displayLabelType=display_label" + ) + self.client.open(url, method="GET", headers=HEADERS) + + # Asserts the pacthed object was called once with correct arguments and parameters + patched_object.assert_called_once() + call_args = patched_object.call_args.kwargs + assert call_args["inputMModelLocation"] + assert call_args["inputMModelLocationType"] == "local" + assert call_args["data_model_labels"] == "display_label" + + # Asserts the patched function was called once with correct arguments and parameters + patched_function.assert_called_once_with(schema_url="url.jsonld") + + def test_404(self) -> None: + """Test for 404 result""" + url = f"{COMPONENT_REQUIREMENTS_GRAPH_URL}not_a_url" + response = self.client.open(url, method="GET", headers=HEADERS) + self.assert404(response, f"Response body is : {response.data.decode('utf-8')}") + + class TestGetConnectedNodePairArray(BaseTestCase): """Tests for connected node pair array endpoint""" diff --git a/apps/schematic/api/schematic_api/test/test_schema_controller_impl.py b/apps/schematic/api/schematic_api/test/test_schema_controller_impl.py index 93e4e7cb2d..6d15cfaf8e 100644 --- a/apps/schematic/api/schematic_api/test/test_schema_controller_impl.py +++ b/apps/schematic/api/schematic_api/test/test_schema_controller_impl.py @@ -1,7 +1,11 @@ """Tests for schema endpoint functions""" # pylint: disable=duplicate-code + from schematic_api.models.basic_error import BasicError +from schematic_api.models.component_requirement_subgraph import ( + ComponentRequirementSubgraph, +) from schematic_api.models.node_property_array import NodePropertyArray from schematic_api.models.validation_rule import ValidationRule from schematic_api.models.validation_rule_array import ValidationRuleArray @@ -13,6 +17,8 @@ from schematic_api.models.connected_node_pair import ConnectedNodePair from schematic_api.controllers.schema_controller_impl import ( get_component, + get_component_requirements_array, + get_component_requirements_graph, get_connected_node_pair_page, get_connected_node_pair_array, get_node_is_required, @@ -45,6 +51,60 @@ def test_internal_error(self, test_schema_url: str) -> None: assert isinstance(result, BasicError) +class TestGetComponentRequirementsArray: + """Tests get_component_requirements_array""" + + def test_success(self, test_schema_url: str) -> None: + """Test for successful result""" + result, status = get_component_requirements_array( + component_label="Biospecimen", + schema_url=test_schema_url, + display_label_type="class_label", + ) + assert status == 200 + assert isinstance(result, list) + for req in result: + assert isinstance(req, str) + + def test_internal_error(self, test_schema_url: str) -> None: + """Test for 500 result""" + result, status = get_component_requirements_array( + component_label="not_a_component", + schema_url=test_schema_url, + display_label_type="class_label", + ) + assert status == 500 + assert isinstance(result, BasicError) + + +class TestGetComponentRequirementsGraph: + """Tests get_component_requirements_graph""" + + def test_success(self, test_schema_url: str) -> None: + """Test for successful result""" + result, status = get_component_requirements_graph( + component_label="Biospecimen", + schema_url=test_schema_url, + display_label_type="class_label", + ) + assert status == 200 + assert isinstance(result, list) + for subgraph in result: + assert isinstance(subgraph, ComponentRequirementSubgraph) + assert isinstance(subgraph.component1, str) + assert isinstance(subgraph.component2, str) + + def test_internal_error(self, test_schema_url: str) -> None: + """Test for 500 result""" + result, status = get_component_requirements_graph( + component_label="not_a_component", + schema_url=test_schema_url, + display_label_type="class_label", + ) + assert status == 500 + assert isinstance(result, BasicError) + + class TestGetConnectedNodePairArray: """Tests get_connected_node_pair_array""" diff --git a/libs/schematic/api-description/build/openapi.yaml b/libs/schematic/api-description/build/openapi.yaml index 29c3b502d6..88e159c94b 100644 --- a/libs/schematic/api-description/build/openapi.yaml +++ b/libs/schematic/api-description/build/openapi.yaml @@ -623,6 +623,48 @@ paths: type: string '500': $ref: '#/components/responses/InternalServerError' + /components/{componentLabel}/requirementsArray: + parameters: + - $ref: '#/components/parameters/componentLabel' + get: + tags: + - Schema + summary: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. + description: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. + operationId: getComponentRequirementsArray + parameters: + - $ref: '#/components/parameters/schemaUrl' + - $ref: '#/components/parameters/displayLabelType' + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: '#/components/schemas/ComponentRequirementArray' + '500': + $ref: '#/components/responses/InternalServerError' + /components/{componentLabel}/requirementsGraph: + parameters: + - $ref: '#/components/parameters/componentLabel' + get: + tags: + - Schema + summary: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. + description: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. + operationId: getComponentRequirementsGraph + parameters: + - $ref: '#/components/parameters/schemaUrl' + - $ref: '#/components/parameters/displayLabelType' + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: '#/components/schemas/ComponentRequirementGraph' + '500': + $ref: '#/components/responses/InternalServerError' /schemaAttributes: get: tags: @@ -872,19 +914,13 @@ paths: - $ref: '#/components/parameters/manifestTitle' - $ref: '#/components/parameters/dataTypeArray' - $ref: '#/components/parameters/displayLabelType' + - $ref: '#/components/parameters/useStrictValidation' - name: assetViewId in: query description: ID of view listing all project data assets. E.g. for Synapse this would be the Synapse ID of the fileview listing all data assets for a given project required: false schema: $ref: '#/components/schemas/AssetViewId' - - name: useStrictValidation - in: query - description: If true, users are blocked from entering incorrect values. If false, users will get a warning when using incorrect values. - required: false - schema: - type: boolean - default: true - name: generateAllManifests in: query description: If true, a manifest for all components will be generated, datasetIds will be ignored. If false, manifests for each id in datasetIds will be generated. @@ -1284,6 +1320,39 @@ components: description: The label of a component in a schema type: string example: MolecularEntity + ComponentRequirementArray: + type: object + description: An array of components + properties: + componentRequirementsList: + type: array + items: + type: string + ComponentRequirementSubgraph: + type: object + description: A pair of components + properties: + component1: + type: string + description: The display name of the first component in the graph + example: component1 + component2: + type: string + description: The display name of the second component in the graph + example: component2 + required: + - component1 + - component2 + x-java-class-annotations: + - '@lombok.Builder' + ComponentRequirementGraph: + type: object + description: A graph of components + properties: + componentRequirementsGraph: + type: array + items: + $ref: '#/components/schemas/ComponentRequirementSubgraph' RelationshipType: description: A type of schema relationship type: string @@ -1359,12 +1428,15 @@ components: description: An array of dataset ids items: $ref: '#/components/schemas/DatasetId' + DataType: + description: A data type + type: string + example: Patient DataTypeArray: description: An array of data types type: array items: - type: string - example: Patient + $ref: '#/components/schemas/DataType' GoogleSheetLinks: type: object description: An array of google sheet links @@ -1681,3 +1753,11 @@ components: required: false schema: $ref: '#/components/schemas/DataTypeArray' + useStrictValidation: + name: useStrictValidation + in: query + description: If true, users are blocked from entering incorrect values. If false, users will get a warning when using incorrect values. + required: false + schema: + type: boolean + default: true diff --git a/libs/schematic/api-description/src/components/parameters/query/dataType.yaml b/libs/schematic/api-description/src/components/parameters/query/dataType.yaml new file mode 100644 index 0000000000..5bd38b8cc2 --- /dev/null +++ b/libs/schematic/api-description/src/components/parameters/query/dataType.yaml @@ -0,0 +1,6 @@ +name: dataType +in: query +description: A data type +required: false +schema: + $ref: ../../schemas/DataType.yaml diff --git a/libs/schematic/api-description/src/components/parameters/query/useStrictValidation.yaml b/libs/schematic/api-description/src/components/parameters/query/useStrictValidation.yaml new file mode 100644 index 0000000000..b403090dcd --- /dev/null +++ b/libs/schematic/api-description/src/components/parameters/query/useStrictValidation.yaml @@ -0,0 +1,8 @@ +name: useStrictValidation +in: query +description: If true, users are blocked from entering incorrect values. + If false, users will get a warning when using incorrect values. +required: false +schema: + type: boolean + default: true diff --git a/libs/schematic/api-description/src/components/schemas/ComponentRequirementArray.yaml b/libs/schematic/api-description/src/components/schemas/ComponentRequirementArray.yaml new file mode 100644 index 0000000000..1a6ff6436a --- /dev/null +++ b/libs/schematic/api-description/src/components/schemas/ComponentRequirementArray.yaml @@ -0,0 +1,7 @@ +type: object +description: An array of components +properties: + componentRequirementsList: + type: array + items: + type: string diff --git a/libs/schematic/api-description/src/components/schemas/ComponentRequirementGraph.yaml b/libs/schematic/api-description/src/components/schemas/ComponentRequirementGraph.yaml new file mode 100644 index 0000000000..a528c2d8e4 --- /dev/null +++ b/libs/schematic/api-description/src/components/schemas/ComponentRequirementGraph.yaml @@ -0,0 +1,7 @@ +type: object +description: A graph of components +properties: + componentRequirementsGraph: + type: array + items: + $ref: ComponentRequirementSubgraph.yaml diff --git a/libs/schematic/api-description/src/components/schemas/ComponentRequirementSubgraph.yaml b/libs/schematic/api-description/src/components/schemas/ComponentRequirementSubgraph.yaml new file mode 100644 index 0000000000..6a22a9c1bb --- /dev/null +++ b/libs/schematic/api-description/src/components/schemas/ComponentRequirementSubgraph.yaml @@ -0,0 +1,16 @@ +type: object +description: A pair of components +properties: + component1: + type: string + description: The display name of the first component in the graph + example: component1 + component2: + type: string + description: The display name of the second component in the graph + example: component2 +required: + - component1 + - component2 +x-java-class-annotations: + - '@lombok.Builder' diff --git a/libs/schematic/api-description/src/components/schemas/DataType.yaml b/libs/schematic/api-description/src/components/schemas/DataType.yaml new file mode 100644 index 0000000000..746e651257 --- /dev/null +++ b/libs/schematic/api-description/src/components/schemas/DataType.yaml @@ -0,0 +1,3 @@ +description: A data type +type: string +example: Patient diff --git a/libs/schematic/api-description/src/components/schemas/DataTypeArray.yaml b/libs/schematic/api-description/src/components/schemas/DataTypeArray.yaml index a73a0cbca2..8500bbedfd 100644 --- a/libs/schematic/api-description/src/components/schemas/DataTypeArray.yaml +++ b/libs/schematic/api-description/src/components/schemas/DataTypeArray.yaml @@ -1,5 +1,4 @@ description: An array of data types type: array items: - type: string - example: Patient + $ref: DataType.yaml diff --git a/libs/schematic/api-description/src/openapi.yaml b/libs/schematic/api-description/src/openapi.yaml index 944ffdf4b2..ebd57d8ec9 100644 --- a/libs/schematic/api-description/src/openapi.yaml +++ b/libs/schematic/api-description/src/openapi.yaml @@ -89,6 +89,12 @@ paths: /components/{componentLabel}/: $ref: paths/components/@{componentLabel}/component.yaml + /components/{componentLabel}/requirementsArray: + $ref: paths/components/@{componentLabel}/requirementsArray.yaml + + /components/{componentLabel}/requirementsGraph: + $ref: paths/components/@{componentLabel}/requirementsGraph.yaml + /schemaAttributes: $ref: paths/schemaAttributes.yaml diff --git a/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsArray.yaml b/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsArray.yaml new file mode 100644 index 0000000000..2605d91337 --- /dev/null +++ b/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsArray.yaml @@ -0,0 +1,20 @@ +parameters: + - $ref: ../../../components/parameters/path/componentLabel.yaml +get: + tags: + - Schema + summary: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. + description: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in array form. + operationId: getComponentRequirementsArray + parameters: + - $ref: ../../../components/parameters/query/schemaUrl.yaml + - $ref: ../../../components/parameters/query/displayLabelType.yaml + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: ../../../components/schemas/ComponentRequirementArray.yaml + '500': + $ref: ../../../components/responses/InternalServerError.yaml diff --git a/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsGraph.yaml b/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsGraph.yaml new file mode 100644 index 0000000000..50fb974060 --- /dev/null +++ b/libs/schematic/api-description/src/paths/components/@{componentLabel}/requirementsGraph.yaml @@ -0,0 +1,20 @@ +parameters: + - $ref: ../../../components/parameters/path/componentLabel.yaml +get: + tags: + - Schema + summary: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. + description: Given a source model component (see https://w3id.org/biolink/vocab/category for definnition of component), return all components required by it in graph form. + operationId: getComponentRequirementsGraph + parameters: + - $ref: ../../../components/parameters/query/schemaUrl.yaml + - $ref: ../../../components/parameters/query/displayLabelType.yaml + responses: + '200': + description: Success + content: + application/json: + schema: + $ref: ../../../components/schemas/ComponentRequirementGraph.yaml + '500': + $ref: ../../../components/responses/InternalServerError.yaml diff --git a/libs/schematic/api-description/src/paths/generateGoogleSheetManifests.yaml b/libs/schematic/api-description/src/paths/generateGoogleSheetManifests.yaml index 251622d549..40ab67dd60 100644 --- a/libs/schematic/api-description/src/paths/generateGoogleSheetManifests.yaml +++ b/libs/schematic/api-description/src/paths/generateGoogleSheetManifests.yaml @@ -11,20 +11,13 @@ get: - $ref: ../components/parameters/query/manifestTitle.yaml - $ref: ../components/parameters/query/dataTypeArray.yaml - $ref: ../components/parameters/query/displayLabelType.yaml + - $ref: ../components/parameters/query/useStrictValidation.yaml - name: assetViewId in: query description: ID of view listing all project data assets. E.g. for Synapse this would be the Synapse ID of the fileview listing all data assets for a given project required: false schema: $ref: ../components/schemas/AssetViewId.yaml - - name: useStrictValidation - in: query - description: If true, users are blocked from entering incorrect values. - If false, users will get a warning when using incorrect values. - required: false - schema: - type: boolean - default: true - name: generateAllManifests in: query description: