Creating a schema
In Infrahub, schemas play a crucial role in defining the structure and relationships of your data. This guide will walk you through the process of creating a new schema file, we will do this using the following steps:
- Adding nodes and attributes
- Adding relationships to the nodes
- Abstracting nodes into generics
- Improving our schema by modifying attributes and adding new attributes
In this guide we will create a schema for a network device and its interfaces.
By no mean this is meant to be a complete schema for a network device, there is far more complexity that goes into modeling a schema for a network device, it is just used as an example to guide you through the process.
For a more detailed explanation of the different Attributes and Relations, you can check out our Schema Topic.
Throughout this guide we will load different schemas into Infrahub into different branches. The fact that multiple schemas can exist simultaneously is a core feature of Infrahub. It is a recommendation and best practice to always use a branch to make changes to the schema.
To help with the development process of a schema definition file, you can leverage schema validation within your editor.
1. Adding nodes and attributes
Create a file named schema_guide.yml
on your computer. The place where you create the file on the file system is not that important, as long as you know the path to the file. For this guide we will be storing the schema file in the /tmp
directory.
In the file we will be creating 2 kinds of nodes in the Network
namespace:
Device
: the network device we want to modelInterface
: the network interface we want to model
The NetworkDevice
node will have the following attributes:
hostname
: thehostname
of the device, needs to be unique and is a required attributemodel
(Text): the model of the device, which is a required attribute
The NetworkInterface
node will have the following attributes:
name
(Text): the name of the interface, which is a required attributedescription
(Text): a description for the interface, which is a required attribute
We define a human_friendly_id
on the hostname
attribute of the NetworkDevice
. This way we can use the hostname
as an alternative for the hfid
in the queries and mutations in this guide.
---
version: "1.0"
nodes:
- name: Device
namespace: Network
human_friendly_id: ['hostname__value']
attributes:
- name: hostname
kind: Text
unique: true
- name: model
kind: Text
- name: Interface
namespace: Network
attributes:
- name: name
kind: Text
- name: description
kind: Text
optional: true
Create a branch network-device-schema
in Infrahub
infrahubctl branch create network-device-schema
Load the schema into Infrahub in the network-device-schema
branch
infrahubctl schema load --branch network-device-schema /tmp/schema_guide.yml
We can inspect the schema in the Web UI (Unified Storage > Schema) as shown below.
We'll create a device and an interface according to the schema by using the web interface, GraphQL Query or cURL.
- Via the GraphQL Interface
- Via the Web Interface
- Using cURL
Open the GraphQL sandbox (bottom left of the web interface) and make a query with the following:
mutation {
NetworkDeviceCreate(data: {hostname: {value: "atl1-edge1"}, model: {value: "Cisco ASR1002-HX"}}) {
ok
object {
id
}
}
NetworkInterfaceCreate(data: {name: {value: "Ethernet1"}, description: {value: "WAN interface"}}) {
ok
object {
id
}
}
}
- Login to Infrahub's web interface.
- Click on Objects > Device in the left side menu.
- Click on 'Add Device'
- Create a new device with 'atl1-edge1' as Hostname and 'Cisco ASR1002-HX' as Model and click save.
- Under Interface, create a new interface with 'Ethernet1' as Name and 'WAN interface' as description.
Here is an example of using cURL
to make the query. Make sure to replace the X-INFRAHUB-KEY
and the IP address with your actual details. Please also make sure to include the name of the branch in the URL. If you want to learn more about GraphQL, you can find more information here.
curl -X POST http://localhost:8000/graphql/network-device-schema \
-H "Content-Type: application/json" \
-H "X-INFRAHUB-KEY: 1802eed5-eeb7-cc45-2e4d-c51de9d66cba" \
-d '{"query": "mutation { NetworkDeviceCreate(data: {hostname: {value: \"atl1-edge1\"}, model: {value: \"Cisco ASR1002-HX\"}}) { ok object { id } } NetworkInterfaceCreate(data: {name: {value: \"Ethernet1\"}, description: {value: \"WAN interface\"}}) { ok object { id } } }"}'
You can verify that both the Device and Interface were added to Infrahub by navigating to Objects
and selecting Device
or Interface
.
We have now successfully created our first schema, loaded into a branch into Infrahub and created the first nodes in the network-device-schema
branch in Infrahub.
2. Adding relationships to the nodes
The next thing we want to do is adding relationships to our schema. We first created device and interface nodes, but we didn't create any relation between them. They are 2 independent nodes within our schema.
We will be adding 2 relationships to the schema:
- device > interface, a relation
interfaces
from the device node to the interface node, of cardinalitymany
and kindComponent
- interface > device, a relation
device
from the interface node to device node of cardinalityone
and kindParent
Overwrite the schema_guide.yml
with the following contents:
---
version: "1.0"
nodes:
- name: Device
namespace: Network
human_friendly_id: ['hostname__value']
attributes:
- name: hostname
kind: Text
unique: true
- name: model
kind: Text
relationships:
- name: interfaces
cardinality: many
peer: NetworkInterface
kind: Component
- name: Interface
namespace: Network
attributes:
- name: name
kind: Text
- name: description
kind: Text
optional: true
relationships:
- name: device
cardinality: one
peer: NetworkDevice
optional: false
kind: Parent
Create a new branch to load the new version of our schema
infrahubctl branch create network-device-relations
Load the schema in the branch
infrahubctl schema load --branch network-device-relations /tmp/schema_guide.yml
We can inspect the schema in the Web UI
Use this GraphQL mutations to create a NetworkInterface Ethernet1
, a NetworkDevice atl1-edge1
.
mutation {
NetworkDeviceCreate(data: {hostname: {value: "atl1-edge1"}, model: {value: "Cisco ASR1002-HX"}}) {
ok
object {
id
}
}
NetworkInterfaceCreate(data: {name: {value: "Ethernet1"}, description: {value: "WAN interface"}, device: {hfid: "atl1-edge1"} }) {
ok
object {
id
}
}
}
In the Web UI we can now see that the device has a relation to the Ethernet1 interface.
3. Abstracting nodes into generics
We would like to add another interface Vlan1
to our device atl1-edge1
. We could add the interface as a NetworkInterface
node, however this could cause some problems. The Vlan1
interface is a logical interface, Ethernet1
on the other hand is a physical interface. While physical interfaces a lot of properties in common with logical interfaces, they also have their differences. A cable can be plugged into a Physical interface, which you cannot do on a logical interface.
This is a perfect case for us to abstract common attributes/relationships into a generic.
We will modify the schema:
- Implement a generic
NetworkInterface
that gathers the common attributes and relationships - Define a
NetworkPhysicalInterface
node - Define a
NetworkLogicalInterface
node
The interfaces relationships on a device does not need to be modified. A relationship can point to a generic node as the peer of the relation. This allows us to add relations to both a NetworkPhysicalInterface
node and NetworkLogicalInterface
.
Overwrite the schema_guide.yml
file with the following contents:
---
version: "1.0"
generics:
- name: Interface
namespace: Network
attributes:
- name: name
kind: Text
- name: description
kind: Text
optional: true
relationships:
- name: device
cardinality: one
peer: NetworkDevice
kind: Parent
optional: false
nodes:
- name: Device
namespace: Network
human_friendly_id: ['hostname__value']
attributes:
- name: hostname
kind: Text
unique: true
- name: model
kind: Text
relationships:
- name: interfaces
cardinality: many
peer: NetworkInterface
kind: Component
- name: PhysicalInterface
namespace: Network
inherit_from:
- NetworkInterface
attributes:
- name: speed
kind: Number
- name: LogicalInterface
namespace: Network
inherit_from:
- NetworkInterface
Create a new branch to load the new version of our schema
infrahubctl branch create network-device-generics
Load the schema in the branch
infrahubctl schema load --branch network-device-generics /tmp/schema_guide.yml
We can inspect the schema in the Web UI
Use this GraphQL mutation to create a NetworkPhysicalInterface Ethernet1
, NetworkLogicalInterface Vlan1
, and a NetworkDevice atl1-edge1
.
mutation {
NetworkDeviceCreate(data: {hostname: {value: "atl1-edge1"}, model: {value: "Cisco ASR1002-HX"}}) {
ok
object {
id
}
}
NetworkPhysicalInterfaceCreate(data: {name: {value: "Ethernet1"}, description: {value: "WAN interface"}, speed: {value: 1000000000}, device: {hfid: "atl1-edge1"}}) {
ok
object {
id
}
}
NetworkLogicalInterfaceCreate(data: {name: {value: "Vlan1"}, description: {value: "SVI for VLAN 1"}, device: {hfid: "atl1-edge1"}}) {
ok
object {
id
}
}
}
In the detailed view of the device in the Web UI, we can now see that the device has 2 interfaces, but they are of a different kind.
4. Improving our schema
Although the schema already closely represents what we wanted to achieve, there are still a few improvements we would like to make. For this, we are going to use Infrahub's schema migrations feature. More details can be found in the Schema topic.
- adding and
mtu
andenabled
attribute on the genericNetworkInterface
- deleting the
description
attribute of the genericNetworkInterface
- setting a default value for the
speed
attribute ofNetworkPhysicalInterface
- renaming the
model
attribute of aNetworkDevice
todevice_type
- add labels for all the attributes and relationships
- a unique constraint needs to be defined on the
NetworkInterface
generic on the name attribute and device relationship
We will be reusing the branch from the previous step.
Overwrite the contents of the schema_guide.yml
file with this content:
To do this nation, we must provide the id
of the attribute and the new name. You will have to grab the id
of the model attribute of the NetworkDevice node from the schema page in the Web UI.
---
version: "1.0"
generics:
- name: Interface
namespace: Network
attributes:
- name: name
kind: Text
label: Name
- name: description
state: absent
kind: Text
optional: true
label: Description
- name: mtu
kind: Number
label: MTU
optional: false
default_value: 1500
- name: enabled
label: Enabled
kind: Boolean
optional: false
default_value: false
relationships:
- name: device
label: Device
cardinality: one
peer: NetworkDevice
kind: Parent
optional: false
nodes:
- name: Device
namespace: Network
human_friendly_id: ['hostname__value']
attributes:
- name: hostname
kind: Text
label: Hostname
unique: true
- name: device_type
label: Device Type
kind: Text
id: 17bcf8a7-9c03-4a6a-3295-c51345cb1c33
relationships:
- name: interfaces
label: Interfaces
cardinality: many
peer: NetworkInterface
kind: Component
- name: PhysicalInterface
namespace: Network
uniqueness_constraints:
- ["device", "name__value"]
inherit_from:
- NetworkInterface
attributes:
- name: speed
label: Speed (bps)
kind: Number
default_value: 1000000000
- name: LogicalInterface
namespace: Network
uniqueness_constraints:
- ["device", "name__value"]
inherit_from:
- NetworkInterface
Use the infrahubctl schema check
command to validate the schema and to get a diff
of the schema changes that will be performed
infrahubctl schema check --branch network-device-generics /tmp/schema_guide.yml
schema '/tmp/schema_guide.yml' is Valid!
diff:
added: {}
changed:
NetworkDevice:
added: {}
changed:
attributes:
added: {}
changed:
device_type:
added: {}
changed:
label: null
name: null
removed: {}
removed: {}
removed: {}
NetworkInterface:
added: {}
changed:
attributes:
added:
enabled: null
mtu: null
changed: {}
removed:
description: null
removed: {}
NetworkLogicalInterface:
added: {}
changed:
attributes:
added:
enabled: null
mtu: null
changed: {}
removed: {}
uniqueness_constraints: null
removed: {}
NetworkPhysicalInterface:
added: {}
changed:
attributes:
added:
enabled: null
mtu: null
changed:
speed:
added: {}
changed:
default_value: null
label: null
optional: null
removed: {}
removed: {}
uniqueness_constraints: null
removed: {}
removed: {}
Load the schema into network-device-generics
branch in Infrahub
infrahubctl schema load --branch network-device-generics /tmp/schema_guide.yml
We can inspect the schema in the Web UI
Notice that besides the schema, also all the existing data in this branch has been migrated to adhere to the new version of the schema.