Use this file to discover all available pages before exploring further.
When working with non-deterministic systems that make model-based decisions (e.g., agents powered by LLMs), it can be useful to examine their decision-making process in detail:
Understand reasoning: Analyze the steps that led to a successful result.
Debug mistakes: Identify where and why errors occurred.
Explore alternatives: Test different paths to uncover better solutions.
LangGraph provides time-travel functionality to support these use cases. Specifically, you can resume execution from a prior checkpoint — either replaying the same state or modifying it to explore alternatives. In all cases, resuming past execution produces a new fork in the history.To use time-travel in LangGraph:
Identify a checkpoint in an existing thread: Use the getStateHistory method to retrieve the execution history for a specific thread_id and locate the desired checkpoint_id.
Alternatively, set a breakpoint before the node(s) where you want execution to pause. You can then find the most recent checkpoint recorded up to that breakpoint.
Resume execution from the checkpoint: Use the invoke or stream methods with an input of null and a configuration containing the appropriate thread_id and checkpoint_id.
This example builds a simple LangGraph workflow that generates a joke topic and writes a joke using an LLM. It demonstrates how to run the graph, retrieve past execution checkpoints, optionally modify the state, and resume execution from a chosen checkpoint to explore alternate outcomes.
To build this workflow in this example you need to set up the Anthropic LLM and install the required dependencies:
Install dependencies
npm install @langchain/langgraph @langchain/core
Initialize the LLM:
import { ChatAnthropic } from "@langchain/anthropic";const llm = new ChatAnthropic({ model: "claude-sonnet-4-5-20250929", apiKey: "<your_anthropic_key>"});
Sign up for LangSmith to quickly spot issues and improve the performance of your LangGraph projects. LangSmith lets you use trace data to debug, test, and monitor your LLM apps built with LangGraph. You can also fetch traces from LangSmith to replay and debug production issues locally.
Implement the workflow
The implementation of the workflow is a simple graph with two nodes, one for generating a joke topic, another for writing the joke itself and a state to storing the intermediate values.
import { v4 as uuidv4 } from "uuid";import * as z from "zod";import { StateGraph, StateSchema, GraphNode, START, END, MemorySaver } from "@langchain/langgraph";import { ChatAnthropic } from "@langchain/anthropic";const State = new StateSchema({ topic: z.string().optional(), joke: z.string().optional(),});const model = new ChatAnthropic({ model: "claude-sonnet-4-5-20250929", temperature: 0,});const generateTopic: GraphNode<typeof State> = async (state) => { // LLM call to generate a topic for the joke const msg = await model.invoke("Give me a funny topic for a joke"); return { topic: msg.content };};const writeJoke: GraphNode<typeof State> = async (state) => { // LLM call to write a joke based on the topic const msg = await model.invoke(`Write a short joke about ${state.topic}`); return { joke: msg.content };};// Build workflowconst workflow = new StateGraph(State) // Add nodes .addNode("generateTopic", generateTopic) .addNode("writeJoke", writeJoke) // Add edges to connect nodes .addEdge(START, "generateTopic") .addEdge("generateTopic", "writeJoke") .addEdge("writeJoke", END);// Compileconst checkpointer = new MemorySaver();const graph = workflow.compile({ checkpointer });
How about "The Secret Life of Socks in the Dryer"? You know, exploring the mysterious phenomenon of how socks go into the laundry as pairs but come out as singles. Where do they go? Are they starting new lives elsewhere? Is there a sock paradise we don't know about? There's a lot of comedic potential in the everyday mystery that unites us all!# The Secret Life of Socks in the DryerI finally discovered where all my missing socks go after the dryer. Turns out they're not missing at all—they've just eloped with someone else's socks from the laundromat to start new lives together.My blue argyle is now living in Bermuda with a red polka dot, posting vacation photos on Sockstagram and sending me lint as alimony.
To continue from a previous point in the graphs run, use get_state_history to retrieve all the states and select the one where you want to resume execution.
// The states are returned in reverse chronological order.const states = [];for await (const state of graph.getStateHistory(config)) { states.push(state);}for (const state of states) { console.log(state.next); console.log(state.config.configurable?.checkpoint_id); console.log();}
// This is the state before last (states are listed in chronological order)const selectedState = states[1];console.log(selectedState.next);console.log(selectedState.values);
Output:
['writeJoke']{'topic': 'How about "The Secret Life of Socks in the Dryer"? You know, exploring the mysterious phenomenon of how socks go into the laundry as pairs but come out as singles. Where do they go? Are they starting new lives elsewhere? Is there a sock paradise we don\\'t know about? There\\'s a lot of comedic potential in the everyday mystery that unites us all!'}