-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Documenting shape-to-template process and adding some QoL improvements (
#278) * documenting shape-to-template process and adding some QoL improvements * fix test * skip 223P tests until gabe can fix them after next standard prerelease is done * remember to add shapes from the graph even when loading from directory * add exception to catch lack of sh:path * add test cases for recent code changes * dedicated test suite for library shape test * dedicated test suite for library shape test
- Loading branch information
Showing
9 changed files
with
310 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
--- | ||
jupytext: | ||
cell_metadata_filter: -all | ||
formats: md:myst | ||
text_representation: | ||
extension: .md | ||
format_name: myst | ||
kernelspec: | ||
display_name: Python 3 | ||
language: python | ||
name: python3 | ||
--- | ||
|
||
# Shapes and Templates | ||
|
||
Shapes and Templates interact in interesting ways in BuildingMOTIF. | ||
In this document, we explain the utility and function of these interactions. | ||
|
||
Recall that a **Shape** (SHACL shape) is a set of conditions and constraints over RDF graphs, and | ||
a **Template** is a function that generates an RDF graph. | ||
|
||
## Converting Shapes to Templates | ||
|
||
BuildingMOTIF automatically converts shapes to templates. | ||
Evaluating the resulting template will generate a graph that validates against the shape. | ||
|
||
When BuildingMOTIF loads a Library, it makes an attempt to find any shapes defined within it. | ||
The way this happens depends on how the library is loaded: | ||
- *Loading library from directory or git repository*: BuildingMOTIF searches for any RDF files in the directory (recursively) and loads them into a Shape Collection; loads any instances of `sh:NodeShape` in the union of these RDF files | ||
- *Loading library from ontology file*: loads all instances of `sh:NodeShape` in the provided graphc | ||
|
||
```{important} | ||
BuildingMOTIF *only* loads shapes which are instances of *both* `sh:NodeShape` **and** `owl:Class`. The assumption is that `owl:Class`-ified shapes could be "instantiated". | ||
``` | ||
|
||
Each shape is "decompiled" into components from which a Template can be constructed. | ||
The implementation of this decompilation is in the [`get_template_parts_from_shape`](/reference/apidoc/_autosummary/buildingmotif.utils.html#buildingmotif.utils.get_template_parts_from_shape) method. | ||
BuildingMOTIF currently recognizes the following SHACL properties: | ||
- `sh:property` | ||
- `sh:qualifiedValueShape` | ||
- `sh:node` | ||
- `sh:class` | ||
- `sh:targetClass` | ||
- `sh:datatype` | ||
- `sh:minCount` / `sh:qualifiedMinCount` | ||
- `sh:maxCount` / `sh:qualifiedMaxCount` | ||
|
||
BuildingMOTIF currently uses the name of the SHACL shape as the name of the generated Template. | ||
All other parameters (i.e., nodes corresponding to `sh:property`) are given invented names *unless* | ||
there is a `sh:name` attribute on the property shape. | ||
|
||
### Example | ||
|
||
Consider the following shape which has been loaded into BuildingMOTIF as part of a Library: | ||
|
||
```ttl | ||
# myshapes.ttl | ||
@prefix brick: <https://brickschema.org/schema/Brick#> . | ||
@prefix sh: <http://www.w3.org/ns/shacl#> . | ||
@prefix owl: <http://www.w3.org/2002/07/owl#> . | ||
@prefix : <urn:example/> . | ||
: a owl:Ontology . | ||
:vav a sh:NodeShape, owl:Class ; | ||
sh:targetClass brick:Terminal_Unit ; | ||
sh:property [ | ||
sh:path brick:hasPart ; | ||
sh:qualifiedValueShape [ sh:node :heating-coil ] ; | ||
sh:name "hc" ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Supply_Air_Flow_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Supply_Air_Temperature_Sensor ] ; | ||
sh:name "sat" ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. | ||
:heating-coil a sh:NodeShape, owl:Class ; | ||
sh:targetClass brick:Heating_Coil ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Position_Command ] ; | ||
sh:name "damper_pos" ; # will be used as the parameter name | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. | ||
``` | ||
|
||
<details> | ||
|
||
This code creates `myshapes.ttl` for you in the current directory. | ||
|
||
```{code-cell} python3 | ||
with open("myshapes.ttl", "w") as f: | ||
f.write(""" | ||
@prefix brick: <https://brickschema.org/schema/Brick#> . | ||
@prefix sh: <http://www.w3.org/ns/shacl#> . | ||
@prefix owl: <http://www.w3.org/2002/07/owl#> . | ||
@prefix : <urn:example/> . | ||
: a owl:Ontology . | ||
:vav a sh:NodeShape, owl:Class ; | ||
sh:targetClass brick:Terminal_Unit ; | ||
sh:property [ | ||
sh:path brick:hasPart ; | ||
sh:qualifiedValueShape [ sh:node :heating-coil ] ; | ||
sh:name "hc" ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Supply_Air_Flow_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Supply_Air_Temperature_Sensor ] ; | ||
sh:name "sat" ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. | ||
:heating-coil a sh:NodeShape, owl:Class ; | ||
sh:targetClass brick:Heating_Coil ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Position_Command ] ; | ||
sh:name "damper_pos" ; # will be used as the parameter name | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. | ||
""") | ||
``` | ||
|
||
</details> | ||
|
||
If this was in a file `myshapes.ttl`, we would load it into BuildingMOTIF as follows: | ||
|
||
```{code-cell} python3 | ||
from buildingmotif import BuildingMOTIF | ||
from buildingmotif.dataclasses import Library | ||
# in-memory instance | ||
bm = BuildingMOTIF("sqlite://") | ||
# load library | ||
brick = Library.load(ontology_graph="https://github.com/BrickSchema/Brick/releases/download/nightly/Brick.ttl") | ||
lib = Library.load(ontology_graph="myshapes.ttl") | ||
``` | ||
|
||
Once the library has been loaded, all of the shapes have been turned into templates. | ||
We can load the template by name (using its *full URI* from the shape) as if it were | ||
defined explicitly: | ||
|
||
```{code-cell} python3 | ||
# reading the template out by name | ||
template = lib.get_template_by_name("urn:example/vav") | ||
# dump the body of the template | ||
print(template.body.serialize()) | ||
``` | ||
|
||
As with other templates, we often want to *inline* all dependencies to get a sense of what metadata will be added to the graph. | ||
|
||
```{code-cell} python3 | ||
# reading the template out by name | ||
template = lib.get_template_by_name("urn:example/vav").inline_dependencies() | ||
# dump the body of the template | ||
print(template.body.serialize()) | ||
``` | ||
|
||
Observe that the generated template uses the `sh:name` property of each property shape to inform the paramter name. If this is not provided (e.g. for the `brick:Supply_Air_Flow_Sensor` property shape), then a generated parameter will be used. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
@prefix brick: <https://brickschema.org/schema/Brick#> . | ||
@prefix owl: <http://www.w3.org/2002/07/owl#> . | ||
@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> . | ||
@prefix rdfs: <http://www.w3.org/2000/01/rdf-schema#> . | ||
@prefix sh: <http://www.w3.org/ns/shacl#> . | ||
@prefix : <urn:shape1/> . | ||
|
||
: a owl:Ontology . | ||
|
||
:vav_shape a owl:Class, sh:NodeShape ; | ||
sh:targetClass brick:VAV ; | ||
sh:property [ | ||
# missing sh:path! | ||
sh:qualifiedValueShape [ sh:class brick:Air_Flow_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
sh:minCount 1; | ||
] ; | ||
. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
@prefix brick: <https://brickschema.org/schema/Brick#> . | ||
@prefix owl: <http://www.w3.org/2002/07/owl#> . | ||
@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> . | ||
@prefix rdfs: <http://www.w3.org/2000/01/rdf-schema#> . | ||
@prefix sh: <http://www.w3.org/ns/shacl#> . | ||
@prefix : <urn:shape/> . | ||
|
||
: a owl:Ontology . | ||
|
||
:vav_shape a owl:Class, sh:NodeShape ; | ||
sh:targetClass brick:VAV ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Air_Flow_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
sh:minCount 1; | ||
] ; | ||
. | ||
|
||
:tu_shape a owl:Class, sh:NodeShape ; | ||
sh:targetClass brick:Terminal_Unit ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Temperature_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
sh:minCount 1; | ||
] ; | ||
. | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
@prefix brick: <https://brickschema.org/schema/Brick#> . | ||
@prefix owl: <http://www.w3.org/2002/07/owl#> . | ||
@prefix rdf: <http://www.w3.org/1999/02/22-rdf-syntax-ns#> . | ||
@prefix rdfs: <http://www.w3.org/2000/01/rdf-schema#> . | ||
@prefix sh: <http://www.w3.org/ns/shacl#> . | ||
@prefix : <urn:shape/> . | ||
|
||
: a owl:Ontology . | ||
|
||
:vav_shape a owl:Class, sh:NodeShape ; | ||
sh:targetClass brick:VAV ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Air_Flow_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. | ||
|
||
:tu_shape a owl:Class, sh:NodeShape ; | ||
sh:targetClass brick:Terminal_Unit ; | ||
sh:property [ | ||
sh:path brick:hasPoint ; | ||
sh:qualifiedValueShape [ sh:class brick:Temperature_Sensor ] ; | ||
sh:qualifiedMinCount 1 ; | ||
] ; | ||
. |
Oops, something went wrong.