Agentic Control Plane governance for LangChain / LangGraph agents. Wrap any tool with @governed; ACP decides allow/deny/redact.
Project description
acp-langchain
Agentic Control Plane governance for LangChain and LangGraph agents.
Wrap any tool with @governed. Before it runs, ACP decides allow / deny / redact based on your workspace's policy, the end user's scopes, rate limits, and PII detection.
Same governance model as Claude Code. If you have workspace policies set up for Claude Code, they apply to LangChain tools automatically.
Install
pip install acp-langchain
Usage (LangGraph ReAct agent)
from fastapi import FastAPI, Header
from langchain_core.tools import tool
from langchain_openai import ChatOpenAI
from langgraph.prebuilt import create_react_agent
from acp_langchain import governed, set_context
app = FastAPI()
# Stack @governed under @tool — the governance check runs inside the
# tool's dispatch.
@tool
@governed("web_search")
def web_search(query: str) -> str:
"""Search the web."""
return my_search(query) # your code, your credentials
@tool
@governed("send_email")
def send_email(to: str, subject: str, body: str) -> str:
"""Send an email on behalf of the user."""
return sendmail(to, subject, body)
@app.post("/run")
def run(prompt: str, authorization: str = Header(...)):
# Bind the end user's JWT to this request's context. Every @governed
# call below carries the user's identity to ACP.
set_context(user_token=authorization.removeprefix("Bearer ").strip())
agent = create_react_agent(
model=ChatOpenAI(model="gpt-4o-mini"),
tools=[web_search, send_email],
)
result = agent.invoke({"messages": [("user", prompt)]})
return {"result": result["messages"][-1].content}
What happens per tool call
- Pre-check — POSTs to ACP
/govern/tool-usewith{ tool_name, tool_input, session_id }+ the user's Bearer JWT. - Decide — ACP evaluates workspace policy, the user's scopes, rate limits, and PII.
- Deny → wrapped function returns
"tool_error: <reason>"— LangChain treats this as the tool's output, the model sees it and adapts. - Allow → your tool runs.
- Post-audit — POSTs to
/govern/tool-outputwith the result. PII scan runs. If ACP returnsredact, the redacted version replaces the output.
View activity
Every tool call shows up in the ACP Activity view, rooted in the end user's identity. Sessions group related calls — one request from one user = one session.
Fail-open
Network errors, timeouts (5s default), gateway errors → the tool proceeds with reason "fail-open". Matches Claude Code behavior. Governance is never a single point of failure for the agent.
Works with plain LangChain too
Not using LangGraph? @governed is framework-agnostic — it just wraps a Python function. Stack under any LangChain tool decorator:
from langchain_core.tools import tool
from langchain.agents import create_tool_calling_agent, AgentExecutor
@tool
@governed("web_search")
def web_search(query: str) -> str: ...
# same @governed, works with AgentExecutor, create_tool_calling_agent,
# create_react_agent, or any custom graph
API
acp-langchain re-exports the full acp-governance API for convenience:
governed(name_or_fn=None) # decorator
set_context(user_token, *, session_id=None, agent_tier=None, agent_name=None)
get_context()
clear_context()
configure(base_url=..., timeout_s=..., client_header=...)
Related
acp-governance— core SDK (this package wraps it)acp-crewai— same story for CrewAI- LangChain integration guide
License
MIT
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 acp_langchain-0.1.0.tar.gz.
File metadata
- Download URL: acp_langchain-0.1.0.tar.gz
- Upload date:
- Size: 3.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 |
f03d98c3dc567f55ee99badea99c229b1f7b53b5aa563d5eb50053e7d4820c9a
|
|
| MD5 |
07c9bdb505a23aa87e59ea83e3422ccf
|
|
| BLAKE2b-256 |
7d9962d4e9fc9dcfefd5234b898476332c86405bd4b832371e4e02a108e60777
|
File details
Details for the file acp_langchain-0.1.0-py3-none-any.whl.
File metadata
- Download URL: acp_langchain-0.1.0-py3-none-any.whl
- Upload date:
- Size: 3.7 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 |
601fc56381e8c7d137c20048bb672ff2f11f538a50c11b39aafbb1b80430fb0b
|
|
| MD5 |
7b6a9eb39f6e7bf95dac65dc8238723f
|
|
| BLAKE2b-256 |
cc3988fe58634ddec793e98d8730b9913e10193b6c922323a537d639000818d0
|