Skip to main content

A simple templates loader

Project description

Stencils — Lightweight Template Loader for Python

PyPI - Version PyPI - Python Version GitHub License PyPI - Wheel PyPI - Downloads

Overview

Stencils is a minimal, zero-dependency Python library for managing text templates embedded in files.
It’s ideal for projects that need small, human-readable, and reusable template snippets (e.g., config fragments, codegen templates, or documentation blocks).

The library parses .tmpl (or any text) files containing named template blocks, resolves duplicates, and provides a clean dictionary-like interface for use in your code.

Installation

pip install stencils

Template Format

A single file can contain multiple named template blocks.
Each block must start with a line of the form:

===== TEMPLATE:<name> =====

and must end with:

===== /TEMPLATE =====

Example:

===== TEMPLATE:greeting =====
Hello {name}!
Welcome to {place}.
===== /TEMPLATE =====

===== TEMPLATE:farewell =====
Goodbye {name}, see you soon!
===== /TEMPLATE =====

Usage

Loading Templates

from stencils import load, render

templates = load("templates/example.tmpl")

print(templates["greeting"])
# Output:
# Hello {name}!
# Welcome to {place}.

You can also load multiple files, directories, or patterns:

templates = load(["./templates", "common/*.tmpl"])

Each .tmpl file is scanned for TEMPLATE: blocks.
Duplicate keys across files will raise TemplateConflictError unless the templates are identical.

Rendering Templates

You can render any loaded template with a dictionary of values:

text = render(templates["greeting"], {"name": "Alice", "place": "Wonderland"})
print(text)

Output:

Hello Alice!
Welcome to Wonderland.

Missing placeholders default to empty strings, so this is safe:

render(templates["farewell"], {})
# -> "Goodbye , see you soon!"

Controlling Missing or Invalid Placeholders

You can control how missing and invalid placeholders are handled via on_missing and on_error options.

render(template, data, on_missing="empty", on_error="keep")

Available options:

Parameter Option Description Example Output
on_missing "empty" (default) Replace missing keys with an empty string. "Hello , welcome!"
"keep" Keep the original placeholder text {key} if it’s missing. "Hello {name}, welcome!"
"error" Raise a KeyError for missing placeholders. Raises KeyError: 'name'
on_error "keep" (default) Keep the original placeholder if formatting fails (e.g., bad spec). "Value: {price:.2f}"
"empty" Replace invalid placeholders with an empty string. "Value: "
"error" Raise the formatting exception immediately. Raises ValueError or similar

Foreign Placeholders (${...})

Some templates may include foreign placeholders (like shell or CI variables) using ${...} syntax.
These are not interpreted by Stencils and are automatically escaped so they remain intact.

tpl = "User: {name}, shell: ${uname -u}, gha: ${{ matrix.os }}"

render(tpl, {"name": "Io"})
# -> "User: Io, shell: ${uname -u}, gha: ${{ matrix.os }}"

You can disable this behavior with protect_foreign=False if you want raw handling.

Errors

Exception Description
TemplateConflictError Raised when a template key appears in multiple files with different content.
TemplateParseError Raised on malformed templates (e.g., unclosed or nested blocks, missing start/end).

API Reference

load(targets)

Load one or more template sources (files, directories, or globs).

Args:

  • targets: Path, string, or iterable of paths/globs.

Returns:
A dict[str, str] mapping template names to their contents.
Each template is also available under an alias prefixed with a colon (:{name}).

render(template, data)

Render a template using Python’s str.format_map, defaulting missing keys to empty strings.

Args:

  • template: Template string.
  • data: Dict of substitutions.

Returns: Rendered string.

Example Project Structure

myproject/
 ├── templates/
 │    ├── main.tmpl
 │    └── footer.tmpl
 └── app.py
# app.py
from stencils import load, render

templates = load("templates")

output = render(templates["main"], {"title": "Hello World"})
print(output)

🧾 License

This project is licensed under the MIT License — see the LICENSE file for details.

© 2025 Ioannis D. (devcoons)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

stencils-0.1.8.tar.gz (6.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

stencils-0.1.8-py3-none-any.whl (7.4 kB view details)

Uploaded Python 3

File details

Details for the file stencils-0.1.8.tar.gz.

File metadata

  • Download URL: stencils-0.1.8.tar.gz
  • Upload date:
  • Size: 6.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.23

File hashes

Hashes for stencils-0.1.8.tar.gz
Algorithm Hash digest
SHA256 3394fc3771530eddadfa61802727c57454e75c42f6698ae5591c7a6be6a83e72
MD5 cbfb09980ad8477884f05a81f52cba96
BLAKE2b-256 9d35b0ccb1f5129ff043bd0062c48c9b4bbb793ec2e5a1e9813121b4367a7ca8

See more details on using hashes here.

File details

Details for the file stencils-0.1.8-py3-none-any.whl.

File metadata

  • Download URL: stencils-0.1.8-py3-none-any.whl
  • Upload date:
  • Size: 7.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.9.23

File hashes

Hashes for stencils-0.1.8-py3-none-any.whl
Algorithm Hash digest
SHA256 4cb206a57fad9b964ef8251e6541549cb8eb76c50ba449e3cde0185f0763a93a
MD5 dfe22b8253772199b1905b2fabba5a27
BLAKE2b-256 5e709af14837aced3440030db771bcb37460f9633ea939db33e0320c4bf540cb

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page