diff --git a/src/metldata/builtin_transformations/count_references/assumptions.py b/src/metldata/builtin_transformations/count_references/assumptions.py index 5258eb2..b6df458 100644 --- a/src/metldata/builtin_transformations/count_references/assumptions.py +++ b/src/metldata/builtin_transformations/count_references/assumptions.py @@ -22,6 +22,7 @@ ) from metldata.builtin_transformations.common.path.path import RelationPath from metldata.builtin_transformations.common.path.path_elements import ( + RelationPathElement, RelationPathElementType, ) from metldata.builtin_transformations.count_references.instruction import ( @@ -30,7 +31,49 @@ from metldata.transform.base import ModelAssumptionError, MultiplicityError -def validate_modification_class(path_element, expected_class_name): +def check_model_assumptions( + schema: SchemaPack, + instructions_by_class: dict[str, list[AddReferenceCountPropertyInstruction]], +) -> None: + """Check the model assumptions for the count references transformation.""" + for _, instructions in instructions_by_class.items(): + for instruction in instructions: + assert_only_direct_relations(instruction) + assert_class_is_source(instruction) + assert_path_classes_and_relations_exist( + schema, instruction.source_relation_path + ) + assert_multiplicity(schema, instruction.source_relation_path) + assert_object_path_exists(schema, instruction) + + +def assert_only_direct_relations(instruction: AddReferenceCountPropertyInstruction): + """Ensure that only direct relations are suppported which should be the case if the + relation path only contains one path element. + """ + num_elements = len(instruction.source_relation_path.elements) + if num_elements != 1: + raise ModelAssumptionError( + f"The provided relation path { + instruction.source_relation_path.path_str}" + f"does not describe a direct relation, but contains { + num_elements} different relations" + ) + + +def assert_class_is_source(instruction: AddReferenceCountPropertyInstruction): + """Ensure that the class being modified with the reference count property is the expected class. + This function iterates over the elements of the relation path in the given instruction + and validates that the class being modified with the reference count property matches + the class specified in the relation path. + """ + for path_element in instruction.source_relation_path.elements: + _validate_modification_class(path_element, instruction.class_name) + + +def _validate_modification_class( + path_element: RelationPathElement, expected_class_name: str +): """Check whether the class specified to be modified with the reference count matches the source or target class in the provided `path_element`, depending on the type of the relation path (i.e., active or passive). If the class does not match, @@ -49,31 +92,6 @@ def validate_modification_class(path_element, expected_class_name): ) -def check_class_exists(model: SchemaPack, class_name: str) -> None: - """Check if a class exists in the model and raise an error if not""" - if class_name not in model.classes: - raise ModelAssumptionError(f"Class {class_name} not found in model.") - - -def check_relation_exists(model: SchemaPack, class_name: str, relation: str): - """Check if a relation exists in a class and raise an error if not""" - if relation not in model.classes[class_name].relations: - raise ModelAssumptionError( - f"Relation property { - relation} not found in class {class_name}." - ) - - -def assert_class_is_source(instruction: AddReferenceCountPropertyInstruction): - """Ensure that the class being modified with the reference count property is the expected class. - This function iterates over the elements of the relation path in the given instruction - and validates that the class being modified with the reference count property matches - the class specified in the relation path. - """ - for path_element in instruction.source_relation_path.elements: - validate_modification_class(path_element, instruction.class_name) - - def assert_path_classes_and_relations_exist(model: SchemaPack, path: RelationPath): """Make sure that all classes and relations defined in the provided path exist in the provided model. @@ -83,14 +101,29 @@ def assert_path_classes_and_relations_exist(model: SchemaPack, path: RelationPat if the model does not fulfill the assumptions. """ for path_element in path.elements: - check_class_exists(model, path_element.source) - check_class_exists(model, path_element.target) + _check_class_exists(model, path_element.source) + _check_class_exists(model, path_element.target) if path_element.type_ == RelationPathElementType.ACTIVE: - check_relation_exists(model, path_element.source, path_element.property) + _check_relation_exists(model, path_element.source, path_element.property) if path_element.type_ == RelationPathElementType.PASSIVE: - check_relation_exists(model, path_element.target, path_element.property) + _check_relation_exists(model, path_element.target, path_element.property) + + +def _check_class_exists(model: SchemaPack, class_name: str) -> None: + """Check if a class exists in the model and raise an error if not""" + if class_name not in model.classes: + raise ModelAssumptionError(f"Class {class_name} not found in model.") + + +def _check_relation_exists(model: SchemaPack, class_name: str, relation: str): + """Check if a relation exists in a class and raise an error if not""" + if relation not in model.classes[class_name].relations: + raise ModelAssumptionError( + f"Relation property { + relation} not found in class {class_name}." + ) def assert_multiplicity(model: SchemaPack, path: RelationPath): @@ -107,7 +140,7 @@ def assert_multiplicity(model: SchemaPack, path: RelationPath): ) -def assert_summary_exists( +def assert_object_path_exists( model: SchemaPack, instruction: AddReferenceCountPropertyInstruction, ) -> None: @@ -141,18 +174,3 @@ def assert_summary_exists( instruction.target_content.property_name} does not exist" + f" in class {class_name}." ) - - -def check_model_assumptions( - schema: SchemaPack, - instructions_by_class: dict[str, list[AddReferenceCountPropertyInstruction]], -) -> None: - """Check the model assumptions for the count references transformation.""" - for _, instructions in instructions_by_class.items(): - for instruction in instructions: - assert_class_is_source(instruction) - assert_path_classes_and_relations_exist( - schema, instruction.source_relation_path - ) - assert_multiplicity(schema, instruction.source_relation_path) - assert_summary_exists(schema, instruction)