- Bump all 18 plugins from 0.110.0 to 0.111.0
- Update rust-toolchain.toml channel to 1.93.1 (nu 0.111.0 requires ≥1.91.1)
Fixes:
- interprocess pin =2.2.x → ^2.3.1 in nu_plugin_mcp, nu_plugin_nats, nu_plugin_typedialog
(required by nu-plugin-core 0.111.0)
- nu_plugin_typedialog: BackendType::Web initializer — add open_browser: false field
- nu_plugin_auth: implement missing user_info_to_value helper referenced in tests
Scripts:
- update_all_plugins.nu: fix [package].version update on minor bumps; add [dev-dependencies]
pass; add nu-plugin-test-support to managed crates
- download_nushell.nu: rustup override unset before rm -rf on nushell dir replace;
fix unclosed ) in string interpolation
699 lines
17 KiB
Markdown
699 lines
17 KiB
Markdown
# nu_plugin_tera
|
|
|
|
A powerful [Nushell](https://nushell.sh/) plugin for rendering [Tera templates](https://keats.github.io/tera/docs/) with structured data from Nushell pipelines.
|
|
|
|
## Overview
|
|
|
|
This plugin integrates Mozilla's Tera templating engine with Nushell, enabling you to generate dynamic content from structured data. Perfect for configuration files, documentation generation, reports, and any scenario where you need to combine templates with data.
|
|
|
|
## Installing
|
|
|
|
Clone this repository
|
|
|
|
> [!WARNING]
|
|
> **nu_plugin_tera** has dependencies to nushell source via local path in Cargo.toml
|
|
> Nushell and plugins require to be **sync** with same **version**
|
|
|
|
Clone [Nushell](https://nushell.sh/) alongside this plugin or change dependencies in [Cargo.toml](Cargo.toml)
|
|
|
|
This plugin is also included as submodule in [nushell-plugins](https://repo.jesusperez.pro/jesus/nushell-plugins)
|
|
as part of plugins collection for [Provisioning project](https://rlung.librecloud.online/jesus/provisioning)
|
|
|
|
Build from source
|
|
|
|
```nushell
|
|
> cd nu_plugin_tera
|
|
> cargo install --path .
|
|
```
|
|
|
|
### Nushell
|
|
|
|
In a [Nushell](https://nushell.sh/)
|
|
|
|
```nushell
|
|
> plugin add ~/.cargo/bin/nu_plugin_tera
|
|
```
|
|
|
|
## Commands
|
|
|
|
### `tera-render`
|
|
|
|
Render Tera templates with structured data from Nushell pipelines or arguments.
|
|
|
|
```nushell
|
|
> tera-render <template> [context]
|
|
```
|
|
|
|
**Parameters:**
|
|
|
|
- **template** `<path>`: Path to the `.tera` template file or directory (with `--directory`)
|
|
- **context** `<any>`: Context data (record or JSON file path) (optional)
|
|
|
|
**Flags:**
|
|
|
|
- **--directory**: Load all templates from a directory (enables `{% include %}`, `{% extends %}`, macros)
|
|
- **--entry-point** `<string>`: Template to render in directory mode (default: `main.j2`, fallback: `index.j2`)
|
|
- **-h**, **--help**: Display the help message for this command
|
|
|
|
## Template Syntax
|
|
|
|
Tera uses a Jinja2-inspired syntax with powerful features:
|
|
|
|
### Basic Variables
|
|
|
|
```jinja2
|
|
Hello, {{ name }}!
|
|
Your age is {{ age }}.
|
|
```
|
|
|
|
### Control Structures
|
|
|
|
```jinja2
|
|
{% if user.is_admin %}
|
|
<p>Welcome, admin!</p>
|
|
{% else %}
|
|
<p>Welcome, user!</p>
|
|
{% endif %}
|
|
|
|
{% for item in items %}
|
|
- {{ item.name }}: {{ item.value }}
|
|
{% endfor %}
|
|
```
|
|
|
|
### Filters
|
|
|
|
```jinja2
|
|
{{ name | upper }}
|
|
{{ price | round(precision=2) }}
|
|
{{ date | date(format="%Y-%m-%d") }}
|
|
```
|
|
|
|
### Macros
|
|
|
|
```jinja2
|
|
{% macro render_field(name, value) %}
|
|
<div class="field">
|
|
<label>{{ name }}</label>
|
|
<span>{{ value }}</span>
|
|
</div>
|
|
{% endmacro %}
|
|
|
|
{{ render_field(name="Username", value=user.name) }}
|
|
```
|
|
|
|
## Usage Examples
|
|
|
|
### Basic Usage
|
|
|
|
**data.json**
|
|
|
|
```json
|
|
{
|
|
"name": "Akasha",
|
|
"projects": [
|
|
{
|
|
"name": "TheProject",
|
|
"status": "active"
|
|
},
|
|
{
|
|
"name": "AnotherProject",
|
|
"status": "completed"
|
|
}
|
|
]
|
|
}
|
|
```
|
|
|
|
**template.tera**
|
|
|
|
```jinja2
|
|
Hello, {{ name }}!
|
|
|
|
Projects:
|
|
{% for project in projects -%}
|
|
- {{ project.name }} ({{ project.status }})
|
|
{% endfor %}
|
|
```
|
|
|
|
**Rendering:**
|
|
|
|
```nushell
|
|
> open data.json | tera-render template.tera
|
|
Hello, Akasha!
|
|
|
|
Projects:
|
|
- TheProject (active)
|
|
- AnotherProject (completed)
|
|
```
|
|
|
|
### Pipeline Data Processing
|
|
|
|
```nushell
|
|
# Process system information
|
|
> sys | tera-render system-report.tera
|
|
|
|
# Generate configuration files from structured data
|
|
> {
|
|
server: {
|
|
host: "localhost",
|
|
port: 8080,
|
|
ssl: true
|
|
},
|
|
database: {
|
|
host: "db.example.com",
|
|
name: "myapp"
|
|
}
|
|
} | tera-render config.tera
|
|
|
|
# Process CSV data
|
|
> open employees.csv
|
|
| tera-render employee-report.tera
|
|
```
|
|
|
|
### Multiple Data Sources
|
|
|
|
```nushell
|
|
# Using JSON file as context
|
|
> tera-render template.tera data.json
|
|
|
|
# Using record from pipeline
|
|
> { name: 'Akasha', projects: [ {'name': 'TheProject', 'status': 'active'} ] }
|
|
| tera-render template.tera
|
|
|
|
# Wrapping data for template access
|
|
> open data.json | wrap value | tera-render template.tera
|
|
```
|
|
|
|
## Template Composition with Directory Mode
|
|
|
|
The `--directory` flag enables true template composition using Tera's include and extends features. Load all templates from a directory and use advanced template patterns.
|
|
|
|
### Directory Structure
|
|
|
|
```plaintext
|
|
templates/
|
|
├── base.j2 # Base layout template
|
|
├── includes/
|
|
│ ├── _header.j2 # Reusable header component
|
|
│ ├── _footer.j2 # Reusable footer component
|
|
│ └── _helpers.j2 # Shared macros and filters
|
|
└── main.j2 # Main template using includes
|
|
```
|
|
|
|
### Usage with Directory Mode
|
|
|
|
```nushell
|
|
# Single render with ALL templates loaded (enables includes, extends, macros)
|
|
> {
|
|
title: "My Report",
|
|
sections: [
|
|
{ name: "Features", items: ["fast", "reliable"] }
|
|
]
|
|
} | tera-render templates --directory
|
|
|
|
# Renders main.j2 with all includes and macros loaded
|
|
```
|
|
|
|
### Infrastructure-as-Code Example
|
|
|
|
**templates/hetzner/** Directory structure:
|
|
```plaintext
|
|
templates/hetzner/
|
|
├── base.j2 # Shared macros
|
|
├── includes/
|
|
│ ├── _ssh_keys.j2 # SSH key setup
|
|
│ ├── _networks.j2 # Network creation
|
|
│ └── _firewalls.j2 # Firewall rules
|
|
└── server.j2 # Main orchestration
|
|
```
|
|
|
|
**templates/hetzner/base.j2**
|
|
```jinja2
|
|
{% macro setup_ssh_key(name, public_key) %}
|
|
hcloud ssh-key create --name "{{ name }}" --public-key "{{ public_key }}"
|
|
{% endmacro %}
|
|
|
|
{% macro create_network(name, ip_range) %}
|
|
hcloud network create --name "{{ name }}" --ip-range "{{ ip_range }}"
|
|
{% endmacro %}
|
|
```
|
|
|
|
**templates/hetzner/server.j2**
|
|
```jinja2
|
|
#!/bin/bash
|
|
set -euo pipefail
|
|
|
|
{% include "includes/_ssh_keys.j2" %}
|
|
{% include "includes/_networks.j2" %}
|
|
{% include "includes/_firewalls.j2" %}
|
|
|
|
hcloud server create ...
|
|
```
|
|
|
|
**Usage:**
|
|
```nushell
|
|
> { ssh_key: {...}, network: {...}, server: {...} }
|
|
| tera-render templates/hetzner --directory
|
|
| save server-setup.sh
|
|
```
|
|
|
|
Result: Single atomic script with all sections composed from DRY templates via includes and shared macros.
|
|
|
|
## Custom Entry Points
|
|
|
|
The `--entry-point` parameter allows you to select different templates within the same directory. This enables multiple configurations or deployment scenarios from a single template library.
|
|
|
|
### Use Cases
|
|
|
|
```plaintext
|
|
templates/deployment/
|
|
├── base.j2 # Shared macros for all configurations
|
|
├── dev.j2 # Development environment config
|
|
├── staging.j2 # Staging environment config
|
|
├── prod.j2 # Production environment config
|
|
├── includes/
|
|
│ ├── _setup.j2 # Common setup steps
|
|
│ └── _cleanup.j2 # Common cleanup steps
|
|
└── main.j2 # Default (used when no entry-point specified)
|
|
```
|
|
|
|
### Examples
|
|
|
|
**Render default template (main.j2):**
|
|
```nushell
|
|
> { env: "production" } | tera-render templates/deployment --directory
|
|
# Uses: main.j2
|
|
```
|
|
|
|
**Render development configuration:**
|
|
```nushell
|
|
> { env: "development", debug: true } | tera-render templates/deployment --directory --entry-point dev.j2
|
|
# Uses: dev.j2 (with development-specific settings)
|
|
```
|
|
|
|
**Render production configuration:**
|
|
```nushell
|
|
> { env: "production", replicas: 5 } | tera-render templates/deployment --directory --entry-point prod.j2
|
|
# Uses: prod.j2 (with production-specific settings)
|
|
```
|
|
|
|
### Template Structure Example
|
|
|
|
**templates/deployment/base.j2**
|
|
```jinja2
|
|
{% macro deploy_service(name, replicas) %}
|
|
echo "Deploying {{ name }} with {{ replicas }} replicas"
|
|
{% endmacro %}
|
|
|
|
{% macro setup_monitoring() %}
|
|
# Common monitoring setup
|
|
{% endmacro %}
|
|
```
|
|
|
|
**templates/deployment/dev.j2**
|
|
```jinja2
|
|
{% import "base.j2" as base %}
|
|
{% include "includes/_setup.j2" %}
|
|
|
|
{{ base::deploy_service(name=service, replicas=1) }}
|
|
{{ base::setup_monitoring() }}
|
|
|
|
# Development-specific settings
|
|
DEBUG=true
|
|
LOG_LEVEL=debug
|
|
```
|
|
|
|
**templates/deployment/prod.j2**
|
|
```jinja2
|
|
{% import "base.j2" as base %}
|
|
{% include "includes/_setup.j2" %}
|
|
|
|
{{ base::deploy_service(name=service, replicas=replicas) }}
|
|
{{ base::setup_monitoring() }}
|
|
|
|
# Production-specific settings
|
|
DEBUG=false
|
|
LOG_LEVEL=warn
|
|
REPLICAS={{ replicas }}
|
|
```
|
|
|
|
### Benefits
|
|
|
|
| Feature | Default | Custom Entry Point |
|
|
|---------|---------|-------------------|
|
|
| Reuse macros/includes | ✅ | ✅ |
|
|
| Multiple configurations | ❌ | ✅ |
|
|
| Dev/staging/prod variants | ❌ | ✅ |
|
|
| Single template directory | ✅ | ✅ |
|
|
| DRY principle | ✅ | ✅ |
|
|
|
|
## Advanced Templates
|
|
|
|
### Configuration File Generation
|
|
|
|
**nginx.conf.tera**
|
|
|
|
```jinja2
|
|
server {
|
|
listen {{ server.port }}{% if server.ssl %} ssl{% endif %};
|
|
server_name {{ server.host }};
|
|
|
|
{% if server.ssl %}
|
|
ssl_certificate {{ ssl.cert_path }};
|
|
ssl_certificate_key {{ ssl.key_path }};
|
|
{% endif %}
|
|
|
|
{% for location in locations %}
|
|
location {{ location.path }} {
|
|
proxy_pass {{ location.upstream }};
|
|
{% for header in location.headers | default(value=[]) %}
|
|
proxy_set_header {{ header.name }} {{ header.value }};
|
|
{% endfor %}
|
|
}
|
|
{% endfor %}
|
|
}
|
|
```
|
|
|
|
**Usage:**
|
|
|
|
```nushell
|
|
> {
|
|
server: {
|
|
host: "example.com",
|
|
port: 443,
|
|
ssl: true
|
|
},
|
|
ssl: {
|
|
cert_path: "/etc/ssl/cert.pem",
|
|
key_path: "/etc/ssl/key.pem"
|
|
},
|
|
locations: [
|
|
{
|
|
path: "/api/",
|
|
upstream: "http://backend:8080",
|
|
headers: [
|
|
{name: "Host", value: "$host"},
|
|
{name: "X-Real-IP", value: "$remote_addr"}
|
|
]
|
|
}
|
|
]
|
|
} | tera-render nginx.conf.tera | save nginx.conf
|
|
```
|
|
|
|
### Documentation Generation
|
|
|
|
**api-docs.md.tera**
|
|
|
|
```jinja2
|
|
# {{ api.title }} API Documentation
|
|
|
|
Version: {{ api.version }}
|
|
Last updated: {{ "now" | date(format="%Y-%m-%d") }}
|
|
|
|
## Endpoints
|
|
|
|
{% for endpoint in api.endpoints %}
|
|
### {{ endpoint.method | upper }} {{ endpoint.path }}
|
|
|
|
{{ endpoint.description }}
|
|
|
|
{% if endpoint.parameters %}
|
|
**Parameters:**
|
|
{% for param in endpoint.parameters %}
|
|
- `{{ param.name }}` ({{ param.type }}){% if param.required %} *required*{% endif %}: {{ param.description }}
|
|
{% endfor %}
|
|
{% endif %}
|
|
|
|
**Example Response:**
|
|
```
|
|
{{ endpoint.example_response | tojson }}
|
|
```
|
|
|
|
---
|
|
{% endfor %}
|
|
```
|
|
|
|
### Report Generation
|
|
|
|
```nushell
|
|
# Generate system report
|
|
def generate-system-report [] {
|
|
let data = {
|
|
timestamp: (date now | format date "%Y-%m-%d %H:%M:%S"),
|
|
system: (sys),
|
|
processes: (ps | first 10),
|
|
disk_usage: (df)
|
|
}
|
|
|
|
$data | tera-render system-report.html.tera | save $"reports/system-report-(date now | format date "%Y%m%d").html"
|
|
}
|
|
|
|
# Generate project status report
|
|
def generate-project-report [project_dir: string] {
|
|
let data = {
|
|
project: (open $"($project_dir)/package.json" | get name),
|
|
files: (ls $project_dir | length),
|
|
git_status: (git -C $project_dir status --porcelain | lines),
|
|
last_commit: (git -C $project_dir log -1 --format="%h %s %an %ad" --date=short)
|
|
}
|
|
|
|
$data | tera-render project-report.md.tera
|
|
}
|
|
```
|
|
|
|
## Workflow Examples
|
|
|
|
### CI/CD Pipeline Configuration
|
|
|
|
```nushell
|
|
# Generate Kubernetes manifests
|
|
def generate-k8s-manifests [env: string] {
|
|
let config = (open $"configs/($env).json")
|
|
|
|
[
|
|
"deployment.yaml.tera",
|
|
"service.yaml.tera",
|
|
"ingress.yaml.tera"
|
|
] | each { |template|
|
|
let output = ($config | tera-render $"templates/($template)")
|
|
$output | save $"manifests/($env)/(($template | str replace '.tera' '')))"
|
|
}
|
|
}
|
|
|
|
# Generate environment-specific configs
|
|
def generate-env-configs [] {
|
|
["dev", "staging", "prod"] | each { |env|
|
|
let base_config = (open base-config.json)
|
|
let env_config = (open $"($env)-config.json")
|
|
let merged = ($base_config | merge $env_config)
|
|
|
|
$merged | tera-render app-config.yaml.tera | save $"configs/($env).yaml"
|
|
}
|
|
}
|
|
```
|
|
|
|
### Content Management
|
|
|
|
```nushell
|
|
# Generate blog posts from structured data
|
|
def generate-blog-posts [] {
|
|
ls content/*.json
|
|
| each { |post_file|
|
|
let post_data = (open $post_file.name)
|
|
let filename = ($post_file.name | path basename | str replace '.json' '.html')
|
|
|
|
$post_data | tera-render blog-post.html.tera | save $"output/($filename)"
|
|
}
|
|
}
|
|
|
|
# Generate navigation menus
|
|
def generate-navigation [site_config: string] {
|
|
open $site_config
|
|
| get pages
|
|
| tera-render navigation.html.tera
|
|
| save includes/navigation.html
|
|
}
|
|
```
|
|
|
|
### Infrastructure as Code
|
|
|
|
```nushell
|
|
# Generate Terraform configurations
|
|
def generate-terraform [environment: string] {
|
|
let vars = (open $"terraform/($environment).tfvars.json")
|
|
|
|
ls terraform/templates/*.tf.tera
|
|
| each { |template|
|
|
let output_name = ($template.name | path basename | str replace '.tera' '')
|
|
$vars | tera-render $template.name | save $"terraform/($environment)/($output_name)"
|
|
}
|
|
}
|
|
|
|
# Generate Docker Compose files
|
|
def generate-docker-compose [services: list] {
|
|
let compose_data = {
|
|
services: $services,
|
|
networks: ["app-network"],
|
|
volumes: ["db-data", "app-logs"]
|
|
}
|
|
|
|
$compose_data | tera-render docker-compose.yml.tera | save docker-compose.yml
|
|
}
|
|
```
|
|
|
|
## Integration with Nushell Data Processing
|
|
|
|
Leverage Nushell's powerful data manipulation with Tera templating:
|
|
|
|
```nushell
|
|
# Process and template CSV data
|
|
> open sales.csv
|
|
| group-by region
|
|
| transpose region sales
|
|
| insert total { |row| $row.sales | math sum }
|
|
| tera-render sales-report.html.tera
|
|
| save reports/sales-summary.html
|
|
|
|
# Combine multiple data sources
|
|
> let weather = (http get "https://api.weather.com/current")
|
|
> let system = (sys)
|
|
> {weather: $weather, system: $system}
|
|
| tera-render dashboard.html.tera
|
|
|
|
# Batch process templates
|
|
> ls templates/*.tera
|
|
| each { |template|
|
|
let data = (open $"data/(($template.name | path basename | str replace '.tera' '.json'))")
|
|
$data | tera-render $template.name
|
|
}
|
|
```
|
|
|
|
## Error Handling
|
|
|
|
The plugin provides detailed error messages for common issues:
|
|
|
|
```nushell
|
|
# Template syntax errors
|
|
> {} | tera-render broken-template.tera
|
|
Error: Template syntax error
|
|
╭─[calling tera-render]
|
|
│ Template syntax error: Unexpected token...
|
|
|
|
# Missing variables
|
|
> {} | tera-render template-with-vars.tera
|
|
Error: Template render error
|
|
╭─[calling tera-render]
|
|
│ Variable 'name' not found in context
|
|
|
|
# File not found
|
|
> {} | tera-render nonexistent.tera
|
|
Error: Failed to read template file
|
|
╭─[calling tera-render]
|
|
│ Failed to read file 'nonexistent.tera': No such file...
|
|
```
|
|
|
|
## Features
|
|
|
|
- ✅ **Pipeline Integration** - Seamlessly process Nushell data structures
|
|
- ✅ **Multiple Data Sources** - Support for JSON files, records, and pipeline data
|
|
- ✅ **Rich Template Syntax** - Full Tera templating features (filters, macros, inheritance)
|
|
- ✅ **Error Handling** - Detailed error messages with context
|
|
- ✅ **File I/O** - Read templates from files, save rendered output
|
|
- ✅ **Data Wrapping** - Automatic handling of different data structures
|
|
- ✅ **Performance** - Efficient template compilation and rendering
|
|
|
|
## Use Cases
|
|
|
|
- **Configuration Management**: Generate environment-specific config files
|
|
- **Documentation**: Create dynamic documentation from structured data
|
|
- **Infrastructure as Code**: Template Terraform, Kubernetes, Docker configurations
|
|
- **Reports**: Generate HTML/Markdown reports from system data
|
|
- **Web Development**: Template generation for static sites
|
|
- **CI/CD**: Dynamic pipeline configurations
|
|
- **Content Management**: Blog posts, newsletters, and content generation
|
|
- **Monitoring**: Dashboard and alert template generation
|
|
|
|
## Template Best Practices
|
|
|
|
### 1. Organize Templates
|
|
|
|
```plaintext
|
|
templates/
|
|
├── base.html.tera # Base layouts
|
|
├── components/
|
|
│ ├── header.html.tera # Reusable components
|
|
│ └── footer.html.tera
|
|
├── pages/
|
|
│ ├── home.html.tera # Page templates
|
|
│ └── about.html.tera
|
|
└── configs/
|
|
├── nginx.conf.tera # Configuration templates
|
|
└── docker.yml.tera
|
|
```
|
|
|
|
### 2. Use Template Inheritance
|
|
|
|
**base.html.tera**
|
|
|
|
```jinja2
|
|
<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<title>{% block title %}{{ site.title }}{% endblock %}</title>
|
|
</head>
|
|
<body>
|
|
{% block content %}{% endblock %}
|
|
</body>
|
|
</html>
|
|
```
|
|
|
|
**page.html.tera**
|
|
|
|
```jinja2
|
|
{% extends "base.html.tera" %}
|
|
|
|
{% block title %}{{ page.title }} - {{ site.title }}{% endblock %}
|
|
|
|
{% block content %}
|
|
<h1>{{ page.title }}</h1>
|
|
{{ page.content }}
|
|
{% endblock %}
|
|
```
|
|
|
|
### 3. Create Reusable Functions
|
|
|
|
```nushell
|
|
# Template rendering helper
|
|
def render-template [template: string, data: any] {
|
|
$data | tera-render $template
|
|
}
|
|
|
|
# Batch template processing
|
|
def render-all-templates [data_dir: string, template_dir: string, output_dir: string] {
|
|
ls $"($data_dir)/*.json"
|
|
| each { |data_file|
|
|
let basename = ($data_file.name | path basename | str replace '.json' '')
|
|
let template = $"($template_dir)/($basename).tera"
|
|
let output = $"($output_dir)/($basename).html"
|
|
|
|
if ($template | path exists) {
|
|
open $data_file.name | tera-render $template | save $output
|
|
print $"Generated: ($output)"
|
|
}
|
|
}
|
|
}
|
|
```
|
|
|
|
## Contributing
|
|
|
|
Contributions are welcome! Please feel free to submit issues, feature requests, or pull requests.
|
|
|
|
## License
|
|
|
|
This project is licensed under the MIT License.
|
|
|
|
## Related Projects
|
|
|
|
- [Tera](https://keats.github.io/tera/) - Template engine for Rust
|
|
- [Nushell](https://nushell.sh/) - A new type of shell
|
|
- [nu_plugin_fluent](../nu_plugin_fluent/) - Fluent localization plugin for Nushell
|