Creating a Jinja rendered file (transform)
Within Infrahub a Transform is defined in an external repository. However, during development and troubleshooting it is easiest to start from your local computer and run the render using infrahubctl render.
The goal of this guide is to develop a Jinja Transform and add it to Infrahub, we will achieve this by following these steps.
- Identify the relevant data you want to extract from the database using a GraphQL query, that can take an input parameter to filter the data
- Write a Jinja2 file that use the GraphQL query to read information from the system and render the data into a new format
- Create an entry for the Jinja2 Transform within an .infrahub.yml file.
- Create a Git repository
- Test the transform rendering with infrahubctl
- Add the repository to Infrahub as an external repository
- Validate that the transform works by using the render API endpoint
In this guide we are going to work with the builtin tag objects in Infrahub. It won't provide a rendered template that is very useful, the goal is instead to show how the Jinja Rendering works. Once you have mastered the basics you will be ready to go on to create more advanced template.
1. Creating a query to collect the desired data
As the first step we need to have some data in the database to actually query.
Create three tags, called "red", "green", "blue", either using the frontend or by submitting three GraphQL mutations as per below (just swapping out the name of the color each time).
mutation CreateTags {
BuiltinTagCreate(
data: {name: {value: "red"}, description: {value: "The red tag"}}
) {
ok
object {
id
}
}
}
The next step is to create a query that returns the data we just created. The rest of this guide assumes that the following query will return a response similar to the response below the query.
query TagsQuery {
BuiltinTag {
edges {
node {
name {
value
}
description {
value
}
}
}
}
}
Response to the tags query:
{
"data": {
"BuiltinTag": {
"edges": [
{
"node": {
"name": {
"value": "blue"
},
"description": {
"value": "The blue tag"
}
}
},
{
"node": {
"name": {
"value": "green"
},
"description": {
"value": "The green tag"
}
}
},
{
"node": {
"name": {
"value": "red"
},
"description": {
"value": "The red tag"
}
}
}
]
}
}
}
While it would be possible to create a transform that targets all of these tags, for example if you want to create a report, the goal for us is to be able to focus on one of these objects. For this reason we need to modify the query from above to take an input parameter so that we can filter the result to what we want.
Create a local directory on your computer.
mkdir tags_render
Then save the below query as a text file named tags_query.gql.
query TagsQuery($tag: String!) {
BuiltinTag(name__value: $tag) {
edges {
node {
name {
value
}
description {
value
}
}
}
}
}
Here the query will require an input parameter called $name
what will refer to the name of each tag. When we want to query for the red tag the input variables to the query would look like this:
{
"tag": "red"
}
2. Create the Jinja template
The next step is to create the actual Jinja Template file. Create a file called tags_tpl.j2
{% if data.BuiltinTag.edges and data.BuiltinTag.edges is iterable %}
{% for tag in data["BuiltinTag"]["edges"] %}
{% set tag_name = tag.node.name.value %}
{% set tag_description = tag.node.description.value %}
{{ tag_name }}
description: {{ tag_description }}
{% endfor %}
{% endif %}
3. Create a .infrahub.yml file
In the .infrahub.yml file you define what transforms you have in your repository that you want to make available for Infrahub.
Create a .infrahub.yml file in the root of the directory.
---
jinja2_transforms:
- name: my-jinja2-transform # Unique name for your transform
description: "short description" # (optional)
query: "tags_query" # Name or ID of the GraphQLQuery
template_path: "tags_tpl.j2" # Path to the main Jinja2 template
The main Jinja2 template can import other templates
Three parts here are required, first the name
of the transform which should be unique across Infrahub, query
the GraphqlQuery linked to our transform and also the template_path
that should point to the Jinja2 file within the repository.
4. Create a Git repository
Within the tags_render
folder you should now have tree files:
- tags_query.gql: Contains the GraphQL query
- tags_tpl.j2: Contains the Jinja2 Template
- .infrahub.yml: Contains the definition for the transform
Before we can test our transform we must add the files to a local Git repository.
git init --initial-branch=main
git add .
git commit -m "First commit"
5. Test the render using infrahubctl
Using infrahubctl you can first verify that the .infrahub.yml
file is formatted correctly by listing available transforms.
Usage: infrahubctl render [OPTIONS] TRANSFORM_NAME [VARIABLES]...
Render a local Jinja2 Transform for debugging purpose.
╭─ Arguments ─────────────────────────────────────────────────────────────────────────────────────╮
│ * transform_name TEXT [default: None] [required] │
│ variables [VARIABLES]... Variables to pass along with the query. Format │
│ key=value key=value. │
│ [default: None] │
╰─────────────────────────────────────────────────────────────────────────────────────────────────╯
╭─ Options ───────────────────────────────────────────────────────────────────────────────────────╮
│ --branch TEXT Branch on which to render the transform. [default: None] │
│ --debug --no-debug [default: no-debug] │
│ --config-file TEXT [env var: INFRAHUBCTL_CONFIG] [default: infrahubctl.toml] │
│ --help Show this message and exit. │
╰─────────────────────────────────────────────────────────────────────────────────────────────────╯
Examples
infrahubctl render <transform name or ID> my-param=XXXXX my-other-param=YYYYY
If --branch
is not provided it will automatically use the name of the local branch.
6. Adding the repository to Infrahub
In order to avoid having the same instructions over and over please refer to the guide adding a repository to Infrahub in order to sync the repository you created and make it available within Infrahub.
7. Accessing the Transform from the API
A transform can be rendered on demand via the REST API with the endpoint: https://<host>/api/transform/jinja2/<transform name or ID>
This endpoint is branch-aware and it accepts the name of the branch and/or the time as URL parameters.
https://<host>/api/transform/jinja2/<transform name or ID>?branch=main
https://<host>/api/transform/jinja2/<transform name or ID>?branch=main&at=<time of your choice>
The name of the branch used in the query will be used to retrieve the right Jinja template and to execute the GraphQL query.
If the GraphQL query accepts parameters, they can be passed directly as URL parameters:
https://<host>/api/transform/jinja2/<transform name or ID>?branch=main&my-param=XXX&my-other-param=YYY