Skip to main content

Framework-agnostic verb vocabulary for LLM agents

Project description

SAIA

Framework-agnostic verb vocabulary for LLM agents

Python Dependencies Coverage Typed Linting: Ruff CI License

SAIA provides a fixed vocabulary of semantic verbs for LLM interactions. Instead of writing raw prompts, you express intent through verbs like ask, verify, critique, and refine.

Why a fixed vocabulary? The same insight that made SCUMM work for adventure games in 1987: constraints enable tooling. A finite set of operations means every interaction is debuggable, testable, and composable.

Why SAIA?

"Can't I just type this into Claude?" For a one-off question, yes. SAIA is for when you're building software, not chatting:

  • Structured outputs - verify() returns VerifyResult(passed: bool, reason: str), not text you parse
  • Composable - chain verify → critique → refine in 3 lines of code
  • Testable - mock the backend, unit test your verbs
  • Traceable - every verb call logged with inputs/outputs for debugging production issues
  • Backend-agnostic - same code works with Anthropic, OpenAI, or local models
  • Zero dependencies - pure Python core; bring your own LLM client

"Why not raw tool calling?" You could write the iteration loop yourself (~50-100 lines). SAIA gives you complete() with terminal detection, tracing, timeouts, and max iterations built in. It's requests vs urllib - both work, one is cleaner.

SAIA's value compounds when combining verbs, switching backends, or building a team around consistent patterns.

"Is this novel?" Perhaps not. These are the patterns that emerge when you build LLM agents that need to actually work. SAIA extracts them into ~2500 lines anyone can use, inspect, and build on.

Installation

pip install llm-saia

Quick Start

from llm_saia import SAIA

async def main():
    saia = (
        SAIA.builder()
        .backend(your_backend)
        .build()
    )

    # Verify a claim
    result = await saia.verify(
        "This code handles null input safely",
        "no null pointer exceptions possible"
    )
    print(f"Passed: {result.passed}, Reason: {result.reason}")

    # Generate counter-arguments
    critique = await saia.critique("Python is slow for all tasks")
    print(f"Counter: {critique.counter_argument}")

    # Break down a complex task
    subtasks = await saia.decompose("Build a REST API with authentication")
    for task in subtasks:
        print(f"- {task}")

Verb Reference

Verb Purpose Returns
ask Query an artifact with a question str
verify Check if artifact satisfies predicate VerifyResult(passed, reason)
critique Generate strongest counter-argument Critique(counter_argument, weaknesses, strength)
refine Improve artifact based on feedback str
synthesize Combine multiple artifacts into one T (structured)
decompose Break complex task into subtasks list[str]
extract Pull structured data from text T (structured)
classify Categorize into predefined classes ClassifyResult(label, confidence)
choose Select best option from choices ChooseResult(choice, reasoning)
find Filter items matching criteria FindResult(indices, reason) (0-indexed)
constrain Rewrite text to comply with rules str
ground Anchor claims to source evidence list[Evidence]
instruct Execute open-ended instructions str

Memory Verbs

Verb Purpose
store Save value to memory
recall Retrieve from memory by query

Configuration

Builder Pattern

saia = (
    SAIA.builder()
    .backend(backend)                    # Required: LLM backend
    .tools(tool_defs, executor)          # Optional: tool calling
    .logger(logger)                      # Optional: logging
    .system("You are a helpful assistant")  # Optional: system prompt
    .max_iterations(10)                  # Optional: tool loop limit
    .max_call_tokens(4096)               # Optional: per-call token limit
    .build()
)

Runtime Modifiers

# Single LLM call (no tool loop)
result = await saia.with_single_call().verify(code, "compiles")

# Custom iteration limit
result = await saia.with_max_iterations(5).instruct(task)

# Timeout
result = await saia.with_timeout(30).decompose(problem)

# Correlation ID for tracing
result = await saia.with_request_id("req-123").ask(doc, question)

Output Guards

Guards validate LLM output and automatically retry with feedback if validation fails.

from llm_saia.guards import english_only, max_length, no_preamble

# Apply guards to any verb
result = await (
    saia
    .with_guards(english_only(), no_preamble(), max_length(500))
    .ask(article, "summarize this")
)

Field-level guards for structured output:

from typing import Annotated
from llm_saia import Guarded
from llm_saia.guards import english_only, max_length

@dataclass
class Summary:
    title: Annotated[str, Guarded(english_only(), max_length(100))]
    body: Annotated[str, Guarded(english_only())]

result = await saia.extract(article, Summary)

Pre-built guards: english_only(), ascii_only(), max_length(n), no_emoji(), no_markdown(), no_preamble().

Iteration Guards

Iteration guards enforce behavioral constraints during a tool-calling loop, rather than validating the final result. When a guard fires, its feedback is injected into the conversation and the loop continues.

from llm_saia import IterationGuard

# Require the LLM to explain what it's doing before calling tools
def require_narrative(response):
    if response.tool_calls and not (response.content or "").strip():
        return "Explain what you're doing and why before calling tools."
    return None

result = await (
    saia
    .with_guard(IterationGuard(require_narrative, name="narrative"))
    .complete(task)
)

with_guard() and with_guards() accept both OutputGuard and IterationGuard — each is routed to the correct bucket automatically.

Examples

See the examples/ directory:

  • investigate.py - Investigate a claim (verify → critique → refine)
  • build.py - Build an app (decompose → instruct → synthesize)
  • build_multi.py - Two LLMs collaborate (local generates, smart verifies)

Design Philosophy

  1. Verbs express intent - not implementation details
  2. Structured over strings - type-safe dataclass responses
  3. Composable primitives - build complex flows from simple verbs
  4. Backend-agnostic - same code works with any LLM
  5. Zero dependencies - pure Python core, you control your LLM client
  6. Debuggable - every operation is traceable

Research Directions

SAIA's structured verb outputs create opportunities beyond inference:

  • Consistency tuning - traces capture (prompt, decision) pairs that can fine-tune models for stable verb behavior. Same semantic question → same semantic answer.
  • Structured generation - backends can use grammar-constrained decoding (xgrammar, outlines) to guarantee valid outputs from verbs like extract, verify, and classify without retry loops.

License

Apache 2.0 - see LICENSE

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

llm_saia-0.3.0.tar.gz (201.5 kB view details)

Uploaded Source

Built Distribution

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

llm_saia-0.3.0-py3-none-any.whl (70.6 kB view details)

Uploaded Python 3

File details

Details for the file llm_saia-0.3.0.tar.gz.

File metadata

  • Download URL: llm_saia-0.3.0.tar.gz
  • Upload date:
  • Size: 201.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for llm_saia-0.3.0.tar.gz
Algorithm Hash digest
SHA256 b3d88e164b5d273e71142fae4640fdc79dd0a3dff6d3eefbe2c637df2de2c00e
MD5 e8ec3302abe1a51d94170e4401eb2188
BLAKE2b-256 6390c4d6b83f7fc8ae606fceb70aabbaace01507e381fe19f2a186f3ff70e7d3

See more details on using hashes here.

Provenance

The following attestation bundles were made for llm_saia-0.3.0.tar.gz:

Publisher: release.yml on llm-works/llm-saia

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

File details

Details for the file llm_saia-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: llm_saia-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 70.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for llm_saia-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 5e22d9efc9e74019dca02c4eb6a96f6723ab73e0f972278f2726d2e11e439d2b
MD5 550f8e001bbb49b16577ecc8e5ddcc0d
BLAKE2b-256 49c5f58fbf58d2657e93c45cfe90312977577aad5ce337c9e2d49845ecf97c3a

See more details on using hashes here.

Provenance

The following attestation bundles were made for llm_saia-0.3.0-py3-none-any.whl:

Publisher: release.yml on llm-works/llm-saia

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