Skip to main content

How to dispatch custom callback events

Prerequisites

This guide assumes familiarity with the following concepts:

In some situations, you may want to dipsatch a custom callback event from within a Runnable so it can be surfaced in a custom callback handler or via the Stream Events API.

For example, if you have a long running tool with multiple steps, you can dispatch custom events between the steps and use these custom events to monitor progress. You could also surface these custom events to an end user of your application to show them how the current task is progressing.

To dispatch a custom event you need to decide on two attributes for the event: the name and the data.

AttributeTypeDescription
namestringA user defined name for the event.
dataanyThe data associated with the event. This can be anything, though we suggest making it JSON serializable.
  • Custom callback events can only be dispatched from within an existing Runnable.
  • If using streamEvents, you must use version: "v2" to consume custom events.
  • Sending or rendering custom callback events in LangSmith is not yet supported.

Stream Events API​

The most useful way to consume custom events is via the .streamEvents() method.

We can use the dispatchCustomEvent API to emit custom events from this method.

Compatibility Dispatching custom callback events requires

@langchain/core>=0.2.16. See this guide for some considerations to take when upgrading @langchain/core.

The default entrypoint below triggers an import and initialization of async_hooks to enable automatic RunnableConfig passing, which is not supported in all environments. If you see import issues, you must import from @langchain/callbacks/dispatch/web and propagate the RunnableConfig object manually (see example below). :::

import { RunnableLambda } from "@langchain/core/runnables";
import { dispatchCustomEvent } from "@langchain/core/callbacks/dispatch";

const reflect = RunnableLambda.from(async (value: string) => {
await dispatchCustomEvent("event1", {
reversed: value.split("").reverse().join(""),
});
await dispatchCustomEvent("event2", 5);
return value;
});

const eventStream = await reflect.streamEvents("hello world", {
version: "v2",
});

for await (const event of eventStream) {
if (event.event === "on_custom_event") {
console.log(event);
}
}
{
event: 'on_custom_event',
run_id: '9eac217d-3a2d-4563-a91f-3bd49bee4b3d',
name: 'event1',
tags: [],
metadata: {},
data: { reversed: 'dlrow olleh' }
}
{
event: 'on_custom_event',
run_id: '9eac217d-3a2d-4563-a91f-3bd49bee4b3d',
name: 'event2',
tags: [],
metadata: {},
data: 5
}

If you are in a web environment that does not support async_hooks, you must import from the web entrypoint and propagate the config manually instead:

import { RunnableConfig, RunnableLambda } from "@langchain/core/runnables";
import { dispatchCustomEvent as dispatchCustomEventWeb } from "@langchain/core/callbacks/dispatch/web";

const reflect = RunnableLambda.from(
async (value: string, config?: RunnableConfig) => {
await dispatchCustomEventWeb(
"event1",
{ reversed: value.split("").reverse().join("") },
config
);
await dispatchCustomEventWeb("event2", 5, config);
return value;
}
);

const eventStream = await reflect.streamEvents("hello world", {
version: "v2",
});

for await (const event of eventStream) {
if (event.event === "on_custom_event") {
console.log(event);
}
}
{
event: 'on_custom_event',
run_id: 'dee1e4f0-c5ff-4118-9391-461a0dcc4cb2',
name: 'event1',
tags: [],
metadata: {},
data: { reversed: 'dlrow olleh' }
}
{
event: 'on_custom_event',
run_id: 'dee1e4f0-c5ff-4118-9391-461a0dcc4cb2',
name: 'event2',
tags: [],
metadata: {},
data: 5
}

Callback Handler​

Let’s see how to emit custom events with dispatchCustomEvent.

Remember, you must call dispatchCustomEvent from within an existing Runnable.

import { RunnableConfig, RunnableLambda } from "@langchain/core/runnables";
import { dispatchCustomEvent } from "@langchain/core/callbacks/dispatch";

const reflect = RunnableLambda.from(async (value: string) => {
await dispatchCustomEvent("event1", {
reversed: value.split("").reverse().join(""),
});
await dispatchCustomEvent("event2", 5);
return value;
});

await reflect.invoke("hello world", {
callbacks: [
{
handleCustomEvent(eventName, data, runId) {
console.log(eventName, data, runId);
},
},
],
});
event1 { reversed: 'dlrow olleh' } 9c3770ac-c83d-4626-9643-b5fd80eb5431
event2 5 9c3770ac-c83d-4626-9643-b5fd80eb5431
hello world

You’ve now seen how to emit custom events from within your chains.

You can check out the more in depth guide for stream events for more ways to parse and receive intermediate steps from your chains.


Was this page helpful?


You can also leave detailed feedback on GitHub.