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.4.0.tar.gz (230.7 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.4.0-py3-none-any.whl (88.1 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: llm_saia-0.4.0.tar.gz
  • Upload date:
  • Size: 230.7 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.4.0.tar.gz
Algorithm Hash digest
SHA256 4001511017e09deff4ddd4b88146ccd41047ce4ed2b775ca6d5242abfa17cc47
MD5 cb6770389fb677c4f172009e9021473b
BLAKE2b-256 3db2b27f3474bf79a9c530c312bf153e6b3c149b26f0662dbf1a338b4fd1f93d

See more details on using hashes here.

Provenance

The following attestation bundles were made for llm_saia-0.4.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.4.0-py3-none-any.whl.

File metadata

  • Download URL: llm_saia-0.4.0-py3-none-any.whl
  • Upload date:
  • Size: 88.1 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.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 f12848f799259a9f69659e16265bbf44936338924b2cb1a34775a7f5ae9ade85
MD5 171c026c3ac8b3dff6bf4a56dffc29c0
BLAKE2b-256 9565c21d49640105134b72c8676a15b40ff5463e1d3928bcc60e34a4b395454f

See more details on using hashes here.

Provenance

The following attestation bundles were made for llm_saia-0.4.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