LangChain agent middleware for Cycles — pre-execution budget authority for model calls, tool calls, and runaway agent loops in Python create_agent workflows.
Project description
Cycles for LangChain — AI agent middleware for budget and action authority
LangChain middleware for pre-execution budget authority over model calls, tool calls, and runaway agent loops in create_agent workflows. Provider-neutral: works with any LangChain 1.x agent regardless of model provider, as long as actions flow through LangChain middleware/tool execution.
Built on LangChain's AgentMiddleware API:
wrap_model_call— pre-model-call authorization plus optional reserve/commit/release lifecycle around each LLM invocation (v0.1.5+)wrap_tool_call— tool-call authorization plus optional reserve/commit/release lifecycle around each tool executionbefore_model(with@hook_config(can_jump_to=["end"])) — fan-out caps and external policy halts before another model turn
Per-call actual-cost extraction is available on CyclesModelGate via the cost_fn parameter (v0.2.0+): supply a Callable[[ModelResponse], Amount] and commits debit at actual provider-reported token usage instead of the configured estimate. langchain_runcycles.extractors ships openai_cost and anthropic_cost factories parameterized by per-million-token pricing. For non-agent LangChain code (bare chains, RAG runnables), the BaseCallbackHandler recipe in cycles-client-python/examples/langchain_integration.py remains the right tool.
Install via pip install langchain-runcycles.
What's in the box
CyclesModelGate(v0.1.5+) — runs before every model call. Authorizes viaclient.decide()and/or reserves budget. Returns aModelResponsecarrying the denial reason on deny so the agent terminates naturally.CyclesToolGate— runs before every tool call. Authorizes viaclient.decide()and/or reserves budget viaclient.create_reservation(). Returns aToolMessageon denial so the model can recover gracefully.CyclesFanOutGate— runs before every model turn. Halts the agent (withjump_to: "end") when a turn cap is hit or when an external policy says to stop. Useful for runaway-loop protection and per-tenant burst caps.
All three work with sync or async LangChain agents and the sync (CyclesClient) or async (AsyncCyclesClient) Cycles client. Compose them in a single middleware=[...] list — typical order is [CyclesFanOutGate, CyclesModelGate, CyclesToolGate] so fan-out caps trigger before model spend before tool side effects.
Installation
pip install langchain-runcycles
Requires Python 3.10+ and langchain >= 1.0.
Quick Start
from langchain.agents import create_agent
from langchain_core.tools import tool
from langchain_runcycles import CyclesToolGate
from runcycles import Action, CyclesClient, CyclesConfig, Subject
@tool
def send_email(to: str, body: str) -> str:
"""Send an email."""
return f"Sent to {to}"
client = CyclesClient(CyclesConfig(base_url="http://localhost:7878", api_key="..."))
gate = CyclesToolGate(
client,
subject=Subject(tenant="acme", agent="researcher"),
action={"send_email": Action(kind="tool.call", name="send_email")},
mode="decide",
)
agent = create_agent(model="claude-sonnet-4-6", tools=[send_email], middleware=[gate])
agent.invoke({"messages": [{"role": "user", "content": "Email alice."}]})
If client.decide() denies the call, send_email is never invoked — the model receives a ToolMessage with the denial reason and can choose another path.
Middleware
CyclesModelGate (v0.1.5+)
Gates each model call. Same three modes as CyclesToolGate. On denial in decide mode, returns a ModelResponse whose AIMessage carries the denial reason — the agent terminates naturally because the AIMessage has no tool_calls.
from langchain_runcycles import CyclesModelGate
model_gate = CyclesModelGate(
client,
subject=Subject(tenant="acme", agent="researcher"),
action=Action(kind="llm.completion", name="gpt-4o"),
mode="reserve",
estimate=Amount(unit=Unit.USD_MICROCENTS, amount=2_000_000), # $0.02 per call
)
Add
cost_fn=openai_cost(prompt_per_million_usd=2.50, completion_per_million_usd=10.00)(oranthropic_cost(...), or a customCallable[[ModelResponse], Amount]) to commit at actual reported token usage instead ofestimate(v0.2.0+). See the "Actual-cost extraction onCyclesModelGate" section below for the full pattern.
CyclesToolGate
Gates each tool call. Three modes:
| Mode | What it does |
|---|---|
"decide" |
Calls client.decide(). Denies the tool call on a non-allow decision. No reservation. |
"reserve" |
Creates a reservation, runs the tool, commits on success / releases on exception. |
"decide+reserve" |
Authorizes via decide(), then reserves+commits. Most strict. |
gate = CyclesToolGate(
client,
subject=Subject(tenant="acme", agent="researcher"),
action={
"search": Action(kind="tool.call", name="search"),
"send_email": Action(kind="tool.call", name="send_email"),
},
mode="decide+reserve",
)
CyclesFanOutGate
Halts the agent when a turn cap or external policy says stop. Optional client argument enables remote policy checks on each turn:
from langchain_runcycles import CyclesFanOutGate
fanout = CyclesFanOutGate(
max_turns=20,
client=client, # optional — for remote policy
subject=Subject(tenant="acme"),
action=Action(kind="model.turn", name="research"),
)
Pair with CyclesToolGate and HumanInTheLoopMiddleware for production-grade agent governance.
Configuration
Subject
Either a static Subject or a callable resolving from request/state:
from runcycles import Subject
# Static
subject = Subject(tenant="acme", agent="bot")
# Per-call extractor (CyclesToolGate: (request, state); CyclesFanOutGate: (state, state))
def per_tenant(request, state):
return Subject(tenant=state["config"]["tenant"], agent="bot")
Action
Static, mapping (per-tool name), or callable:
from runcycles import Action
# Static
action = Action(kind="tool.call", name="any")
# Per-tool mapping
action = {
"send_email": Action(kind="tool.call", name="send_email"),
"search": Action(kind="tool.call", name="search"),
}
# Callable
def derive(request):
return Action(kind="tool.call", name=request.tool_call["name"])
Idempotency-key namespacing (v0.1.3+)
Cycles idempotency keys default to {prefix}-{tool_call_id} — deterministic per tool call so retries land on the same reservation. If your runtime can reuse short tool-call ids across runs (tc_1, tc_2, ...), set idempotency_namespace on the middleware to scope keys by run / workflow / tenant. Keys then become {prefix}-{namespace}-{tool_call_id}.
# Static — same namespace every call
gate = CyclesToolGate(
client,
subject=Subject(tenant="acme"),
action=Action(kind="tool.call", name="send_email"),
idempotency_namespace="run_2026_05_10_abc",
)
# Callable — receives the LangChain ToolCallRequest. Pull the run id from
# wherever your runtime carries it: request state, a contextvar, your own
# middleware, etc.
def my_run_id(request):
return request.state["run_id"]
gate = CyclesToolGate(
client,
subject=Subject(tenant="acme"),
action=Action(kind="tool.call", name="send_email"),
idempotency_namespace=my_run_id,
)
CyclesFanOutGate.idempotency_namespace is the same shape; the callable receives the agent state instead of the tool-call request. Without idempotency_namespace, keys keep the v0.1.2 shape exactly — no behavior change.
Per-call opt-out: a callable that returns None (or empty string) for a particular call disables namespacing for that call only, producing the v0.1.2 shape {prefix}-{tool_call_id}. Useful when some calls should be globally scoped (admin / system tools) while others get run-scoped namespacing — branch on the request and return None from the unscoped path.
Errors in the callable propagate: if your callable raises, the exception surfaces from wrap_tool_call / before_model to the agent. This is intentional — fail-fast on a misconfigured callable rather than silently producing keys with no namespace. Wrap in try/except inside the callable if you want a fallback.
Actual-cost extraction on CyclesModelGate (v0.2.0+)
Reserve-mode model calls commit at the configured estimate by default. Pass a cost_fn to commit at actual provider-reported token usage instead:
from langchain_runcycles import CyclesModelGate
from langchain_runcycles.extractors import anthropic_cost, openai_cost
from runcycles import Action, Amount, Subject, Unit
# OpenAI gpt-4o pricing (2026-05): $2.50/M input, $10.00/M output
gate = CyclesModelGate(
client,
subject=Subject(tenant="acme"),
action=Action(kind="llm.completion", name="gpt-4o"),
mode="reserve",
estimate=Amount(unit=Unit.USD_MICROCENTS, amount=2_000_000), # worst-case headroom
cost_fn=openai_cost(prompt_per_million_usd=2.50, completion_per_million_usd=10.00),
)
# Anthropic claude-sonnet-4-6 pricing (2026-05): $3.00/M input, $15.00/M output
gate = CyclesModelGate(
client,
subject=Subject(tenant="acme"),
action=Action(kind="llm.completion", name="claude-sonnet-4-6"),
mode="reserve",
estimate=Amount(unit=Unit.USD_MICROCENTS, amount=2_500_000),
cost_fn=anthropic_cost(input_per_million_usd=3.00, output_per_million_usd=15.00),
)
Both factories read AIMessage.usage_metadata (LangChain's normalized usage shape, populated by langchain-openai and langchain-anthropic) and return an Amount in USD_MICROCENTS. Pricing arguments are keyword-only so they can't be swapped accidentally.
You can also pass a custom cost_fn: Callable[[ModelResponse], Amount] — the middleware calls it after the wrapped handler returns and uses the returned Amount for the commit. If your callable raises or returns a non-Amount, the gate logs a warning and falls back to estimate — a costing bug never erases the model result.
Denial messages
denial_message accepts a format string (placeholders: {reason}, {tool}, {decision}) or a callable receiving the CyclesResponse:
gate = CyclesToolGate(
client,
subject=...,
action=...,
denial_message="Cycles denied {tool}: {reason}",
)
Error handling
- Denied tool calls return a
ToolMessagewith the denial content; the underlying handler is never invoked. The agent's model sees the denial as if a tool returned an error and can recover. - Reservation failures in
"reserve"mode are returned asToolMessage(handler not invoked). - Tool exceptions in
"reserve"mode trigger an automaticrelease_reservation, then the exception propagates. - Async/sync mismatch raises
TypeError— pairCyclesClientwith.invoke()andAsyncCyclesClientwith.ainvoke().
Settlement (commit) failures
In "reserve" and "decide+reserve" modes, the tool runs first, then the reservation is committed. If the commit call itself fails (network blip, server overload, etc.), the tool already ran — its side effect is real. You have two reasonable options, controlled by settlement_error_policy:
| Policy | Behavior | When to choose |
|---|---|---|
"raise" (default) |
Propagate the commit exception to the agent. The tool's return value is lost. | Strict governance — no tool-level cost can go unaccounted. |
"log" |
Log a warning, return the tool result anyway. The reservation will eventually expire via TTL. | UX-first — keep the agent moving, accept best-effort accounting. |
gate = CyclesToolGate(
client,
subject=...,
action=...,
mode="reserve",
settlement_error_policy="log", # opt out of strict default
)
Trade-off worth understanding: "raise" surfaces the commit failure as a tool exception, so a LangChain agent may retry — at which point the tool's side effect (e.g. an email send, a payment, a CRM write) repeats. Choose "log" if your tool's side effects are not safely idempotent on retry.
This only affects commit (success-path settlement); release on tool failure always logs and continues so the original tool exception wins.
Async support
Async middleware variants run automatically when the LangChain agent is invoked with .ainvoke(). Pass an AsyncCyclesClient:
from runcycles import AsyncCyclesClient
async_client = AsyncCyclesClient(CyclesConfig(...))
gate = CyclesToolGate(async_client, subject=..., action=..., mode="decide")
agent = create_agent(model="...", tools=[...], middleware=[gate])
await agent.ainvoke({"messages": [...]})
Examples
examples/tenant_budget_agent.py— single-tenant budget gate with risky-tool denial recovery.examples/multi_agent_fanout.py— multi-agent / HITL flow withCyclesToolGate+CyclesFanOutGate+HumanInTheLoopMiddleware.
Known limitations (v0.2)
CyclesToolGatereserve mode commits at the configuredestimate, not actual usage. Per-tool actual-cost instrumentation (analogous toCyclesModelGate.cost_fn) is still on the roadmap; setestimateto the worst-case spend per call you're willing to debit, or usemode="decide"for policy gating without budget movement.- Streaming verification deferred.
CyclesModelGate.wrap_model_callruns once per model turn and the commit fires after the handler returns; we haven't yet shipped explicit tests againstastream/astream_eventsto confirm the commit waits for stream completion. Tracked on the v0.2.x roadmap (issue #13). - Per-call subject only via the extractor form. Static
Subjectpins one tenant per middleware instance. For per-tenant/per-agent routing in a multi-tenant deployment, supply aSubjectExtractorcallable. - Idempotency keys are deterministic only when
tool_call_idis present. Keys take the shape{prefix}-{tool_call_id}so retries land on the same Cycles reservation. If the upstream omitstool_call_id, the middleware synthesizes a freshmissing-<hex>id (and logs a warning) — that path is non-deterministic across retries because the synthesis itself is random. Conformant LangChain runtimes always supplyid.
Development
pip install -e ".[dev]"
pytest # all tests
pytest --cov=langchain_runcycles # with coverage (gate: ≥95%)
ruff check . && ruff format
mypy langchain_runcycles
Documentation
- LangChain integration page: https://docs.langchain.com/oss/python/integrations/middleware/runcycles (pending PR review)
- Cycles protocol & SDK: https://runcycles.io
- Architecture: see AUDIT.md
Requirements
- Python 3.10+
runcycles >= 0.4.1langchain >= 1.0, < 2.0langchain-core >= 1.0, < 2.0
License
Apache-2.0. See LICENSE.
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 langchain_runcycles-0.2.0.tar.gz.
File metadata
- Download URL: langchain_runcycles-0.2.0.tar.gz
- Upload date:
- Size: 58.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
47fff20eb2a2e20bc019fa52ee8bf168a7b8d7eafb539a32b075429587a41c41
|
|
| MD5 |
a53e31ce442fb774258652a96ae4c7f6
|
|
| BLAKE2b-256 |
919ebbb7b704e46db6a45ca17ea366fa28a5be553374a6fa5c519ff7c06cf862
|
Provenance
The following attestation bundles were made for langchain_runcycles-0.2.0.tar.gz:
Publisher:
python-publish.yml on runcycles/langchain-runcycles
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
langchain_runcycles-0.2.0.tar.gz -
Subject digest:
47fff20eb2a2e20bc019fa52ee8bf168a7b8d7eafb539a32b075429587a41c41 - Sigstore transparency entry: 1506847042
- Sigstore integration time:
-
Permalink:
runcycles/langchain-runcycles@6bc44e4eec063e5dce7d8b8454c44c1e573d6a47 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/runcycles
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@6bc44e4eec063e5dce7d8b8454c44c1e573d6a47 -
Trigger Event:
push
-
Statement type:
File details
Details for the file langchain_runcycles-0.2.0-py3-none-any.whl.
File metadata
- Download URL: langchain_runcycles-0.2.0-py3-none-any.whl
- Upload date:
- Size: 27.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
148e36d27721dd67bb1963af1444e99b98621ad748c99317208d9b03b33d2bbe
|
|
| MD5 |
fd986091243a3f7014a9163d4df29ea7
|
|
| BLAKE2b-256 |
b25bb425f4c7ec33251537227c4dee247b5a1428868ba099587fad9ee750e9de
|
Provenance
The following attestation bundles were made for langchain_runcycles-0.2.0-py3-none-any.whl:
Publisher:
python-publish.yml on runcycles/langchain-runcycles
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
langchain_runcycles-0.2.0-py3-none-any.whl -
Subject digest:
148e36d27721dd67bb1963af1444e99b98621ad748c99317208d9b03b33d2bbe - Sigstore transparency entry: 1506847161
- Sigstore integration time:
-
Permalink:
runcycles/langchain-runcycles@6bc44e4eec063e5dce7d8b8454c44c1e573d6a47 -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/runcycles
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@6bc44e4eec063e5dce7d8b8454c44c1e573d6a47 -
Trigger Event:
push
-
Statement type: