cd /news/artificial-intelligence/nobody-is-measuring-what-your-ai-age… Β· home β€Ί topics β€Ί artificial-intelligence β€Ί article
[ARTICLE Β· art-27182] src=github.com β†— pub= topic=artificial-intelligence verified=true sentiment=↑ positive

Nobody Is Measuring What Your AI Agents Are Worth

A new open-source tool called agent-panorama converts raw LLM agent traces into plain-English reports for managers, answering whether agents are worth their cost. It works with LangChain and LangGraph apps, requires only a single callback to install, and outputs Markdown, HTML, or JSON reports with value scoring and cost-per-valuable-conversation metrics.

read18 min publishedJun 14, 2026

Turn raw LLM agent traces into a report a manager can actually read - what your agents did, whether it was worth it, and what it cost. Point it at a Langfuse (or LangSmith) export (or add a one-line live callback) and get clean Markdown + a self-contained HTML report - and a local dashboard - in plain business language.

It's one line to switch on. An engineer drops a single callback into your existing agents - no rebuild, no new infrastructure, no traces to wire up - and the live dashboard starts filling in. Works in any LangChain or LangGraph app today (more frameworks on the roadmap).

Three questions about any agent in production. Your existing tools answer the first two:

Question Answered by
Does it run? observability - traces, tokens, latency
Is it correct? evals - scores on a test set
β†’
Is it worth it?
agent-panorama

It answers the third - the one your CEO, client, or PM actually asks - across three rungs over the same conversations:

Clarity - "what did they do?" One plain-English feed across the fleet (or a single agent): asked X β†’ did Y β†’ outcome. A 30-message chat becomes one line. No spans, no JSON.Value - "was it worth it?" An LLM judge scores each conversation againstyourdefinition of value (your domain, your user goal, your success criteria) and reports the value delivered, the value lost, and what to fix.Cost - "what did it cost?" Tokens β†’ dollars β†’cost per valuable conversation, the ROI number nobody else gives you.

The fleet view - one plain-English activity feed across every agent, with per-run details, outcomes, and cost.

Define value without YAML - a guided wizard fills in each agent's value ontology as a live constellation map, then a Value Blueprint summarizes it.

Traces are great for engineers and terrible for everyone else. agent-panorama

translates tool calls, retries, token usage, and errors into plain English. It also pulls the real user request and final answer out of LangGraph/LangChain messages

payloads, so the report reads like a story, not a JSON dump:

get_weather({"city": "Paris"})

β†’**"Looked up the weather"**- 3 failed model calls β†’ "High retry count: 3 failed attempts before completing." human_handoff(...)

β†’ run outcomehuman-escalated

Tokens are the primary metric. USD cost is opt-in (since v0.2): supply a model_prices

table in your config and the report adds dollar estimates alongside tokens (no prices β‡’ cost stays hidden).

pip install agent-panorama
uv pip install -e ".[dev]"

Requires Python 3.10+. Dependencies are intentionally minimal: click

, jinja2

, pyyaml

, python-dotenv

.

agent-panorama generate --input traces.json --output ./report --format html

Options:

Option Description
--input
Path, glob, or directory of JSON exports. Repeatable; globs/dirs are expanded (required).
--output
Output directory (default ./report ).
--format
md , html , json , or both (= md+html; default both ).
--input-type
langfuse or langsmith (default langfuse ).
--config
Optional YAML config (tool naming, thresholds, model_prices ).
--detail
Step narrative detail: minimal , standard (default), or richer .
--session
Keep only runs matching this session id.
--since / --until
Keep only runs whose start time is within this ISO date/datetime window (UTC).
--summarize
Phrase each minimal result via a cheap LLM (opt-in, off by default). See below.
--summarize-model
LLM id for --summarize (default google_genai:gemini-2.5-flash-lite ).

Try it on the bundled example, or aggregate a whole fleet:

agent-panorama generate --input examples/langfuse_traces.json --output ./report

agent-panorama generate --input 'traces/*.json' --input more/ \
  --since 2026-05-01 --until 2026-05-31 --format json --output ./report

Multiple --input

flags, glob patterns, and directories are all expanded and aggregated into one report. The report then carries a cross-agent activity feed and per-agent rollups (runs, actions, success/escalation/retry rates, tokens, and cost when model_prices

is set). --format json

writes a report.json

with a stable contract (generated_at

, time_range

, totals

, feed

, rollups

, decision_log

) consumed by the frontend dashboard.

Add a model_prices

table to your config to get dollar estimates next to tokens (prices are USD per 1M tokens; keys match model names by substring, longest match wins):

model_prices:
  gpt-4o-mini: { input: 0.15, output: 0.60 }
  gpt-4o:      { input: 2.50, output: 10.00 }
  claude-3-5-sonnet: { input: 3.00, output: 15.00 }

With no model_prices

block, cost is omitted entirely and tokens remain the only metric.

--format json

writes a report.json

with a stable shape (also the input the dashboard consumes). Every timestamp is ISO-8601 UTC or null

; every *cost_usd

is a number or null

(null when no model_prices

matched). outcome

is one of success

, human-escalated

, failure

, unknown

.

{
  "generated_at": "2026-05-31T09:42:00+00:00",
  "time_range": { "start": "…", "end": "…" },
  "totals":     { "runs": 4, "steps": 7, "tokens": 3990, "cost_usd": 0.0134,
                  "value": null },           // value summary when the value layer is on
  "feed": [                                  // one entry per run, newest first
    {
      "run_id": "…", "agent_name": "research-assistant",
      "agent_key": "research-assistant",     // slug, for stable UI grouping/colour
      "action": "Searched the web and summarized 3 papers.",
      "outcome": "success", "timestamp": "…",
      "retry_count": 0, "anomaly_count": 0,
      "tokens": 1234, "cost_usd": 0.006,
      "summary": "…", "facts": [["Steps", "5"], ["Retries", "0"]],
      "anomalies": [],
      "value": null                          // ValueJudgment when judged (see value layer)
    }
  ],
  "rollups": [                               // one per agent
    {
      "agent_name": "research-assistant", "agent_key": "research-assistant",
      "runs": 1, "actions": 5,
      "success_rate": 1.0, "escalation_rate": 0.0,
      "failure_rate": 0.0, "retry_rate": 0.0,
      "total_tokens": 1234, "total_cost_usd": 0.006,
      "judged": 0, "avg_value_score": null,  // value layer metrics (null when off)
      "valuable_rate": null, "cost_per_valuable_usd": null
    }
  ],
  "decision_log": [                          // consequential actions across agents
    { "timestamp": "…", "agent_name": "…", "action": "…",
      "parameters": "…", "outcome": "succeeded" }
  ]
}
python
from agent_panorama import generate_report

report = generate_report(
    "traces.json",
    output_dir="./report",
    formats=["md", "html"],
    input_type="langfuse",
    config="config.yaml",  # optional
)

print(report.total_runs, report.total_tokens)

generate_report

returns the in-memory Report

, so you can also inspect runs, the decision log, and anomalies programmatically without touching disk (use build_report_from_file

if you want the report without writing files).

generate_report

(and the lower-level build_report_from_inputs

) accept a glob, a directory, or a list of paths via inputs=

, plus session

/ since

/ until

filters. The returned Report

exposes the cross-agent feed

and per-agent rollups

; serialize_report

gives you the report.json

dict directly.

from agent_panorama import (
    generate_report, build_report_from_inputs, load_runs,
    load_config, serialize_report,
)

report = generate_report(
    inputs=["traces/*.json", "more/"],   # globs, dirs, or a single path
    formats=["json"],                    # writes report.json
    since="2026-05-01", until="2026-05-31",
    config="config.yaml",                # model_prices here β‡’ cost is populated
)

for item in report.feed:                 # newest-first activity feed
    print(item.agent_name, item.action, item.outcome.value, item.tokens, item.cost_usd)

for r in report.rollups:                 # per-agent success/escalation/retry rates
    print(r.agent_name, r.runs, r.success_rate, r.escalation_rate, r.retry_rate)

runs = load_runs("traces/*.json", session="abc123")
mem = build_report_from_inputs("traces/*.json", "langfuse", load_config("config.yaml"))
payload = serialize_report(mem, load_config("config.yaml"))  # -> dict

Summary- time range, total runs, total steps, total tokens (and total cost whenmodel_prices

is set).Fleet activity feed*(v0.2)- one scannable, newest-first line per run across every agent: who did what, in plain English, with outcome and timing.Per-agent rollups(v0.2)*- one row per agent: runs, actions, and success / escalation / retry rates, plus tokens and cost.Per-agent section- what it was asked to do, what it did step by step (graph nodes / tool calls in plain English, at the chosen--detail

level), final outcome, and a confidence signal (retries / fallback).Decision log- a sortable table of every consequential action: timestamp, agent, action, parameters summarized in plain English, outcome.Anomalies- high retry counts, slow runs, high activity, errors, fallbacks.

All configuration is optional. See config.example.yaml for the full set. Highlights:

tool_descriptions:
  get_weather: "Looked up the weather"

consequential_tools: [send_email, human_handoff]
escalation_tools: [human_handoff, handoff_to_agent]

anomaly_thresholds:
  max_retries: 2
  max_latency_seconds: 30
  max_tool_calls: 15

By default the report uses no LLM - it just reformats trace data. But in --detail minimal

, a long final answer (e.g. a big Markdown table) is condensed with a simple heuristic, which keeps the agent's own wording ("Here are all the open support tickets"). If you'd rather get a crisp past-tense action line that keeps the identifying details and the bottom-line takeaway ("Resolved Acme Corp's billing question - refund issued, ticket closed."), enable the opt-in --summarize

flag, which rewrites just the result via a cheap model.

It is intentionally tiny: a ~40-token fixed system prompt, at most ~250 input tokens (the result is hard-capped at 1,000 characters), and a ~25-token reply

  • roughly 300 tokens total per run. On a free-tier model this costs nothing; on the cheapest paid model it's a fraction of a cent.

Install a provider extra (pick the one matching your model):

pip install "agent-panorama[gemini]"     # Google Gemini (recommended, free tier)
pip install "agent-panorama[openai]"     # OpenAI
pip install "agent-panorama[anthropic]"  # Anthropic

Get your own API key from the provider and either

export

it or put it in a.env

file in the working directory (auto-loaded; real env vars win):

export GOOGLE_API_KEY=...      # Gemini

Run with

--summarize

:

agent-panorama generate --input traces.json --output ./report \
  --detail minimal --summarize
agent-panorama generate --input traces.json --output ./report \
  --detail minimal --summarize --summarize-model openai:gpt-5-nano

If the provider package or key is missing, summarization is skipped gracefully (you just get the heuristic line) - it never breaks report generation.

Every call is logged to ** <output>/llm_calls.log** - the exact system prompt, the input sent (with its character count), and the output (or error) for each run - so you can audit precisely what went to the model.

For this tiny one-shot call any of these is more than capable, so free-tier access and price dominate. Only Gemini Flash / Flash-Lite have a genuine no-credit-card free tier; OpenAI/Anthropic require a positive balance.

Model (--summarize-model ) | Price /1M (in β†’ out) | Free tier | Provider extra | API key env var | |---|---|---|---|---| google_genai:gemini-2.5-flash-lite (default) | $0.10 β†’ $0.40 | βœ… free, no card (~1,500 req/day) | gemini | GOOGLE_API_KEY | google_genai:gemini-2.5-flash | $0.30 β†’ $2.50 | βœ… free tier (lower quota) | gemini | GOOGLE_API_KEY | openai:gpt-5-nano | $0.05 β†’ $0.40 | openai | OPENAI_API_KEY | | openai:gpt-4.1-nano | $0.10 β†’ $0.40 | openai | OPENAI_API_KEY | | openai:gpt-4o-mini | $0.15 β†’ $0.60 | openai | OPENAI_API_KEY | | anthropic:claude-haiku-4-5 | $1.00 β†’ $5.00 | anthropic | ANTHROPIC_API_KEY |

Pick google_genai:gemini-2.5-flash-lite (the default) to run this for free.

gpt-5-nano

has the lowest paid input price if you already use OpenAI. Prices verified May 2026 against providers' official pricing pages; check them for current rates.

Langfuse trace exports - a single trace dict, the single-trace{"trace": {...}, "observations": [...]}

shape, a list of traces, or the{"data": [...]}

list-API shape. Tool calls are read fromTOOL

observations (falling back to tool spans), and fromtoolCalls

/ OpenAI-styletool_calls

declared on generations.LangSmith run exports - a flat list (or{"runs": [...]}

) of run nodes; each root run is flattened into one agent run.

Token usage is read from the trace (inputUsage

/outputUsage

or usage

/usage_metadata

). Dollar cost is opt-in via a model_prices

config table (see USD cost).

A manager-facing Agent Panorama dashboard lives in frontend/ (Vite + React + TypeScript, outside the Python package). It renders the

report.json

produced by --format json

, falling back to bundled demo data when no JSON is present.See frontend/README.md

for setup; in short:

agent-panorama generate --input 'traces/*.json' --format json --output ./report
cp report/report.json frontend/public/feed.json
cd frontend && npm install && npm run dev

Watch your agents live instead of from after-the-fact exports. One line in any LangChain / LangGraph app streams every completed run to a local dashboard:

from agent_panorama.live import PanoramaCallbackHandler

agent.invoke(inputs, config={"callbacks": [PanoramaCallbackHandler()]})

Then run the dashboard server (one-time install of the live

extra):

pip install 'agent-panorama[live]'
agent-panorama serve --open        # dashboard at http://localhost:8321

Each run appears in the activity feed within seconds of finishing - outcome, tool calls, tokens, anomalies, and per-agent rollups all update live (the dashboard polls /api/report

every 3 s).

Designed to be safe in the instrumented app:

  • The handler ships with the base package and posts runs over the standard library - your agent app never needs the server dependencies.
  • Delivery never raises and never blocks beyond a 2 s timeout: if the dashboard is down, the app logs one warning and keeps working.
  • The server keeps runs in memory ( --max-runs

caps retention) and applies the same analysis as batch reports, so outcomes/anomalies matchgenerate

.

Useful flags: --port

, --host

, --config your.yaml

(same YAML as generate

  • tool descriptions, escalation tools, model prices), --max-runs

. Point the handler elsewhere with PanoramaCallbackHandler(endpoint=...)

or the AGENT_PANORAMA_ENDPOINT

env var.

A chat agent answering 4 questions is still doing one thing for one user - so the feed aggregates by (session, actor). Pass them in the invoke config (LangGraph's thread_id

works automatically):

agent.invoke(inputs, config={
    "callbacks": [PanoramaCallbackHandler()],
    "metadata": {"session_id": "support-42", "user_id": "user-7"},
})

All turns of that pair collapse into a single feed entry with an Interactions: 4 Β· 3 ok Β· 1 failed

breakdown, the worst turn's outcome as the status, and summed tokens/cost. An LLM layer then phrases the whole session in one line - keeping the identifying details and the outcome, e.g. "Worked through Acme Corp's onboarding - integration is live, handed back to their team." - using the same cheap model as --summarize

(install a provider extra such as agent-panorama[gemini]

and set its API key; without one, a deterministic summary line is shown instead). Override the model with serve --summarize-model ...

. Batch reports (generate

) aggregate the same way - Langfuse's native sessionId

/userId

are picked up automatically. Runs without a session id stay one-entry-per-run.

Try it without LangChain: start agent-panorama serve --open

, then run python examples/live_demo.py

to stream three synthetic runs into the dashboard. More demos live in examples/, organized by complexity (

one_step/

, two_step/

, multi_step/

) - including a real LangChain example in examples/one_step/langchain_agent.py

.The activity feed tells you what your agents did. The value layer tells you whether it mattered - judged against your definition of value, not a generic rubric. An LLM judge reads each conversation (batch exports and live mode alike) and produces a ValueJudgment

: scores 0-10, the outcome in your domain language, the concrete moments value was delivered or lost, actionable fixes, and a pass/fail verdict per success criterion.

Enable it by adding a value:

block to your YAML config (no new install - it uses the same provider extra and API key as --summarize

):

value:
  judge_model: google_genai:gemini-2.5-flash   # default; any init_chat_model id
  max_judgments: 50            # hard cap per report - the cost guard
  include_single_runs: true    # false = judge only multi-turn sessions
  default:                     # your definition of value (the generic fallback)
    domain: customer support
    user_goal: resolve the user's issue without human escalation
    success_criteria:
      - issue resolved in the conversation
    custom_dimensions:
      self_service: Did the user finish without needing a human?
  contexts:                    # per-agent overrides, keyed by agent_key
    kb-assistant:
      domain: customer support
      user_goal: the user resolves their issue

A fleet rarely has one goal, so contexts are per agent: each agent's entry merges field-wise over default

. With model_prices

also configured, every agent gets the number managers actually want - cost per valuable conversation (total spend Γ· conversations scoring β‰₯ 6).

In the dashboard this appears as a second Value view (it shows up in the sidebar only when something was judged): fleet averages, a per-agent value table, and conversations sorted lowest-value first - because the manager's job is finding lost value. Judged feed cards carry a score pill, and the detail panel shows the full verdict.

Cost notes: each judgment is one capped LLM call (transcript hard-capped at ~8k chars); max_judgments

bounds batch reports, and live mode caches one judgment per conversation, re-judging only when a new turn arrives. Every call is audited to llm_calls.log

. Without a provider/key, judging degrades silently - the report still generates, just unjudged.

Managers don't have to hand-write the value:

block. The live dashboard has a Value Ontology section that builds it with them:

  • A guided wizard asks one plain-language question at a time - who the agent serves, the user's goal, what success looks like, how it fails, what's at stake - while a live constellation map fills in as they answer. "Help me figure out" proposes domain-specific examples (LLM-phrased with a provider key; plain deterministic questions without one). - On finish, each agent gets a Value Blueprint: a one-glance briefing - an executive summary, a completeness score, the ontology snapshot (click to expand), a plain-language "how value is created" narrative, and success-criteria / value-dimension / failure-mode / stakes cards, plus a fleet comparison. Switch between agents with the top pills, re-open the wizard to edit, or define a new agent's ontology from scratch.

Definitions are saved by agent-panorama serve

to a sidecar in --data-dir

and override the YAML value: block, so the judge re-maps and re-judges with the manager's own words.

agent-panorama

starts as a report generator and is growing into an oversight layer for fleets of agents - a single pane of glass for everything your agents did, decided, and got wrong. More than logs, across more than one agent.

βœ… v0.1 - Read one run clearly (today)

  • Langfuse + LangSmith trace ingestion
  • Plain-language per-agent summaries, decision log, anomalies
  • Markdown + self-contained HTML output; CLI and library API

βœ… v0.2 - See the whole fleet (the panorama view)

A unified

cross-agent activity feed- one scannable timeline of what every agent did, in plain English:

Agent Activity - May 28, 14:30-15:00

research-assistant    β†’ searched the web, summarized 3 papers            βœ“ success
scheduling-assistant  β†’ checked the calendar, handed the task to a human ‴ escalated
weather-assistant     β†’ looked up the weather (retried once), emailed it βœ“ success
billing-agent         β†’ issued 2 refunds, flagged 1 for review           ⚠ anomaly

Aggregate many traces into one report (by session, time window, or file glob)

Per-agent rollups: runs, actions, success / escalation / retry rates

Cross-agent decision log spanning every agent in the window

βœ… v0.3 - Continuous oversight: the live dashboard

  • One-line LangChain/LangGraph integration ( PanoramaCallbackHandler

) agent-panorama serve

  • a local server with the dashboard bundled in- Runs stream in as they finish; feed, rollups, and totals update live

βœ… v0.4 - The value layer: was it worth it?

  • LLM-as-judge scores every conversation against yourvalue definition (domain, user goal, success criteria, custom dimensions - per agent) - Value delivered / value lost / recommended fixes, cited from the transcript
  • A second dashboard view: avg value score, valuable rate, and cost per valuable conversation - A Value Ontology builder in the dashboard: a guided wizard plus a per-agentValue Blueprint so managers define value without touching YAML

πŸ“ˆ v0.5 - Trends & regressions

  • Track rates over time, not just a point-in-time snapshot
  • Flag regressions (escalations or retries spiking vs. a baseline)
  • Period-over-period comparison ("this week vs. last")

πŸ”Œ v0.6 - More frameworks & sources

  • One-line callbacks/adapters for more agent frameworks - CrewAI, AutoGen / AG2, the OpenAI Agents SDK, AWS Strands, and more (today: LangChain / LangGraph)
  • OpenTelemetry / OpenInference and raw OpenAI-style logs
  • Optionally fetch full input/output from the Langfuse API to enrich decision-log parameters
  • Pluggable parser interface for custom trace formats

🎯 The vision - Full continuous oversight

  • In-flight runs on the live dashboard (watch a run while it's still working)
  • Scheduled/continuous reports instead of one-off runs
  • Accountability views a non-engineer can sign off on (what happened, what needs a human)
  • Alerting on anomalies across the fleet

Have a use case or a trace format you want supported? Open an issue.

uv pip install -e ".[dev]"
python tests/run_all_tests.py     # run the full suite
ruff check . && ruff format --check .

Contributions are very welcome - and kept deliberately easy. No CLA, no strict process, no style police. If you use agents and want better reports, jump in.

Good first things to do:

  • Add a parser for a trace format you use (see the registry in parsers/__init__.py

  • writeparse(payload) -> list[AgentRun]

and register it; nothing downstream changes). - Improve a plain-language summary, fix a parsing edge case, or polish the report.

  • Open an issue with a (scrubbed) trace that doesn't render well - that alone helps a lot.

The whole flow:

  • Fork & branch.
  • Make your change. Run ruff check . && ruff format .

andpython tests/run_all_tests.py

(a green suite is all that's expected - add a test if it makes sense, but don't sweat it). - Open a PR. Rough is fine - we'll iterate together.

Questions, ideas, half-finished patches: all welcome. Star the repo, open an issue, or just say hi. πŸ™Œ

MIT - see LICENSE.

── more in #artificial-intelligence 4 stories Β· sorted by recency
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain β€” perfect for shipping the agent you just read about.

$git push zahid main
β†’ Live at https://your-agent.zahid.host βœ“
Get free account β†’ Pricing
from €0/mo Β· no card required
LIVE [news/nobody-is-measuring-…] indexed:0 read:18min 2026-06-14 Β· β€”