A powerful toolkit for building, managing, and composing hierarchical prompts as code.
Project description
PragmaPrompt: A toolkit for building, managing, and composing hierarchical prompts as Python code.
- Prompts as real Python (lint, test, type, ship). Get linters, formatters, type checking, unit tests, imports, and refactors—the whole Python toolchain applied to your prompts.
- Compose & standardize.
Nest prompts inside prompts, reuse modules, and snap in built-in renderers (
warning,bullets,output_example,section, …) for consistent, maintainable, and token-efficient outputs. - Context-aware rendering. Use plain Python control flow (ifs/loops/guards) to generate the right prompt for each situation—no more nasty string templating.
Installation
pip install pragma-prompt
import pragma_prompt as pp
from examples.getting_started.prompts import MyPrompts
# typed context and render model
ctx = MyPrompts.daily_motivation.context
rm = MyPrompts.daily_motivation.render_model
"# Your job is to motivate the user."
if ctx.is_user_sad:
"Go easy on the user, he is sad."
else:
"Show him some tough love, he can handle it."
with pp.section("user_data"):
f"The users name is: {rm.user_id}"
Result:
# Your job is to motivate the user.
Show him some tough love, he can handle it.
<USER_DATA>
The users name is: 123
</USER_DATA>
How It Works
At its core, PragmaPrompt executes your prompts as Python files. This means you can use the full power of Python—loops, conditionals, functions, and imports—to dynamically construct your prompts. Any bare string literal is automatically captured and rendered as a normalized text block.
This approach turns prompt engineering into a familiar software development process. You can lint, test, and version your prompts just like any other code.
for i in range(3):
f"Repetition {i+1}: Be concise and clear."
This will render:
Repetition 1: Be concise and clear.
Repetition 2: Be concise and clear.
Repetition 3: Be concise and clear.
Defining and Using Prompts
To get started, you define PromptModule and ComponentModule classes. These act as namespaces that organize your prompts and reusable components.
1. Create a Prompt Module
A PromptModule is a class that groups related prompts and provides them with shared constants.
# in examples/getting_started/prompts.py
class MyConstants:
default_tone = "helpful"
max_length = 1000
@dataclass
class MyRenderModel:
user_id: str = "123"
@dataclass
class MyContext:
text_to_summarize: str = "This is some text to summarize..."
is_user_sad: bool = False
user_is_new: bool = False
class MyComponents(pp.ComponentModule[None]):
module_dir = Path(__file__).parent / "component_files"
output_format: pp.Component[MyRenderModel]
class MyPrompts(pp.PromptModule[MyConstants]):
module_dir = Path(__file__).parent / "prompt_files"
constants = MyConstants()
daily_motivation: pp.Prompt[MyContext, MyRenderModel]
summarize: pp.Prompt[MyContext, MyRenderModel]
2. Write Your Prompt File
Your prompt file is a standard Python file. You can access the prompt's context and render_model using a tuple assignment, and the module's constants via MyPrompts.constants.
# in examples/getting_started/prompt_files/summarize.py
from examples.getting_started.prompts import MyComponents
from examples.getting_started.prompts import MyPrompts
# Get typed handles to context, render_model and constants inside an active session
ctx = MyPrompts.summarize.context
rm = MyPrompts.summarize.render_model
constants = MyPrompts.constants
f"Your tone should be {constants.default_tone}."
f"Summarize the following text for user {rm.user_id}:"
f"{ctx.text_to_summarize}"
# render reusable components
MyComponents.output_format.render()
Result:
Your tone should be helpful.
Summarize the following text for user 123:
This is some text to summarize...
<NOTICE>
You must respond in JSON.
</NOTICE>
{
"summary": "...",
"tags": [
]
}
3. Render the Prompt
To render the prompt, you call the .render() method on the prompt object, passing in the required context and render_model data.
# The data you want to inject into the prompt
@dataclass
class MyRenderModel:
user_id: str = "123"
@dataclass
class MyContext:
text_to_summarize: str = "This is some text to summarize..."
is_user_sad: bool = False
user_is_new: bool = False
if __name__ == "__main__":
MyPrompts.daily_motivation.render(
context=MyContext(),
render_model=MyRenderModel()
)
⚠️ When you call
.render(...)from module scope, guard it withif __name__ == "__main__":to avoid importing the same module recursively during rendering.
Composing Prompts with Components
Components are reusable snippets that can be imported into your prompts. They are ideal for standardizing instructions, formats, or examples.
1. Create a Component Module
Similar to a PromptModule, a ComponentModule organizes your reusable components.
# in my_app/components.py
from pathlib import Path
import pragma_prompt as pp
class MyComponents(pp.ComponentModule):
module_dir = Path(__file__).parent / "component_files"
# This component will be loaded from 'output_format.py'
output_format: pp.Component[MyRenderModel]
2. Write Your Component File
Component files are simpler than prompt files. They can access shared constants and the active render_model, but they do not receive a prompt context.
import pragma_prompt as pp
pp.warning("You must respond in JSON.")
pp.output_example({"summary": "...", "tags": []})
3. Import and Use the Component in a Prompt
You can import component modules directly into your prompt files and call the component's .render() method to insert its content.
from examples.getting_started.prompts import MyComponents
from examples.getting_started.prompts import MyPrompts
ctx = MyPrompts.summarize.context
rm = MyPrompts.summarize.render_model
constants = MyPrompts.constants
f"Your tone should be {constants.default_tone}."
f"Summarize the following text for user {rm.user_id}:"
f"{ctx.text_to_summarize}"
# Render the component to include its content
MyComponents.output_format.render()
When MyPrompts.summarize is rendered, the content of the output_format component will be included directly in the final output. This allows you to build complex prompts from standardized, reusable parts.
Renderers
block(...) — normalized text block
Signature: pp.block(content: str) -> str
What it does: Dedents and strips a string so it drops cleanly into your prompt (no leading indentation, no extra surrounding whitespace).
When to use: Any freeform text chunk (instructions, short paragraphs).
Implicit call: Bare string literals inside a prompt body are treated as block(...) automatically.
# implicit
"Summarize the discussion in 3 sentences."
# explicit (equivalent)
pp.block("Summarize the discussion in 3 sentences.")
Returns: normalized string (dedented + stripped).
bullets(...) — compact bullet list
Signatures:
pp.bullets(items: Mapping[str, Any]) -> str
pp.bullets(items: Sequence[tuple[str, Any]]) -> str
pp.bullets(items: Sequence[Any]) -> str
What it does: Renders a tight list with - markers.
- Mapping →
- key: value(order follows the mapping’s iteration order) - Sequence of
(key, value)→- key: value - Sequence of values →
- value
pp.bullets({'role': 'analyst', 'tone': 'concise'})
# - role: analyst
# - tone: concise
pp.bullets([('role', 'analyst'), ('tone', 'concise')])
# - role: analyst
# - tone: concise
pp.bullets(['Discussion', 'Serious', 'Debate'])
# - Discussion
# - Serious
# - Debate
Returns: newline-joined bullet string.
code_block(...) — fenced Markdown code
Signature: pp.code_block(source: str, lang: str | None = None) -> str
What it does: Emits a fenced Markdown code block. If lang is set, it tags the fence for syntax highlighting. If the source already contains ``` fences, it automatically switches to ```` fences.
pp.code_block("print('hi')", 'python')
# ```python
# print('hi')
# ```
Returns: fenced Markdown code block as a string.
output_example(...) — pretty JSON with inline // comments
Signature
pp.output_example(
data: LlmResponseLike,
comments: CommentTreeOrStr | None = None,
) -> str
What it does
Renders pretty JSON and lets you attach inline // comments to any node:
- Primitives: comment appears on the same line as the value.
- Objects & arrays: comment appears on the closing brace/bracket line of that node.
Comments syntax (simple & strict)
-
commentsmay be:-
a root string → comment on the whole value, or
-
a nested mapping that mirrors your data shape:
- At any object/array node you can EITHER:
- provide a single string (comment on that whole node), or
- provide a mapping of subkeys (per-field/per-index comments),
- not both at the same node.
- Arrays use numeric string indices (
"0","1", …).
- At any object/array node you can EITHER:
-
-
All comment keys/indices must exist in
data; unknown keys raiseValueError.
Examples
Root comment (whole object):
pp.output_example({'a': 1}, comments='Overall guidance')
# {
# "a": 1
# } // Overall guidance
Object-level comment:
data = {'user': {'name': 'Ada', 'age': 30}, 'count': 1}
pp.output_example(data, comments={'user': 'User metadata'})
# {
# "count": 1,
# "user": {
# "age": 30,
# "name": "Ada"
# } // User metadata
# }
Per-field comments:
schema = {'sentiment': 'positive|neutral|negative', 'summary': '', 'tags': []}
pp.output_example(
schema,
comments={'summary': '1–2 concise sentences', 'sentiment': 'pick exactly one', 'tags': '0–5 lowercase topics'},
)
# {
# "sentiment": "positive|neutral|negative", // pick exactly one
# "summary": "", // 1–2 concise sentences
# "tags": [
# ] // 0–5 lowercase topics
# }
Nested and array index comments:
data = {'user': {'name': 'Ada', 'roles': ['admin', 'ops']}, 'count': 2}
pp.output_example(
data,
comments={'user': {'name': 'Display name', 'roles': {'0': 'Primary role'}}, 'count': 'Number of items'},
)
# {
# "count": 2, // Number of items
# "user": {
# "name": "Ada", // Display name
# "roles": [
# "admin", // Primary role
# "ops"
# ]
# }
# }
Returns
A string containing pretty JSON with inline // comments. The JSON structure is valid if you strip the // … parts.
separator(...) — visual divider (optional title)
Signature
pp.separator(
title: str | None = None,
*,
char: str = "-",
width: int = 80,
) -> str
What it does
Draws a clean ASCII divider line. With a title, it centers the text and pads both sides.
Rules & defaults
widthdefaults to 80.charmust be a single visible character; defaults to"-".- When
titleis provided, emits a single centered line.
Examples
pp.separator()
# --------------------------------------------------------------------------------
pp.separator(char='=', width=10)
# ==========
pp.separator(title='CONTEXT')
# ----------------------------------- CONTEXT ------------------------------------
pp.separator(title='X', char='*', width=9)
# *** X ***
pp.separator(width=2)
# --
Returns A single string.
shot(...) — a structured, single example (prompt + reasoning + tools + output)
Signature
pp.shot(
*,
title: str | None = None,
context: LlmResponseLike | None = None,
user: str | None = None,
input: LlmResponseLike | None = None,
tools: Sequence[ToolStep] = (),
thought: str | None = None,
output: LlmResponseLike,
) -> str
What it does Emits a compact, readable “example shot”: the user prompt, optional context/input, an optional tool-call chain, optional intermediate thought, and the final output. Great for few-shot prompting and rich examples that still parse well for LLMs.
Block order (when present)
title(as a one-line heading)- User (skipped when no user message is provided)
- Context (pretty-printed, deterministic JSON from dict/dataclass/Pydantic)
- Input (pretty JSON)
- Tool call chain (each step: name, rationale, input, output, optional thought)
- Thought
- Output (pretty JSON or string)
Tool steps
ToolStep(
name: str
rationale: str | None = None
input: LlmResponseLike | None = None
output: LlmResponseLike | None = None
thought: str | None = None
)
Examples
Minimal:
pp.shot(
user="What is the capital of Switzerland?",
output="Bern",
)
# User: What is the capital of Switzerland?
# Output: Bern
Context + input:
class UserProfile(BaseModel):
username: str
expertise_level: str = Field(description="User's familiarity with the subject")
pp.shot(
user="What's the outlook for this company?",
context=UserProfile(username="investor_bob", expertise_level="beginner"),
input={"company_ticker": "TCORP", "timeframe": "6 months"},
output="The outlook is positive, but volatile.",
)
# User: What's the outlook for this company?
# Context:
# {
# "expertise_level": "beginner",
# "username": "investor_bob"
# }
# Input:
# {
# "company_ticker": "TCORP",
# "timeframe": "6 months"
# }
# Output: The outlook is positive, but volatile.
With chain-of-thought note:
pp.shot(
user="Write a short poem about the moon.",
thought="Plan: evoke night/light imagery; 3 short stanzas.",
output="Silver orb in velvet night...",
)
# User: Write a short poem about the moon.
# Thought: Plan: evoke night/light imagery; 3 short stanzas.
# Output: Silver orb in velvet night...
With a single tool call:
pp.shot(
user="Search for today's top finance news.",
tools=[
ToolStep(
name="web_search",
rationale="User asked for current news.",
input={"query": "top finance news today", "domain": "finance"},
output={"url": "[news.com/1](https://news.com/1)", "title": "Market Hits New High", "snippet": "..."},
)
],
output={"summary": "The market reached a new high today, driven by tech stocks."},
)
# User: Search for today's top finance news.
# Tool (web_search): User asked for current news.
# Input:
# {
# "domain": "finance",
# "query": "top finance news today"
# }
# Output:
# {
# "snippet": "...",
# "title": "Market Hits New High",
# "url": "[news.com/1](https://news.com/1)"
# }
# Output:
# {
# "summary": "The market reached a new high today, driven by tech stocks."
# }
“Kitchen sink” (title, context, multiple tools, thought, structured output):
pp.shot(
title="Example: Research and Summarize Financial Report",
user="Find the latest quarterly earnings for 'TechCorp' and summarize key takeaways.",
context={"username": "analyst_jane", "expertise_level": "expert"},
tools=[
ToolStep(
name="web_search",
rationale="Find the official press release.",
input={"query": "TechCorp latest quarterly earnings report"},
output={"url": "[investors.techcorp.com/q3-2025-earnings](https://investors.techcorp.com/q3-2025-earnings)", "title": "TechCorp Reports Q3 2025", "snippet": "..."},
thought="First result is the official source."
),
ToolStep(
name="summarize_document",
rationale="Summarize key points.",
input={"url": "[investors.techcorp.com/q3-2025-earnings](https://investors.techcorp.com/q3-2025-earnings)"},
output={"summary": "Strong earnings from cloud.", "takeaways": ["Revenue hit $50B."]},
),
],
thought="Tool chain complete; formatting final answer.",
output={
"summary": "TechCorp reported record revenue of $50B in Q3 2025, driven by cloud.",
"key_takeaways": ["Record revenue", "Cloud +25% YoY"],
"confidence_score": 0.95,
},
)
# Example: Research and Summarize Financial Report
# ------------------------------------------------
# User: Find the latest quarterly earnings for 'TechCorp' and summarize key takeaways.
# Context:
# {
# "expertise_level": "expert",
# "username": "analyst_jane"
# }
# Tool (web_search): Find the official press release.
# Thought: First result is the official source.
# Input:
# {
# "query": "TechCorp latest quarterly earnings report"
# }
# Output:
# {
# "snippet": "...",
# "title": "TechCorp Reports Q3 2025",
# "url": "[investors.techcorp.com/q3-2025-earnings](https://investors.techcorp.com/q3-2025-earnings)"
# }
# Tool (summarize_document): Summarize key points.
# Input:
# {
# "url": "[investors.techcorp.com/q3-2025-earnings](https://investors.techcorp.com/q3-2025-earnings)"
# }
# Output:
# {
# "summary": "Strong earnings from cloud.",
# "takeaways": [
# "Revenue hit $50B."
# ]
# }
# Thought: Tool chain complete; formatting final answer.
# Output:
# {
# "confidence_score": 0.95,
# "key_takeaways": [
# "Record revenue",
# "Cloud +25% YoY"
# ],
# "summary": "TechCorp reported record revenue of $50B in Q3 2025, driven by cloud."
# }
Returns A single string with all sections rendered in a stable, LLM-friendly layout.
table(...) — render small tables from dicts/lists/CSV/DataFrames
Overloads
pp.table(
rows: RowsLike,
headers: Sequence[str] | None = None,
fmt: TableFormat = "csv",
) -> str
What it does Turns your data into a compact table. Accepts:
- Rows of mappings (
Sequence[Mapping[str, Any]]) - Rows of sequences (
Sequence[Sequence[Any]]) - Pandas-like DataFrame (
.columnsand.itertuples(index=False, name=None)) - CSV (pass a string of CSV text, or a
Path/PathLiketo read a file)
Headers & normalization
-
If
headers=None:- Mappings: union of keys in iteration order (first row’s order, then unseen keys).
- Sequences: headers auto-generated:
col1,col2, … - DataFrame: uses
df.columns. - CSV text/path: uses first row as headers.
-
If
headersprovided, rows are padded/truncated to that width.
Formats
fmt="pretty"→ uses prettytable (must be installed).fmt="csv"→ emits CSV text (viacsv.writer).- Raises
RuntimeErroriffmt="pretty"andprettytableis missing;ValueErroriffmtis not"pretty"or"csv".
Examples
Mappings (pretty):
rows = [{'name': 'Ada', 'role': 'admin'}, {'name': 'Bob'}]
pp.table(rows)
# name,role
# Ada,admin
# Bob,
Sequences with explicit headers (csv):
rows = [
['Ada', 'admin', 5],
['Bob', 'viewer', 2],
]
pp.table(rows, headers=['name', 'role'], fmt='csv')
# name,role
# Ada,admin
# Bob,viewer
CSV text in, pretty out:
csv_text = 'name,score\nAda,10\nBob,8\n'
pp.table(csv_text, fmt='pretty')
# +------+-------+
# | name | score |
# +------+-------+
# | Ada | 10 |
# | Bob | 8 |
# +------+-------+
DataFrame-like:
class MiniDf:
columns = ['name', 'v']
def itertuples(self, *, index: bool, name: None):
yield ('Ada', 1)
yield ('Bob', 2)
pp.table(MiniDf(), fmt='csv')
# name,v
# Ada,1
# Bob,2
Returns A single string containing the rendered table.
warning(...) — tagged warning blocks (3 severity levels)
Overloads
pp.warning(body: str, *, level: Literal[1,2,3] = 1, title: str | None = None) -> str
pp.warning(body: LlmResponseLike, *, level: Literal[1,2,3] = 1, title: str | None = None) -> str
What it does
Emits a warning using XML-style tags, with escalating emphasis by level.
Levels
level=1→<NOTICE>…</NOTICE>level=2→<WARNING>…</WARNING>level=3→<CONSTRAINT>…</CONSTRAINT>
Args
body: string or anyLlmResponseLike(normalized viato_display_block).title: optional text prepended to the body as"Title: ..."inside the tag.- Raises
ValueErroriflevelnot in{1,2,3}.
Examples
Basic:
pp.warning('Be careful with rate limits.')
# <NOTICE>
# Be careful with rate limits.
# </NOTICE>
With a title and level 2:
pp.warning('Requests must be idempotent.', level=2, title='API')
# <WARNING>
# API: Requests must be idempotent.
# </WARNING>
Critical (level 3) with structured body:
pp.warning({'do': 'return JSON only', 'dont': 'write prose'}, level=3)
# <CONSTRAINT>
# {
# "do": "return JSON only",
# "dont": "write prose"
# }
# </CONSTRAINT>
Returns A single string with the formatted warning block.
Notes & Limitations
- ⚠️ Compare enums with
==/!=instead ofis/is not. PragmaPrompt recompiles prompt files in a fresh module during each render, so enum objects do not preserve identity across runs.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pragma_prompt-0.0.2.tar.gz.
File metadata
- Download URL: pragma_prompt-0.0.2.tar.gz
- Upload date:
- Size: 34.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
03d06f0d958aa046aee540238a8fe82b744953bfdd1e7e76577915665a2806bb
|
|
| MD5 |
e4dcaadfac4afe134e500a96aed7af92
|
|
| BLAKE2b-256 |
643e12e025f91b75f820ba5b66da73b5ab9e8cc963c90caece8ff26db591724d
|
Provenance
The following attestation bundles were made for pragma_prompt-0.0.2.tar.gz:
Publisher:
release.yml on DavidTokar12/PragmaPrompt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pragma_prompt-0.0.2.tar.gz -
Subject digest:
03d06f0d958aa046aee540238a8fe82b744953bfdd1e7e76577915665a2806bb - Sigstore transparency entry: 630716584
- Sigstore integration time:
-
Permalink:
DavidTokar12/PragmaPrompt@8e128cce831f9d273c7c392298d5badfd47e5746 -
Branch / Tag:
refs/tags/v0.0.2 - Owner: https://github.com/DavidTokar12
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8e128cce831f9d273c7c392298d5badfd47e5746 -
Trigger Event:
release
-
Statement type:
File details
Details for the file pragma_prompt-0.0.2-py3-none-any.whl.
File metadata
- Download URL: pragma_prompt-0.0.2-py3-none-any.whl
- Upload date:
- Size: 33.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a5211ecc4821c3a3204be8e430083753ac4ced4f974b7cf0a0ecc2a0ccb6e5c2
|
|
| MD5 |
a29a598d6db09d9773c15cd1c26c3c05
|
|
| BLAKE2b-256 |
6e57e21bd18cec90a62126fa1710c1df702e7cadf1b2c42779170b879637cc84
|
Provenance
The following attestation bundles were made for pragma_prompt-0.0.2-py3-none-any.whl:
Publisher:
release.yml on DavidTokar12/PragmaPrompt
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pragma_prompt-0.0.2-py3-none-any.whl -
Subject digest:
a5211ecc4821c3a3204be8e430083753ac4ced4f974b7cf0a0ecc2a0ccb6e5c2 - Sigstore transparency entry: 630716611
- Sigstore integration time:
-
Permalink:
DavidTokar12/PragmaPrompt@8e128cce831f9d273c7c392298d5badfd47e5746 -
Branch / Tag:
refs/tags/v0.0.2 - Owner: https://github.com/DavidTokar12
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8e128cce831f9d273c7c392298d5badfd47e5746 -
Trigger Event:
release
-
Statement type: