Skip to main content

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.

License Code style: black Type hints codecov


  • 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_format, 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.

import pragma_prompt as pp
from my_module import MyModule

ctx = MyModule.my_prompt.context
rm = MyModule.my_prompt.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 though love, he can handle it."

with pp.section("user_data"):
    f"The users name is: {rm.user_name}"

Result:

# Your job is to motivate the user.
Go easy on the user, he is sad.
<user_data>
The users name is: John
</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.

# in my_prompts/daily_motivation.py

# Use a loop to repeat instructions
for i in range(3):
    f"Repetition {i+1}: Be concise and clear."

# Use a conditional to change the prompt based on context
if ctx.user_is_new:
    "Explain the concepts simply."
else:
    "Assume the user is an expert."

This will render:

Repetition 1: Be concise and clear.
Repetition 2: Be concise and clear.
Repetition 3: Be concise and clear.
Assume the user is an expert.

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 my_app/prompts.py
from pathlib import Path
import pragma_prompt as pp

# (Optional) Define a class for shared constants
class MyConstants:
    default_tone = "helpful"
    max_length = 1000

# Create a module class
class MyPrompts(pp.PromptModule):
    # 1. Set the directory where your prompt files are located
    module_dir = Path(__file__).parent / "prompt_files"
    
    # 2. (Optional) Attach your constants
    constants = MyConstants()

    # 3. Define your prompts
    # PragmaPrompt will look for 'summarize.py' in your module_dir
    summarize = pp.Prompt() 
    
    # You can also specify a different filename
    translate = pp.Prompt("translate_text.py")

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 pp.constants().

# in prompt_files/summarize.py
import pragma_prompt as pp

# Get handles to context and render_model inside an active session
ctx = MyPrompts.summarize.context
rm = MyPrompts.summarize.render_model

# Access constants from the module
f"Your tone should be {pp.constants().default_tone}."

# Access data from the context and render_model
f"Summarize the following text for user {rm.user_id}:"
f"{ctx.text_to_summarize}"

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.

# in your application code
from my_app.prompts import MyPrompts

# The data you want to inject into the prompt
class RenderData:
    user_id = "user123"

class ContextData:
    text_to_summarize = "This is a long article..."

prompt_text = MyPrompts.summarize.render(
    context=ContextData(),
    render_model=RenderData()
)

print(prompt_text)

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()

2. Write Your Component File

Component files are simpler than prompt files. They can access shared constants but do not have context or render_model.

# in component_files/output_format.py
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.

# in prompt_files/summarize.py
import pragma_prompt as pp
from my_app.components import MyComponents

ctx, rm = pp.this_prompt()

f"Summarize this text: {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: JsonObj | BaseModel | DataclassInstance | SupportsModelDump | None,
    *,
    comments: str | Mapping[str, str | Mapping] | 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)

  • comments may 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", …).
  • All comment keys/indices must exist in data; unknown keys raise ValueError.

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

  • width defaults to 80.
  • char must be a single visible character; defaults to "-".
  • When title is provided, emits a single centered line.

Examples

pp.separator()
# "--------------------------------------------------------------------------------"  # 80 x "-"

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(
    *,
    user: str,
    output: Any,
    title: str | None = None,
    context: Any | None = None,
    input: Any | None = None,
    tools: list[ToolStep] | None = None,
    thought: str | None = None,
) -> 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)

  1. title (as a one-line heading)
  2. User
  3. Context (pretty-printed, deterministic JSON from dict/dataclass/Pydantic)
  4. Input (pretty JSON)
  5. Tool call chain (each step: name, rationale, input, output, optional thought)
  6. Thought
  7. Output (pretty JSON or string)

Tool steps

ToolStep(
    name: str,
    rationale: str,
    input: Any,          # dict/dataclass/Pydantic also supported
    output: Any,         # dict/dataclass/Pydantic also supported
    thought: str | None = None,
)

Examples

Minimal:

pp.shot(
    user="What is the capital of Switzerland?",
    output="Bern",
)

Context + input:

from pydantic import BaseModel, Field

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.",
)

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...",
)

With a single tool call:

from pragma_prompt import ToolStep

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", "title": "Market Hits New High", "snippet": "..."},
        )
    ],
    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", "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"},
            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,
    },
)

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: Sequence[Mapping[str, Any]], *, headers: Sequence[str] | None = None, fmt: Literal["pretty","csv"] = "pretty") -> str
pp.table(rows: Sequence[Sequence[Any]],     *, headers: Sequence[str] | None = None, fmt: Literal["pretty","csv"] = "pretty") -> str
pp.table(rows: PandasLikeDataFrame,         *, headers: Sequence[str] | None = None, fmt: Literal["pretty","csv"] = "pretty") -> str
pp.table(rows: str | Path | PathLike[str],  *, headers: Sequence[str] | None = None, fmt: Literal["pretty","csv"] = "pretty") -> 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 (.columns and .itertuples(index=False, name=None))
  • CSV (pass a string of CSV text, or a Path/PathLike to 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 headers provided, rows are padded/truncated to that width.

Formats

  • fmt="pretty" → uses prettytable (must be installed).
  • fmt="csv" → emits CSV text (via csv.writer).
  • Raises RuntimeError if fmt="pretty" and prettytable is missing; ValueError if fmt is not "pretty" or "csv".

Examples

Mappings (pretty):

rows = [{"name": "Ada", "role": "admin"}, {"name": "Bob"}]
print(pp.table(rows))
# +------+-------+
# | name | role  |
# +------+-------+
# | Ada  | admin |
# | Bob  |       |
# +------+-------+

Sequences with explicit headers (csv):

rows = [
    ["Ada", "admin", 5],
    ["Bob", "viewer", 2],
]
print(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"
print(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)

print(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<WARNING>…</WARNING>
  • level=2<IMPORTANT-WARNING>…</IMPORTANT-WARNING>
  • level=3<CRITICAL-WARNING>…</CRITICAL-WARNING> and prepends: HARD REQUIREMENT: You must follow the instruction below exactly.

Args

  • body: string or any LlmResponseLike (normalized via to_display_block).
  • title: optional text prepended to the body as "Title: ..." inside the tag.
  • Raises ValueError if level not in {1,2,3}.

Examples

Basic:

pp.warning("Be careful with rate limits.")
# <WARNING>
# Be careful with rate limits.
# </WARNING>

With a title and level 2:

pp.warning("Requests must be idempotent.", level=2, title="API")
# <IMPORTANT-WARNING>
# API: Requests must be idempotent.
# </IMPORTANT-WARNING>

Critical (level 3) with structured body:

pp.warning({"do": "return JSON only", "dont": "write prose"}, level=3)
# <CRITICAL-WARNING>
# HARD REQUIREMENT: You must follow the instruction below exactly.
# {
#   "do": "return JSON only",
#   "dont": "write prose"
# }
# </CRITICAL-WARNING>

Returns A single string with the formatted warning block.

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

pragma_prompt-0.0.1.tar.gz (33.3 kB view details)

Uploaded Source

Built Distribution

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

pragma_prompt-0.0.1-py3-none-any.whl (32.6 kB view details)

Uploaded Python 3

File details

Details for the file pragma_prompt-0.0.1.tar.gz.

File metadata

  • Download URL: pragma_prompt-0.0.1.tar.gz
  • Upload date:
  • Size: 33.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pragma_prompt-0.0.1.tar.gz
Algorithm Hash digest
SHA256 df3e61048c7554ca653fc11151c7e4910ecddde859aa54264e059a44937f753b
MD5 e0254f1f97bd1bb54fc0d44efbb0e479
BLAKE2b-256 4528c53195ed7ac57186a2d79195a8a81bdd878fb0dfcf09dbdb86ac5cc73e5e

See more details on using hashes here.

Provenance

The following attestation bundles were made for pragma_prompt-0.0.1.tar.gz:

Publisher: release.yml on DavidTokar12/PragmaPrompt

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pragma_prompt-0.0.1-py3-none-any.whl.

File metadata

  • Download URL: pragma_prompt-0.0.1-py3-none-any.whl
  • Upload date:
  • Size: 32.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pragma_prompt-0.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 d7f546df236ae587847055ed70f774fd58ef38105b37a8d41de946db21786c6b
MD5 0c52d8510efa7f5390d9d6d1bf7eb6bf
BLAKE2b-256 597306b889618a7e941ea65c1cec039c0358646a05cd977e4878d27ef5763e1f

See more details on using hashes here.

Provenance

The following attestation bundles were made for pragma_prompt-0.0.1-py3-none-any.whl:

Publisher: release.yml on DavidTokar12/PragmaPrompt

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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