diff --git a/doc/06_aggregate/00_introduction.md b/doc/06_aggregate/00_introduction.md new file mode 100644 index 0000000..1730fe4 --- /dev/null +++ b/doc/06_aggregate/00_introduction.md @@ -0,0 +1,47 @@ +# Aggregates + +Here is the Idea behind it. Sometimes we have Things in our Graph that are Parts of a bigger Thing. For example a Database has Tables, a Table has Columns. There is no Column without a Table and there is no Table without a Database. The Database "consists" of Tables and the Table "consists" of Columns. + +Another Example is a Namespace that "consists" of Deployments, StatefulSets and Hosts. The Pictures below show the Graphs of an Example Namespace. + +![Namespace Image](img/namespace.png) + +Sometimes we are interested in all these tiny elements in the Graph. But when it becomes big the Graph becomes unreadable. Especially when you are not an expert in the Domain. To illustrate this, here is the Graph of the same Namespace but with its neighbors. + +![Namespace Image](img/bigger_graph.png) + +Only one who is an expert in the Domain can really understand this Graph. The problem is that an expert knows the boundaries of higher level concepts. But a non-expert does not. Here is the same graph with the "Namespace Boundary" highlighted. + +![Namespace Boundary](img/boundary.png) + +What if we add this "Boundary" information to the Graph? Then we can make it explicit what belongs to a Namespace and what does not. Furthermore we can hide the details of the Namespace and only show the Boundary. We can define "connection" Points where we can connect to a Namespace. The blue ellipses in the Picture below are the "connection" Points. + +![Namespace Boundary](img/boundary_with_connection.png) + +If we add this information to the Graph we are able to decide if we look "into" the Namespace or "at" the Namespace. We can "hide" the internals of the Namespace and only show the Boundary. Here is a possible view where we look "at" the Namespace. It's an outside view. + +![Namespace to Image](img/namespace_image.png) + +What is this outside view? I will show you. + +The namespace is "Aggregated". You see just namespace and not the internals. + +![Namespace Aggregate](img/namespace_aggregate.png) + +The next thing are the connection Points. These are point we can connect to the Namespace. In our example it the Deployment. + +![Namespace Deployment Connection Point](img/deployment_connection.png) + +The last thing are the neighbors connected to the namespace. In our example these neighbors are the Images. + +![Namespace Image Connection](img/images.png) + +This is the Idea behind Aggregates. We can define a Boundary and hide the internals. We can define connection Points and show the neighbors. We can define a view where we look "at" the Aggregate and not "into" the Aggregate. + +Here is the View from the image to the Namespace. The Image is connected to Namespaces. We can directly see where is this Image used. + +![Image to Namespace](img/image_namespace.png) + +Which is the same as the following Graph but much more readable. + +![Image Graph with Namespaces](img/image_graph.png) \ No newline at end of file diff --git a/doc/06_aggregate/01_configuration.md b/doc/06_aggregate/01_configuration.md new file mode 100644 index 0000000..c6da9a1 --- /dev/null +++ b/doc/06_aggregate/01_configuration.md @@ -0,0 +1,182 @@ +# Configure the Aggregation / Composition / Container + +There are two kind of Aggregates. The first is the "Composition". The second is the "Container". The Composition is a collection of things that compose a higher level thing. The Container is a collection of things that are contained in another thing but it is not composed by them. + +Containers are not as strong as Compositions. Use Compositions for a start. + + +```turtle +bp:Aggregate a rdfs:Class ; + rdfs:label "Aggregate" ; + rdfs:comment "An aggregate is a collection of things, such as a list of items, a set of items, or a group of items." ; +. + +bp:Composition a rdfs:Class ; + rdfs:subClassOf bp:Aggregate ; + rdfs:label "Composition" ; + rdfs:comment "A composition is a collection of things, that compose a higher level thing." ; +. + +bp:Container a rdfs:Class ; + rdfs:subClassOf bp:Aggregate ; + rdfs:label "Container" ; + rdfs:comment "A container is a collection of things, that are contained in a another thing but it is not composed by them." ; +. +``` + + +## Tree +Here we configure a Namespace to be an Aggregate. We define the Boundary and the Connection Points. + +First we define the internals of a Namespace. We define a Tree Structure with the following levels: + +- Namespace + - Ingress + - Host + - Deployment + - StatefulSet + +The Namespace is the Root of the Tree. The Ingress and Deployment are the first level of the Tree. The Host is a child of the Ingress. The Deployment and StatefulSet are children of the Namespace. + +Technically we define a SHACL Shape for each level of the Tree. You can see it as a "chained" SHACL shapes. + +```turtle +PREFIX sh: +PREFIX rdf: +PREFIX rdfs: +PREFIX bp: +PREFIX data: +PREFIX oci: +PREFIX k8s: + + +## k8s namespace hierarchy +# Root - Namespace +data:K8SNamespaceHierarchyShape a sh:NodeShape ; + rdfs:label "K8S Namespace Hierarchy"; # Just a label shown in the UI + sh:targetClass k8s:Namespace; # The target class of the Shape - Namespace in our case + sh:group data:K8SNamespaceTree ; # The group of the Shape - the Tree + sh:property [ + sh:name "Origin"; # A label show in the UI + sh:path [ sh:inversePath k8s:namespace ] ; # the path to follow - no lists here, just a single path or single inverse path + sh:node data:K8SIngressHierarchyShape; # the next (level) shape in the hierarchy - Ingress + ] , + [ + sh:name "Origin"; # A label show in the UI + sh:path [ sh:inversePath k8s:namespace ] ; # the path to follow - no lists here, just a single path or single inverse path + sh:node data:K8SDeploymentHierarchyShape; # the next (level) shape in the hierarchy - Deployment + ], + [ + sh:name "Origin"; # A label show in the UI + sh:path [ sh:inversePath k8s:namespace ] ; # the path to follow - no lists here, just a single path or single inverse path + sh:node data:K8SStatefulSetHierarchyShape; # the next (level) shape in the hierarchy - StatefulSet + ] +. + +# level 1 - Ingress +data:K8SIngressHierarchyShape a sh:NodeShape ; + rdfs:label "Namespace"; + sh:targetClass k8s:Ingress; + sh:group data:K8SNamespaceTree ; + sh:property [ + sh:path k8s:host ; + sh:node data:K8SIngressHostHierarchyShape ; + ] ; +. + +# level 2 - Ingress - Host +data:K8SIngressHostHierarchyShape a sh:NodeShape ; + rdfs:label "Ingress"; + sh:targetClass k8s:Host; + sh:group data:K8SNamespaceTree ; + . + +# level 1 - Deployment +data:K8SDeploymentHierarchyShape a sh:NodeShape, bp:ConnectionPoint; # this is a connection point - here we accept links .... later down in this document + rdfs:label "Namespace"; + sh:targetClass k8s:Deployment; + sh:group data:K8SNamespaceTree ; +. + +# level 1 - StatefulSet +data:K8SStatefulSetHierarchyShape a sh:NodeShape, bp:ConnectionPoint; # this is a connection point - here we accept links .... later down in this document + rdfs:label "Namespace"; + sh:targetClass k8s:StatefulSet; + sh:group data:K8SNamespaceTree ; +. + +# All nodes of this tree are part of the same group +# This is used to reference the whole tree and add metadata to it +data:K8SNamespaceTree a sh:Group, bp:Hierarchy, bp:Composition; # As you can see this is a Group, a Hierarchy and a Composition - Composition means it is composed of these things defined in the tree + rdfs:label "K8S Namespaces" ; # The name of this tree + rdfs:comment "Information about Kubernetes namespaces, including details about the ingress, deployment and statefulset." ; # a description of the tree + bp:hasRoot data:K8SNamespaceHierarchyShape; # define the tree root shape +. + +``` + +With a configuration like this Blueprint knows a lot more about your Namespace. It has some effects in the UI. + +**Inventory** + +You see it in the Inventory. + +![Inventory](img/inventory.png) + +And you can search it or you can have a look at the contents of the Namespace. In many cases it's more to use this and not the "Search". Because you have more context. + +![Inventory](img/inventory_detail.png) + +**Explore** +Blueprint shows the information about trees in the Explore View in two ways. + +![Explore](img/explore.png) + +In the example are exploring the Ingress. In the blue Box you see the tree "upwards". It means this Ingress is part of a Namespace. In the green Box you see the tree "downwards". It means this Ingress has a Host. It is all based on the same Tree configuration. + + + +# Links between Aggregates + +## Aggregate to Node links +Now we have defined an Aggregates. Lets say we have defined our Namespace tree as a composition. We can now look "into" the Namespace. But we can also look "at" the Namespace. That means we define links to a Namespace (as the composed one - basically a link to the namespace tree). + +Here we go. We define a link between a Namespace and an OCI Image. The link is a CompositionToNodeLink. It means the Namespace is the composed one and the OCI Image is the Node. + +```turtle +PREFIX sh: +PREFIX rdf: +PREFIX rdfs: +PREFIX bp: +PREFIX data: +PREFIX oci: +PREFIX k8s: + + +data:K8SNamespaceTreeToOciTreeLink a bp:CompositionToNodeLink ; + sh:targetClass data:K8SNamespaceTree ; # the target class is the targetClass of the SHACL Shape - but the source of our link + bp:target oci:Image ; # this is the target of our link - the OCI Image + rdfs:label "Depends on OCI Images" ; # a label we can show in the UI + bp:inverseLabel "OCI Images are used by" ; # if we see the link in the opposite direction we show this label + # there a two links between a Namespace and an OCI Image + # the "connection" points are the Deployment and the StatefulSet + # here we define the path from a "connection" point to the OCI Image + sh:property [ + sh:targetClass k8s:Deployment ; # the source of the link (connection point Deployment) + sh:path k8s:image ; # the path to follow to reach the target node (the OCI Image) + sh:class oci:Image ; # the target class of the instance on the other side of the link + ]; + sh:property [ + sh:targetClass k8s:StatefulSet ; # the source of the link (connection point StatefulSet) + sh:path k8s:image ; # the path to follow to reach the target node (the OCI Image) + sh:class oci:Image ; # the target class of the instance on the other side of the link + ] +. +``` + +This leas to the following UI. +![Link](img/namespace_image.png) +or this in the opposite direction +![Link](img/image_namespace.png) + + diff --git a/doc/06_aggregate/img/bigger_graph.png b/doc/06_aggregate/img/bigger_graph.png new file mode 100644 index 0000000..82c1437 Binary files /dev/null and b/doc/06_aggregate/img/bigger_graph.png differ diff --git a/doc/06_aggregate/img/boundary.png b/doc/06_aggregate/img/boundary.png new file mode 100644 index 0000000..988aa0b Binary files /dev/null and b/doc/06_aggregate/img/boundary.png differ diff --git a/doc/06_aggregate/img/boundary_with_connection.png b/doc/06_aggregate/img/boundary_with_connection.png new file mode 100644 index 0000000..4f2f8a4 Binary files /dev/null and b/doc/06_aggregate/img/boundary_with_connection.png differ diff --git a/doc/06_aggregate/img/deployment_connection.png b/doc/06_aggregate/img/deployment_connection.png new file mode 100644 index 0000000..4b42d61 Binary files /dev/null and b/doc/06_aggregate/img/deployment_connection.png differ diff --git a/doc/06_aggregate/img/explore.png b/doc/06_aggregate/img/explore.png new file mode 100644 index 0000000..ac3640a Binary files /dev/null and b/doc/06_aggregate/img/explore.png differ diff --git a/doc/06_aggregate/img/image_graph.png b/doc/06_aggregate/img/image_graph.png new file mode 100644 index 0000000..db04377 Binary files /dev/null and b/doc/06_aggregate/img/image_graph.png differ diff --git a/doc/06_aggregate/img/image_namespace.png b/doc/06_aggregate/img/image_namespace.png new file mode 100644 index 0000000..2af7552 Binary files /dev/null and b/doc/06_aggregate/img/image_namespace.png differ diff --git a/doc/06_aggregate/img/images.png b/doc/06_aggregate/img/images.png new file mode 100644 index 0000000..7f86da6 Binary files /dev/null and b/doc/06_aggregate/img/images.png differ diff --git a/doc/06_aggregate/img/inventory.png b/doc/06_aggregate/img/inventory.png new file mode 100644 index 0000000..33453c1 Binary files /dev/null and b/doc/06_aggregate/img/inventory.png differ diff --git a/doc/06_aggregate/img/inventory_detail.png b/doc/06_aggregate/img/inventory_detail.png new file mode 100644 index 0000000..0755e79 Binary files /dev/null and b/doc/06_aggregate/img/inventory_detail.png differ diff --git a/doc/06_aggregate/img/namespace.png b/doc/06_aggregate/img/namespace.png new file mode 100644 index 0000000..e1ceda5 Binary files /dev/null and b/doc/06_aggregate/img/namespace.png differ diff --git a/doc/06_aggregate/img/namespace_aggregate.png b/doc/06_aggregate/img/namespace_aggregate.png new file mode 100644 index 0000000..9e11926 Binary files /dev/null and b/doc/06_aggregate/img/namespace_aggregate.png differ diff --git a/doc/06_aggregate/img/namespace_image.png b/doc/06_aggregate/img/namespace_image.png new file mode 100644 index 0000000..bc302d8 Binary files /dev/null and b/doc/06_aggregate/img/namespace_image.png differ