Capture decision-id-joined evidence from LangGraph agents. Wraps your checkpoint saver to emit OEP tool_permission_packet.v0 records covering the 19 surfaces LangGraph's StateSnapshot does not bind natively.
Project description
langgraph-oep
langgraph-oep records evidence about important LangGraph tool decisions.
Wrap your existing LangGraph checkpointer, mark the state update that represents a permission decision, and the wrapper writes an OEP tool_permission_packet.v0 JSONL record.
Use it when you need to answer questions like:
- Which policy version allowed this tool call?
- Which model alias and resolved model version were active?
- Which release manifest, trace, actor, cache entry, or scoped credential was bound to the decision?
- Would this old decision be worth replaying against a changed policy, model, cache, or credential surface?
This is an illustration-grade reference implementation: one inspectable wrapping pattern, not a production-certified observability platform.
Install
pip install langgraph-oep
Requires Python 3.10+. CI covers LangGraph 0.2.x, 0.3.x, and 1.x.
When To Use It
Use langgraph-oep if you already use LangGraph checkpoints and want an append-only evidence record for selected tool permission decisions.
You probably do not need it if you only want normal LangGraph resume, time travel, local debugging, or ordinary application logs. LangGraph time travel and OEP-style evidence records solve different problems at different layers.
60-Second Example
This example writes one OEP packet to ./oep-records.jsonl.
from typing import TypedDict
from langgraph.checkpoint.memory import MemorySaver
from langgraph.graph import END, START, StateGraph
from langgraph_oep import (
OEP_EMIT_PERMISSION_CHANNEL,
Actor,
DecisionContext,
LocalJsonlSink,
ModelBinding,
attach_oep_writer,
)
class State(TypedDict, total=False):
tool: dict
requested_action: dict
resource: dict
oep_emit_permission: str
output: str
def inspect_diff(state: State) -> State:
return {
"tool": {"name": "read_diff", "version": "0.1.0", "operation": "read"},
"requested_action": {
"action_type": "inspect_diff",
"name": "inspect repository diff",
"input_ref": "diffs/001.patch",
},
"resource": {
"type": "repository_diff",
"id": "diff_001",
"uri": "diffs/001.patch",
"mutable": False,
},
OEP_EMIT_PERMISSION_CHANNEL: "tool_read_diff_001",
"output": "diff inspected",
}
builder = StateGraph(State)
builder.add_node("inspect_diff", inspect_diff)
builder.add_edge(START, "inspect_diff")
builder.add_edge("inspect_diff", END)
graph = builder.compile(checkpointer=MemorySaver())
attach_oep_writer(graph, sink=LocalJsonlSink("./oep-records.jsonl"))
with DecisionContext(
actor=Actor(type="agent", id="agent_code_review", display_name="Code Review Agent"),
release_manifest_id="rmf_code_review_2026_06",
policy_response={
"policy_ref": {
"engine": "opa",
"package": "data.code_review.permissions",
"policy_id": "tool-permission-policy",
"policy_version": "0.1.0",
"policy_uri": "permissions/tool_permissions.rego",
},
"decision": {
"allow": True,
"reason": "policy allowed read-only diff inspection",
"matched_rule": "allow_read_only_diff_inspection",
"opa_result_ref": "opa://decision/code_review/001",
},
"links": {
"event_ref": "events/code_review_001.json",
"release_manifest_ref": "manifest/code_review_2026_06.json",
"trace_ref": "traces/code_review_001.json",
},
},
policy_bundle_version="sha256:" + "0" * 64,
model_binding=ModelBinding(
alias="claude-sonnet-4-6",
resolved_version="claude-sonnet-4-6-20260512",
provider="anthropic",
),
scoped_credential_lifetime="PT15M",
):
graph.invoke({"output": "started"}, config={"configurable": {"thread_id": "demo-thread"}})
The output is one JSON line. It includes fields like:
{
"schema_version": "oep.tool_permission_packet.v0",
"tool_call_id": "tool_lg_...",
"tool": {"name": "read_diff", "version": "0.1.0", "operation": "read"},
"policy_bundle_version": "sha256:0000...",
"model_alias": "claude-sonnet-4-6",
"decision_id": {
"schema_version": "0.3",
"permission": {
"permission_packet_ref": "pder_lg_...",
"tool_call_id": "tool_lg_...",
"policy_bundle_version": "sha256:0000..."
}
}
}
How It Works
- You keep your existing checkpointer (
MemorySaver, SQLite, Postgres, or custom). attach_oep_writer(...)wraps that checkpointer.- Your graph node writes
tool,requested_action,resource, andOEP_EMIT_PERMISSION_CHANNEL. DecisionContextsupplies the evidence surfaces LangGraph does not know about: actor, release manifest, policy response, model binding, scoped credential lifetime, cache/cost/drift/identity extras.- On checkpoint write, the wrapper validates and writes an OEP packet to the sink.
policy_response is required by default. The package does not evaluate OPA or invent policy evidence; it records the policy result you pass in.
Learn More
- Quickstart details
- Policy response and marker semantics
- The 19 wrapper-injected fields
- Operational caveats
- Relationship to OEP
Examples
Runnable examples live under examples/:
01_code_review_agent- single decision capture02_customer_support_with_credentials- scoped credential lifetime03_rag_pipeline_with_cache- cache identity04_multi_step_orchestration_with_policy- repeated decisions under one policy bundle05_interrupt_resume_with_model_version- approval capture and model-version drift
Boundary
LangGraph's checkpoint design is correct for execution recovery and branched exploration. langgraph-oep records a separate evidence layer for selected decisions. No upstream approval or sponsorship is implied.
Treat emitted JSONL as sensitive operational evidence: records can include actor IDs, resource URIs, trace references, cache identifiers, policy refs, scoped credential metadata, and approval metadata.
License
Apache 2.0.
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 langgraph_oep-0.1.0.tar.gz.
File metadata
- Download URL: langgraph_oep-0.1.0.tar.gz
- Upload date:
- Size: 39.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8503b43c6f9cde15816cc1663d54ed1fab859765c0e8feb729a9f9b292562bc7
|
|
| MD5 |
f962ce7ad88fb77bb5edad62d509f4e5
|
|
| BLAKE2b-256 |
ae596613dda23a741f2829df7766e26cc294d844eed9a2b04959c1495ff85ec7
|
File details
Details for the file langgraph_oep-0.1.0-py3-none-any.whl.
File metadata
- Download URL: langgraph_oep-0.1.0-py3-none-any.whl
- Upload date:
- Size: 23.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c21afdadc8ba039611515228ffe6d333210a2a8ca6bb02a26146cd4126bb94b0
|
|
| MD5 |
638875c5447f726211abc25b302a26ab
|
|
| BLAKE2b-256 |
d1e77afb3a6c696a68f4ac897f7fdf79d8185187593dc8568b771b139d698532
|