Note
Want to use the Agent Inbox without deploying it yourself? Check out the hosted version here: dev.agentinbox.ai
Before getting started, ensure you have:
- Node.js, and yarn installed
- A LangGraph deployment set up and running (locally, or in production through LangGraph Platform)
- Your LangGraph API key
Note
🎥 Prefer a video walkthrough? Checkout this video for a step-by-step guide.
Tip
You can also view a minimum working version of a LangGraph agent using the Agent Inbox in Python here, and TypeScript here.
To start running the Agent Inbox, first clone the repository and install dependencies:
git clone https://github.com/langchain-ai/agent-inbox.git
cd agent-inbox
yarn install
Once up and running, you'll need to take two actions so that the Agent Inbox can connect to your LangGraph deployment.
-
Add your LangSmith API key: Click the "Settings" button in the sidebar, and enter your LangSmith API key.
-
Create your first inbox by opening the settings popover (bottom left, inside the sidebar), and clicking "Add Inbox". This will open a dialog with three fields:
- Assistant/Graph ID: (required) The name of your LangGraph graph, or an assistant ID. This will be used when sending human responses to your graph.
- Deployment URL: (required) The URL of your LangGraph deployment. This is how the Agent Inbox will connect to your LangGraph deployment for fetching interrupts, and sending human responses.
- Name: (optional) A name for your inbox. This will be used as a label for the inbox.
These values are stored in your browser's local storage, and are only used to connect & authenticate requests to the LangGraph deployment.
In order to use the Agent Inbox with your LangGraph project, you'll need to update all instances of where interrupts are used in your codebase (where you want them to be compatible with the Agent Inbox).
Below, you'll find the interrupt input & output schemas for both Python and TypeScript.
Python Interrupt Schema
class HumanInterruptConfig(TypedDict):
allow_ignore: bool
allow_respond: bool
allow_edit: bool
allow_accept: bool
class ActionRequest(TypedDict):
action: str
args: dict
class HumanInterrupt(TypedDict):
action_request: ActionRequest
config: HumanInterruptConfig
description: Optional[str]
class HumanResponse(TypedDict):
type: Literal['accept', 'ignore', 'response', 'edit']
args: Union[None, str, ActionRequest]
TypeScript Interrupt Schema
export interface HumanInterruptConfig {
allow_ignore: boolean;
allow_respond: boolean;
allow_edit: boolean;
allow_accept: boolean;
}
export interface ActionRequest {
action: string;
args: Record<string, any>;
}
export interface HumanInterrupt {
action_request: ActionRequest;
config: HumanInterruptConfig;
description?: string;
}
export type HumanResponse = {
type: "accept" | "ignore" | "response" | "edit";
args: null | string | ActionRequest;
};
The human interrupt schema is used to define the types of interrupts, and what actions can be taken in response to each interrupt. We've landed on four types of actions:
accept
: Accept the interrupt's arguments, or action. Will send anActionRequest
in theargs
field onHumanResponse
. ThisActionRequest
will be the exact same as theaction_request
field onHumanInterrupt
, but with all keys of theargs
field onActionRequest
converted to strings.edit
: Edit the interrupt's arguments. Sends an instance ofActionRequest
in theargs
field onHumanResponse
. Theargs
field onActionRequest
will have the same structure as theargs
field onHumanInterrupt
, but the values of the keys will be strings, and will contain any edits the user has made.response
: Send a response to the interrupt. Does not require any arguments. Will always send back a single string in theargs
field onHumanResponse
.ignore
: Ignore the interrupt's arguments, or action. Returnsnull
for theargs
field onHumanResponse
.
You can set any combination of these actions in the config
field on HumanInterrupt
.
At the moment, you're required to pass a list of HumanInterrupt
objects in the interrupt
function, however the UI is currently limited to rendering only the first object in the list. (We are open to suggestions for how to improve this schema, so if you have feedback, please reach out to discuss!). The same goes for the HumanResponse
, which the Agent Inbox will always send back as a list with a single HumanResponse
object in it.
action_request
: The action and arguments for the interruptaction
: The name, or title of the action. This is rendered in the Agent Inbox as the main header for the interrupt event.args
: The arguments for the action. E.g tool call arguments.
config
: The configuration for the interruptallow_ignore
: Whether the user can ignore the interruptallow_respond
: Whether the user can respond to the interruptallow_edit
: Whether the user can edit the interruptallow_accept
: Whether the user can accept the interrupt
description
: A description of the interrupt- Should be detailed, and may be markdown. This will be rendered in the Agent Inbox as the description, and is commonly used to include additional context about the interrupt, and/or instructions on how to respond to the interrupt.
To use the Agent Inbox, you'll have to use the interrupt
function, instead of raising a NodeInterrupt
exception in your codebase. To read more about how the interrupt
function works, see the LangGraph documentation: conceptual guide how-to guide (TypeScript docs coming soon, but the concepts & implementation are the same).
Then, when calling the interrupt
function, you should pass an instance of HumanInterrupt
as the interrupt
argument. Then, when the user sends a response to the interrupt, you should expect the return value of the interrupt
function to be an instance of HumanResponse
.
Here's a simple example of using the interrupt function in your LangGraph project:
from typing import TypedDict, Literal, Optional, Union
from langgraph.types import interrupt
def my_graph_function():
# Extract the last tool call from the `messages` field in the state
tool_call = state["messages"][-1].tool_calls[0]
# Create an interrupt
request: HumanInterrupt = {
"action_request": {
"action": tool_call['name'],
"args": tool_call['args']
},
"config": {
"allow_ignore": True,
"allow_respond": True,
"allow_edit": False,
"allow_accept": False
},
"description": _generate_email_markdown(state) # Generate a detailed markdown description.
}
# Send the interrupt request inside a list, and extract the first response
response = interrupt([request])[0]
if response['type'] == "response":
# Do something with the response
Common issues and solutions:
-
Connection Issues
- Verify your
LANGSMITH_API_KEY
is set correctly - Ensure your Deployment URL is accessible
- Check if your LangGraph deployment is running
- Verify your
-
Schema Validation Errors
- Ensure all required fields are present in your interrupt objects
- Verify the types of all fields match the schema
- Check that response handling matches expected types
- Check you're sending a single
HumanInterrupt
object inside a list to theinterrupt
function - Check you're extracting the first object from the response list returned by the
interrupt
function