61488042d8
* feat: Add batch fetching for observations and update documentation - Implemented a new endpoint for fetching multiple observations by IDs in a single request. - Updated the DataRoutes to include a POST /api/observations/batch endpoint. - Enhanced SKILL.md documentation to reflect changes in the search process and batch fetching capabilities. - Increased the default limit for search results from 5 to 40 for better usability. * feat!: Fix timeline parameter passing with SearchManager alignment BREAKING CHANGE: Timeline MCP tools now use standardized parameter names - anchor_id → anchor - before → depth_before - after → depth_after - obs_type → type (timeline tool only) Fixes timeline endpoint failures caused by parameter name mismatch between MCP layer and SearchManager. Adds new SessionStore methods for fetching prompts and session summaries by ID. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * docs: reframe timeline parameter fix as bug fix, not breaking change The timeline tools were completely broken due to parameter name mismatch. There's nothing to migrate from since the old parameters never worked. Co-authored-by: Alex Newman <thedotmack@users.noreply.github.com> * Refactor mem-search documentation and optimize API tool definitions - Updated SKILL.md to emphasize batch fetching for observations, clarifying usage and efficiency. - Removed deprecated tools from mcp-server.ts and streamlined tool definitions for clarity. - Enhanced formatting in FormattingService.ts for better output readability. - Adjusted SearchManager.ts to improve result headers and removed unnecessary search tips from combined text. * Refactor FormattingService and SearchManager for table-based output - Updated FormattingService to format search results as tables, including methods for formatting observations, sessions, and user prompts. - Removed JSON format handling from SearchManager and streamlined result formatting to consistently use table format. - Enhanced readability and consistency in search tips and formatting logic. - Introduced token estimation for observations and improved time formatting. * refactor: update documentation and API references for version bump and search functionalities * Refactor code structure for improved readability and maintainability * chore: change default model from haiku to sonnet 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * feat: unify timeline formatting across search and context services Extract shared timeline formatting utilities into reusable module to align MCP search output format with context-generator's date/file-grouped format. Changes: - Create src/shared/timeline-formatting.ts with reusable utilities (parseJsonArray, formatDateTime, formatTime, formatDate, toRelativePath, extractFirstFile, groupByDate) - Refactor context-generator.ts to use shared utilities - Update SearchManager.search() to use date/file grouping - Add search-specific row formatters to FormattingService - Fix timeline methods to extract actual file paths from metadata instead of hardcoding 'General' - Remove Work column from search output (kept in context output) Result: Consistent date/file-grouped markdown formatting across both systems while maintaining their different column requirements. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * refactor: remove redundant legend from search output Remove legend from search/timeline results since it's already shown in SessionStart context. Saves ~30 tokens per search result. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * Refactor session summary rendering to remove links - Removed link generation for session summaries in context generation and search manager. - Updated output formatting to exclude links while maintaining the session summary structure. - Adjusted related components in TimelineService to ensure consistency across the application. * fix: move skillPath declaration outside try block to fix scoping bug The skillPath variable was declared inside the try block but referenced in the catch block for error logging. Since const is block-scoped, this would cause a ReferenceError when the error handler executes. Moved skillPath declaration before the try block so it's accessible in both try and catch scopes. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * fix: address PR #317 code review feedback **Critical Fixes:** - Replace happy_path_error__with_fallback debug calls with proper logger methods in mcp-server.ts - All HTTP API calls now use logger.debug/error for consistent logging **Code Quality Improvements:** - Extract 90-day recency window magic numbers to named constants - Added RECENCY_WINDOW_DAYS and RECENCY_WINDOW_MS constants in SearchManager **Documentation:** - Document model cost implications of Haiku → Sonnet upgrade in CHANGELOG - Provide clear migration path for users who want to revert to Haiku 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> * refactor: simplify CHANGELOG - remove cost documentation Removed model cost comparison documentation per user feedback. Kept only the technical code quality improvements. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Sonnet 4.5 <noreply@anthropic.com> Co-authored-by: claude[bot] <41898282+claude[bot]@users.noreply.github.com> Co-authored-by: Alex Newman <thedotmack@users.noreply.github.com>
390 lines
11 KiB
Markdown
390 lines
11 KiB
Markdown
# TypeScript SDK V2 interface (preview)
|
|
|
|
Preview of the simplified V2 TypeScript Agent SDK, with session-based send/receive patterns for multi-turn conversations.
|
|
|
|
---
|
|
|
|
<Warning>
|
|
The V2 interface is an **unstable preview**. APIs may change based on feedback before becoming stable. Some features like session forking are only available in the [V1 SDK](/docs/en/agent-sdk/typescript).
|
|
</Warning>
|
|
|
|
The V2 Claude Agent TypeScript SDK removes the need for async generators and yield coordination. This makes multi-turn conversations simpler—instead of managing generator state across turns, each turn is a separate `send()`/`receive()` cycle. The API surface reduces to three concepts:
|
|
|
|
- `createSession()` / `resumeSession()`: Start or continue a conversation
|
|
- `session.send()`: Send a message
|
|
- `session.receive()`: Get the response
|
|
|
|
## Installation
|
|
|
|
The V2 interface is included in the existing SDK package:
|
|
|
|
```bash
|
|
npm install @anthropic-ai/claude-agent-sdk
|
|
```
|
|
|
|
## Quick start
|
|
|
|
### One-shot prompt
|
|
|
|
For simple single-turn queries where you don't need to maintain a session, use `unstable_v2_prompt()`. This example sends a math question and logs the answer:
|
|
|
|
```typescript
|
|
import { unstable_v2_prompt } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
const result = await unstable_v2_prompt('What is 2 + 2?', {
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
console.log(result.result)
|
|
```
|
|
|
|
<details>
|
|
<summary>See the same operation in V1</summary>
|
|
|
|
```typescript
|
|
import { query } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
const q = query({
|
|
prompt: 'What is 2 + 2?',
|
|
options: { model: 'claude-sonnet-4-5-20250929' }
|
|
})
|
|
|
|
for await (const msg of q) {
|
|
if (msg.type === 'result') {
|
|
console.log(msg.result)
|
|
}
|
|
}
|
|
```
|
|
|
|
</details>
|
|
|
|
### Basic session
|
|
|
|
For interactions beyond a single prompt, create a session. V2 separates sending and receiving into distinct steps:
|
|
- `send()` dispatches your message
|
|
- `receive()` streams back the response
|
|
|
|
This explicit separation makes it easier to add logic between turns (like processing responses before sending follow-ups).
|
|
|
|
The example below creates a session, sends "Hello!" to Claude, and prints the text response. It uses [`await using`](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-5-2.html#using-declarations-and-explicit-resource-management) (TypeScript 5.2+) to automatically close the session when the block exits. You can also call `session.close()` manually.
|
|
|
|
```typescript
|
|
import { unstable_v2_createSession } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
await using session = unstable_v2_createSession({
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
|
|
await session.send('Hello!')
|
|
for await (const msg of session.receive()) {
|
|
// Filter for assistant messages to get human-readable output
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log(text)
|
|
}
|
|
}
|
|
```
|
|
|
|
<details>
|
|
<summary>See the same operation in V1</summary>
|
|
|
|
In V1, both input and output flow through a single async generator. For a basic prompt this looks similar, but adding multi-turn logic requires restructuring to use an input generator.
|
|
|
|
```typescript
|
|
import { query } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
const q = query({
|
|
prompt: 'Hello!',
|
|
options: { model: 'claude-sonnet-4-5-20250929' }
|
|
})
|
|
|
|
for await (const msg of q) {
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log(text)
|
|
}
|
|
}
|
|
```
|
|
|
|
</details>
|
|
|
|
### Multi-turn conversation
|
|
|
|
Sessions persist context across multiple exchanges. To continue a conversation, call `send()` again on the same session. Claude remembers the previous turns.
|
|
|
|
This example asks a math question, then asks a follow-up that references the previous answer:
|
|
|
|
```typescript
|
|
import { unstable_v2_createSession } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
await using session = unstable_v2_createSession({
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
|
|
// Turn 1
|
|
await session.send('What is 5 + 3?')
|
|
for await (const msg of session.receive()) {
|
|
// Filter for assistant messages to get human-readable output
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log(text)
|
|
}
|
|
}
|
|
|
|
// Turn 2
|
|
await session.send('Multiply that by 2')
|
|
for await (const msg of session.receive()) {
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log(text)
|
|
}
|
|
}
|
|
```
|
|
|
|
<details>
|
|
<summary>See the same operation in V1</summary>
|
|
|
|
```typescript
|
|
import { query } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
// Must create an async iterable to feed messages
|
|
async function* createInputStream() {
|
|
yield {
|
|
type: 'user',
|
|
session_id: '',
|
|
message: { role: 'user', content: [{ type: 'text', text: 'What is 5 + 3?' }] },
|
|
parent_tool_use_id: null
|
|
}
|
|
// Must coordinate when to yield next message
|
|
yield {
|
|
type: 'user',
|
|
session_id: '',
|
|
message: { role: 'user', content: [{ type: 'text', text: 'Multiply by 2' }] },
|
|
parent_tool_use_id: null
|
|
}
|
|
}
|
|
|
|
const q = query({
|
|
prompt: createInputStream(),
|
|
options: { model: 'claude-sonnet-4-5-20250929' }
|
|
})
|
|
|
|
for await (const msg of q) {
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log(text)
|
|
}
|
|
}
|
|
```
|
|
|
|
</details>
|
|
|
|
### Session resume
|
|
|
|
If you have a session ID from a previous interaction, you can resume it later. This is useful for long-running workflows or when you need to persist conversations across application restarts.
|
|
|
|
This example creates a session, stores its ID, closes it, then resumes the conversation:
|
|
|
|
```typescript
|
|
import {
|
|
unstable_v2_createSession,
|
|
unstable_v2_resumeSession,
|
|
type SDKMessage
|
|
} from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
// Helper to extract text from assistant messages
|
|
function getAssistantText(msg: SDKMessage): string | null {
|
|
if (msg.type !== 'assistant') return null
|
|
return msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
}
|
|
|
|
// Create initial session and have a conversation
|
|
const session = unstable_v2_createSession({
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
|
|
await session.send('Remember this number: 42')
|
|
|
|
// Get the session ID from any received message
|
|
let sessionId: string | undefined
|
|
for await (const msg of session.receive()) {
|
|
sessionId = msg.session_id
|
|
const text = getAssistantText(msg)
|
|
if (text) console.log('Initial response:', text)
|
|
}
|
|
|
|
console.log('Session ID:', sessionId)
|
|
session.close()
|
|
|
|
// Later: resume the session using the stored ID
|
|
await using resumedSession = unstable_v2_resumeSession(sessionId!, {
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
|
|
await resumedSession.send('What number did I ask you to remember?')
|
|
for await (const msg of resumedSession.receive()) {
|
|
const text = getAssistantText(msg)
|
|
if (text) console.log('Resumed response:', text)
|
|
}
|
|
```
|
|
|
|
<details>
|
|
<summary>See the same operation in V1</summary>
|
|
|
|
```typescript
|
|
import { query } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
// Create initial session
|
|
const initialQuery = query({
|
|
prompt: 'Remember this number: 42',
|
|
options: { model: 'claude-sonnet-4-5-20250929' }
|
|
})
|
|
|
|
// Get session ID from any message
|
|
let sessionId: string | undefined
|
|
for await (const msg of initialQuery) {
|
|
sessionId = msg.session_id
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log('Initial response:', text)
|
|
}
|
|
}
|
|
|
|
console.log('Session ID:', sessionId)
|
|
|
|
// Later: resume the session
|
|
const resumedQuery = query({
|
|
prompt: 'What number did I ask you to remember?',
|
|
options: {
|
|
model: 'claude-sonnet-4-5-20250929',
|
|
resume: sessionId
|
|
}
|
|
})
|
|
|
|
for await (const msg of resumedQuery) {
|
|
if (msg.type === 'assistant') {
|
|
const text = msg.message.content
|
|
.filter(block => block.type === 'text')
|
|
.map(block => block.text)
|
|
.join('')
|
|
console.log('Resumed response:', text)
|
|
}
|
|
}
|
|
```
|
|
|
|
</details>
|
|
|
|
### Cleanup
|
|
|
|
Sessions can be closed manually or automatically using [`await using`](https://www.typescriptlang.org/docs/handbook/release-notes/typescript-5-2.html#using-declarations-and-explicit-resource-management), a TypeScript 5.2+ feature for automatic resource cleanup. If you're using an older TypeScript version or encounter compatibility issues, use manual cleanup instead.
|
|
|
|
**Automatic cleanup (TypeScript 5.2+):**
|
|
|
|
```typescript
|
|
import { unstable_v2_createSession } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
await using session = unstable_v2_createSession({
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
// Session closes automatically when the block exits
|
|
```
|
|
|
|
**Manual cleanup:**
|
|
|
|
```typescript
|
|
import { unstable_v2_createSession } from '@anthropic-ai/claude-agent-sdk'
|
|
|
|
const session = unstable_v2_createSession({
|
|
model: 'claude-sonnet-4-5-20250929'
|
|
})
|
|
// ... use the session ...
|
|
session.close()
|
|
```
|
|
|
|
## API reference
|
|
|
|
### `unstable_v2_createSession()`
|
|
|
|
Creates a new session for multi-turn conversations.
|
|
|
|
```typescript
|
|
function unstable_v2_createSession(options: {
|
|
model: string;
|
|
// Additional options supported
|
|
}): Session
|
|
```
|
|
|
|
### `unstable_v2_resumeSession()`
|
|
|
|
Resumes an existing session by ID.
|
|
|
|
```typescript
|
|
function unstable_v2_resumeSession(
|
|
sessionId: string,
|
|
options: {
|
|
model: string;
|
|
// Additional options supported
|
|
}
|
|
): Session
|
|
```
|
|
|
|
### `unstable_v2_prompt()`
|
|
|
|
One-shot convenience function for single-turn queries.
|
|
|
|
```typescript
|
|
function unstable_v2_prompt(
|
|
prompt: string,
|
|
options: {
|
|
model: string;
|
|
// Additional options supported
|
|
}
|
|
): Promise<Result>
|
|
```
|
|
|
|
### Session interface
|
|
|
|
```typescript
|
|
interface Session {
|
|
send(message: string): Promise<void>;
|
|
receive(): AsyncGenerator<SDKMessage>;
|
|
close(): void;
|
|
}
|
|
```
|
|
|
|
## Feature availability
|
|
|
|
Not all V1 features are available in V2 yet. The following require using the [V1 SDK](/docs/en/agent-sdk/typescript):
|
|
|
|
- Session forking (`forkSession` option)
|
|
- Some advanced streaming input patterns
|
|
|
|
## Feedback
|
|
|
|
Share your feedback on the V2 interface before it becomes stable. Report issues and suggestions through [GitHub Issues](https://github.com/anthropics/claude-code/issues).
|
|
|
|
## See also
|
|
|
|
- [TypeScript SDK reference (V1)](/docs/en/agent-sdk/typescript) - Full V1 SDK documentation
|
|
- [SDK overview](/docs/en/agent-sdk/overview) - General SDK concepts
|
|
- [V2 examples on GitHub](https://github.com/anthropics/claude-agent-sdk-demos/tree/main/hello-world-v2) - Working code examples |