Structured output for any LLM call. Validate-and-retry loop for JSON responses; BYO LLM and validator. Python port of @mukundakatta/agentcast.
Project description
agentcast-py
Structured output for any LLM call. Validate the response, retry with the validation error as feedback, raise CastError if all retries fail. BYO LLM, BYO validator. Zero runtime dependencies.
Python port of @mukundakatta/agentcast.
Install
pip install agentcast-py
# pydantic adapter is optional:
pip install "agentcast-py[pydantic]"
Usage
import asyncio
from anthropic import AsyncAnthropic
from pydantic import BaseModel
from agentcast import cast, adapters, CastError
client = AsyncAnthropic()
async def llm(messages):
resp = await client.messages.create(
model="claude-sonnet-4-6",
max_tokens=512,
messages=messages,
)
return resp.content[0].text
class Diagnosis(BaseModel):
code: str
severity: str
notes: str
async def main():
out = await cast(
llm=llm,
validate=adapters.pydantic(Diagnosis),
prompt="Triage: server is on fire. Output a Diagnosis JSON object.",
max_retries=3,
)
print(out) # validated dict
asyncio.run(main())
If the LLM returns prose, fenced JSON, or near-misses, cast() extracts the JSON, validates it, and retries with the validation error in the chat history -- so the next attempt has the spec. After max_retries failures, raises CastError carrying every attempt for postmortem.
Sync LLMs work too
def sync_llm(messages):
return openai_client.chat.completions.create(...).choices[0].message.content
out = await cast(llm=sync_llm, validate=schema, prompt="...", max_retries=2)
cast() is async (so you can await your async LLM), but the llm callable can be sync or async.
API
cast(*, llm, validate, prompt, system=None, max_retries=2, on_attempt=None) -> Any
Async. Returns the validated value (whatever the validator's value field is). Raises CastError on exhausted retries.
extract_json(text) -> Any | None
Sync. Pulls the largest valid JSON value out of arbitrary text -- whole text, fenced block, or balanced substring. Returns None if nothing parseable.
adapters.shape(spec)
Tiny built-in shape checker. Same as the sibling agentvet. {"field": "str|int|...|str?"}.
adapters.fn(predicate, error_builder?)
Predicate adapter.
adapters.pydantic(model_cls)
Wraps a pydantic v2 BaseModel. Requires pip install agentcast-py[pydantic].
CastError
Carries attempts: list[dict], last_error, last_text, last_parsed for postmortem.
API differences from the JS sibling
cast()isasync def(Python's natural pattern for LLM calls).llmmay be sync or async; both are auto-detected.adapters.pydanticreplacesadapters.zodas the natural Python adapter.- Keyword args throughout (
max_retries=,on_attempt=). - Validators return Python dicts:
{"valid": bool, "value": ... | "error": str}.
See the JS sibling's README for the full design notes.
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 agentcast_py-0.1.0.tar.gz.
File metadata
- Download URL: agentcast_py-0.1.0.tar.gz
- Upload date:
- Size: 9.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b4512249e1216969a2b13708bd2031c6eaf713a2a8fc8b8ad4594f73b0a2482d
|
|
| MD5 |
c3d290aa77f23bda894df97a1ed001ae
|
|
| BLAKE2b-256 |
ebe6b21cab337cb03bda3bf001801ac3b208b962113a69b507a525aec9afc827
|
File details
Details for the file agentcast_py-0.1.0-py3-none-any.whl.
File metadata
- Download URL: agentcast_py-0.1.0-py3-none-any.whl
- Upload date:
- Size: 9.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3e69af18b97119a984bbad565fadf98110adb184c1f165b6978a8bb2a71f5d65
|
|
| MD5 |
c4585433b7bef78c1876b9815ec3ecbe
|
|
| BLAKE2b-256 |
7ae901731939565c83819ff2a27c74863a493fa722a3ad1a45937916e9e1f1c3
|