name: infrahub-managing-transforms description: >- Creates Infrahub transforms that convert data into JSON, text, CSV, or device configs using Python or Jinja2 templates, with YAML-driven tests. TRIGGER when: building config generation, data export, format conversion, Jinja2 templates, artifact pipelines, writing or running tests for a transform. DO NOT TRIGGER when: designing schemas, writing validation checks, creating generators, querying live data. allowed-tools: - Read - Write - Edit - Bash - Grep argument-hint: "[transform-name] [format]" metadata: version: 1.2.7 author: OpsMill
Infrahub Transform Creator
Overview
Expert guidance for creating Infrahub transforms. Transforms convert Infrahub data into different formats -- JSON, text, CSV, device configs, or any text-based output -- using Python classes or Jinja2 templates.
Project Context
Infrahub config:
!cat .infrahub.yml 2>/dev/null || echo "No .infrahub.yml found"
Existing transforms:
!find . -name "*.py" -path "*/transforms/*" -o -name "*.j2" -path "*/templates/*" 2>/dev/null | head -20
When to Use
- Building data transformations (Infrahub data -> another format)
- Generating device configurations from infrastructure data
- Creating CSV reports, cable matrices, or inventory exports
- Rendering Jinja2 templates with query data
- Combining Python logic with Jinja2 rendering
- Connecting transforms to artifacts for automated output
Rule Categories
| Priority | Category | Prefix | Description |
|---|---|---|---|
| CRITICAL | Types | types- |
Python vs Jinja2 choice |
| CRITICAL | Python | python- |
InfrahubTransform class |
| CRITICAL | Jinja2 | jinja2- |
Template syntax, filters |
| HIGH | Hybrid | hybrid- |
Python + Jinja2 combined |
| HIGH | Artifacts | artifacts- |
Output files, targets |
| HIGH | API Ref | api- |
Class attrs, methods |
| MEDIUM | Patterns | patterns- |
Utilities, CSV, shared |
| HIGH | Testing | testing- |
Resources Testing Framework, transform/render commands |
Schema Features This Skill Depends On
A transform reads schema-shaped data and produces a file. Misalignment between the transform and the schema fails late — at artifact-render time, when someone is waiting for the output.
| If the transform... | The schema (or .infrahub.yml) must... | See |
|---|---|---|
Will feed an artifact_definitions entry |
The target node must inherit_from: CoreArtifactTarget so the artifact pipeline can attach to it |
../infrahub-managing-schemas/rules/extension-artifact-target.md |
| Reads attributes from a node | Define those attributes with their full __value access path in GraphQL — silent empty strings come from accessing the node, not the value |
../infrahub-managing-schemas/rules/attribute-defaults-and-types.md |
| Picks a template per device by platform/role | The schema must expose that platform/role as a real attribute or relationship — string-matching on display_label is brittle |
../infrahub-managing-schemas/rules/display-human-friendly-id.md |
Is referenced from artifact_definitions.transformation |
The transform's registered name must match the transformation: field exactly — mismatch produces "transformation not found" at render time |
rules/artifacts-definitions.md |
| Uses Jinja2 (not Python) | Register under jinja2_transforms with a top-level query: field — python_transforms binds query on the class, the two keys are not interchangeable |
rules/api-reference.md |
Transform Basics
Two types of transforms:
| Type | Output | Entry Point |
|---|---|---|
| Python | JSON/dict or text | InfrahubTransform.transform() |
| Jinja2 | Text | .j2 template file |
from infrahub_sdk.transforms import InfrahubTransform
class MyTransform(InfrahubTransform):
query = "my_query"
async def transform(self, data: dict) -> dict:
device = data["DcimDevice"]["edges"][0]["node"]
return {"hostname": device["name"]["value"]}
Workflow
Follow these steps when creating a transform:
- Choose the transform type — Python for JSON/dict or complex logic, Jinja2 for text templates, hybrid for both. Read rules/types-overview.md.
- Write the GraphQL query — Create a
.gqlfile that fetches the data to transform. Read ../infrahub-common/graphql-queries.md for query patterns. - Implement the transform — For Python, inherit
from
InfrahubTransformand implementtransform(). Read rules/python-transform.md. For Jinja2, create a.j2template. Read rules/jinja2-template.md. For hybrid, read rules/hybrid-python-jinja2.md. - Connect to artifacts — If the transform output should be stored as a file, configure artifact definitions. See rules/artifacts-definitions.md.
- Register in .infrahub.yml — Add under
python_transformsorjinja2_transforms. See rules/api-reference.md. - Add tests — Create YAML-driven test definitions (smoke, unit, integration) alongside the transform so it is validated automatically in the proposed change pipeline. Read rules/testing-resource-framework.md.
- Test locally — Run
infrahubctl transformorinfrahubctl renderto validate. See rules/testing-commands.md.
Supporting References
- reference.md -- Class API,
lifecycle, return-type matrix,
.infrahub.ymlregistration shapes, filter env overview - examples.md -- Complete transform patterns (Python, Jinja2, hybrid, CSV)
- ../infrahub-common/graphql-queries.md -- GraphQL query writing reference
- infrahub-yml-reference.md -- .infrahub.yml project configuration
- ../infrahub-common/rules/ -- Shared rules (git integration, caching) across all skills
- ../infrahub-managing-schemas/SKILL.md -- Schema definitions transforms work with
- rules/ -- Individual rules by category