Simulation and trace-based evaluation for agentic systems
Project description
understudy: Scenario Testing for AI Agents
Understudy is a scenario-driven testing framework for AI agents that simulates realistic multi-turn users, runs those scenes against an agent through a simple app adapter, records a structured execution trace of messages, tool calls, and handoffs, and then evaluates behavior with deterministic checks, optional LLM judges, and run reports.
How It Works
Testing with understudy is 4 steps:
- Wrap your agent — Adapt your agent (ADK, LangGraph, HTTP) to understudy's interface
- Mock your tools — Register handlers that return test data instead of calling real services
- Write scenes — YAML files defining what the simulated user wants and what you expect
- Run and assert — Execute simulations, check traces, generate reports
The key insight: assert against the trace, not the prose. Don't check what the agent said—check what it did (tool calls).
See real examples:
- Example scene — YAML defining a test scenario
- ADK test file — pytest assertions against traces
- LangGraph test file — same tests, different framework
Installation
pip install understudy[all]
Quick Start
1. Wrap your agent
from understudy.adk import ADKApp
from my_agent import agent
app = ADKApp(agent=agent)
2. Mock your tools
Your agent has tools that call external services. Mock them for testing:
from understudy.mocks import MockToolkit
mocks = MockToolkit()
@mocks.handle("lookup_order")
def lookup_order(order_id: str) -> dict:
return {"order_id": order_id, "items": [...], "status": "delivered"}
@mocks.handle("create_return")
def create_return(order_id: str, item_sku: str, reason: str) -> dict:
return {"return_id": "RET-001", "status": "created"}
3. Write a scene
Create scenes/return_backpack.yaml:
id: return_eligible_backpack
description: Customer wants to return a backpack
starting_prompt: "I'd like to return an item please."
conversation_plan: |
Goal: Return the hiking backpack from order ORD-10031.
- Provide order ID when asked
- Return reason: too small
persona: cooperative
max_turns: 15
expectations:
required_tools:
- lookup_order
- create_return
forbidden_tools:
- issue_refund
4. Run simulation
from understudy import Scene, run
scene = Scene.from_file("scenes/return_backpack.yaml")
trace = run(app, scene, mocks=mocks)
assert trace.called("lookup_order")
assert trace.called("create_return")
assert not trace.called("issue_refund")
Or with pytest (define app and mocks fixtures in conftest.py):
pytest test_returns.py -v
Suites and Batch Runs
Run multiple scenes with multiple simulations per scene:
from understudy import Suite, RunStorage
suite = Suite.from_directory("scenes/")
storage = RunStorage()
# Run each scene 3 times and tag for comparison
results = suite.run(
app,
mocks=mocks,
storage=storage,
n_sims=3,
tags={"version": "v1"},
)
print(f"{results.pass_count}/{len(results.results)} passed")
Simulation and Evaluation
Understudy separates simulation (generating traces) from evaluation (checking traces). Use together or separately:
Combined (most common)
understudy run \
--app mymodule:agent_app \
--scene ./scenes/ \
--n-sims 3 \
--junit results.xml
Separate workflows
Generate traces only:
understudy simulate \
--app mymodule:agent_app \
--scenes ./scenes/ \
--output ./traces/ \
--n-sims 3
Evaluate existing traces:
understudy evaluate \
--traces ./traces/ \
--output ./results/ \
--junit results.xml
Python API:
from understudy import simulate_batch, evaluate_batch
# Generate traces
traces = simulate_batch(
app=agent_app,
scenes="./scenes/",
n_sims=3,
output="./traces/",
)
# Evaluate later
results = evaluate_batch(
traces="./traces/",
output="./results/",
)
CLI Commands
# Run simulations
understudy run --app mymodule:app --scene ./scenes/
understudy simulate --app mymodule:app --scenes ./scenes/
understudy evaluate --traces ./traces/
# View results
understudy list
understudy show <run_id>
understudy summary
# Compare runs by tag
understudy compare --tag version --before v1 --after v2
# Generate reports
understudy report -o report.html
understudy compare --tag version --before v1 --after v2 --html comparison.html
# Interactive browser
understudy serve --port 8080
# HTTP simulator server (for browser/UI testing)
understudy serve-api --port 8000
# Cleanup
understudy delete <run_id>
understudy clear
LLM Judges
For qualities that can't be checked deterministically:
from understudy.judges import Judge
empathy_judge = Judge(
rubric="The agent acknowledged frustration and was empathetic while enforcing policy.",
samples=5,
)
result = empathy_judge.evaluate(trace)
assert result.score == 1
Built-in rubrics:
from understudy.judges import (
TOOL_USAGE_CORRECTNESS,
POLICY_COMPLIANCE,
TONE_EMPATHY,
ADVERSARIAL_ROBUSTNESS,
TASK_COMPLETION,
)
Report Contents
The understudy summary command shows:
- Pass rate — percentage of scenes that passed all expectations
- Avg turns — average conversation length
- Tool usage — distribution of tool calls across runs
- Agents — which agents were invoked
The HTML report (understudy report) includes:
- All metrics above
- Full conversation transcripts
- Tool call details with arguments
- Expectation check results
- Judge evaluation results (when used)
Documentation
See the full documentation for:
- Installation guide
- Writing scenes
- ADK integration
- LangGraph integration
- HTTP client for deployed agents
- API reference
License
MIT
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
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 understudy-0.4.0.tar.gz.
File metadata
- Download URL: understudy-0.4.0.tar.gz
- Upload date:
- Size: 49.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
59e561e16ca0ca4d0384f1efbf6202bb278fc4754329d1494c646943457f7d68
|
|
| MD5 |
375bee673177ed5cb7b8a2c37a97e487
|
|
| BLAKE2b-256 |
64f7df4f44987f561b2aae91a2a3fe20a1ecf18d62ee3f0fc1e9a9639b16ebc7
|
Provenance
The following attestation bundles were made for understudy-0.4.0.tar.gz:
Publisher:
python-publish.yml on gojiplus/understudy
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
understudy-0.4.0.tar.gz -
Subject digest:
59e561e16ca0ca4d0384f1efbf6202bb278fc4754329d1494c646943457f7d68 - Sigstore transparency entry: 1109084419
- Sigstore integration time:
-
Permalink:
gojiplus/understudy@6d475d51ca6c1c1e6cae23c9f5114c9deef9f67f -
Branch / Tag:
refs/heads/main - Owner: https://github.com/gojiplus
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@6d475d51ca6c1c1e6cae23c9f5114c9deef9f67f -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file understudy-0.4.0-py3-none-any.whl.
File metadata
- Download URL: understudy-0.4.0-py3-none-any.whl
- Upload date:
- Size: 64.2 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 |
23aa02b89e98020edf14c54dbff92ebcc28c0f28385a672b363410636ccca563
|
|
| MD5 |
17e52a51423415aa50d8dd80f7886a49
|
|
| BLAKE2b-256 |
cd16cd44df05b4fbc8f1f256b0a0f4a47cf90f58a34614f9c717af3f74dab9e7
|
Provenance
The following attestation bundles were made for understudy-0.4.0-py3-none-any.whl:
Publisher:
python-publish.yml on gojiplus/understudy
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
understudy-0.4.0-py3-none-any.whl -
Subject digest:
23aa02b89e98020edf14c54dbff92ebcc28c0f28385a672b363410636ccca563 - Sigstore transparency entry: 1109084435
- Sigstore integration time:
-
Permalink:
gojiplus/understudy@6d475d51ca6c1c1e6cae23c9f5114c9deef9f67f -
Branch / Tag:
refs/heads/main - Owner: https://github.com/gojiplus
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@6d475d51ca6c1c1e6cae23c9f5114c9deef9f67f -
Trigger Event:
workflow_dispatch
-
Statement type: