skills/openrouter-agent-migration/SKILL.md
Migration guide from @openrouter/sdk to @openrouter/agent for callModel, tool(), stop conditions, and agent features. This skill should be used when code imports callModel, tool(), or stop conditions from @openrouter/sdk and needs to migrate to @openrouter/agent.
npx skillsauth add openrouterteam/skills openrouter-agent-migrationInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Agent functionality (callModel, tool(), stop conditions, format converters, streaming helpers) has moved from @openrouter/sdk to the standalone @openrouter/agent package. The @openrouter/agent package includes its own OpenRouter client class, so you do not need @openrouter/sdk for agent use cases.
Migrate if your code imports any of these from @openrouter/sdk:
callModel or uses client.callModel()tool() factory functionstepCountIs, hasToolCall, maxCost, maxTokensUsed, finishReasonIsfromClaudeMessages, toClaudeMessage, fromChatMessages, toChatMessageTool, ToolWithExecute, ToolWithGenerator, ManualTool, CallModelInput, ModelResultnpm install @openrouter/agent
If you only use agent features, you can remove @openrouter/sdk:
npm uninstall @openrouter/sdk
npm install @openrouter/agent
If you also use non-agent SDK features (models list, chat completions, credits, OAuth, API keys), keep both packages installed.
The OpenRouter client class and client.callModel() pattern work identically. Only the import source changes:
- import OpenRouter from '@openrouter/sdk';
+ import { OpenRouter } from '@openrouter/agent';
The rest of your code stays the same:
const client = new OpenRouter({ apiKey: process.env.OPENROUTER_API_KEY });
const result = client.callModel({
model: 'openai/gpt-5-nano',
input: 'Hello!',
});
const text = await result.getText();
| Old | New |
|-----|-----|
| import OpenRouter from '@openrouter/sdk' | import { OpenRouter } from '@openrouter/agent' |
| import OpenRouter, { tool, stepCountIs } from '@openrouter/sdk' | import { OpenRouter } from '@openrouter/agent'<br>import { tool } from '@openrouter/agent/tool'<br>import { stepCountIs } from '@openrouter/agent/stop-conditions' |
A standalone callModel function is also available for advanced use cases where a pre-existing OpenRouterCore instance is available:
import { callModel } from '@openrouter/agent/call-model';
// Requires an OpenRouterCore instance (from @openrouter/sdk/core)
const result = callModel(coreInstance, { model: 'openai/gpt-5-nano', input: 'Hello' });
For most use cases, prefer the client.callModel() method shown above.
| Old | New |
|-----|-----|
| import { tool } from '@openrouter/sdk' | import { tool } from '@openrouter/agent/tool' |
| Old | New |
|-----|-----|
| import { stepCountIs, hasToolCall, maxCost } from '@openrouter/sdk' | import { stepCountIs, hasToolCall, maxCost } from '@openrouter/agent/stop-conditions' |
| import { maxTokensUsed, finishReasonIs } from '@openrouter/sdk' | import { maxTokensUsed, finishReasonIs } from '@openrouter/agent/stop-conditions' |
| Old | New |
|-----|-----|
| import type { Tool, ToolWithExecute, ToolWithGenerator, ManualTool } from '@openrouter/sdk/lib/tool-types' | import type { Tool, ToolWithExecute, ToolWithGenerator, ManualTool } from '@openrouter/agent/tool-types' |
| import type { CallModelInput } from '@openrouter/sdk/lib/async-params' | import type { CallModelInput } from '@openrouter/agent/async-params' |
| import { ModelResult } from '@openrouter/sdk/lib/model-result' | import { ModelResult } from '@openrouter/agent/model-result' |
| Old | New |
|-----|-----|
| import { fromClaudeMessages, toClaudeMessage } from '@openrouter/sdk' | import { fromClaudeMessages, toClaudeMessage } from '@openrouter/agent' |
| import { fromChatMessages, toChatMessage } from '@openrouter/sdk' | import { fromChatMessages, toChatMessage } from '@openrouter/agent' |
| Old | New |
|-----|-----|
| import { hasExecuteFunction, isGeneratorTool, isRegularExecuteTool } from '@openrouter/sdk' | import { hasExecuteFunction, isGeneratorTool, isRegularExecuteTool } from '@openrouter/agent/tool-types' |
import OpenRouter, { tool, stepCountIs, hasToolCall } from '@openrouter/sdk';
import { z } from 'zod';
const client = new OpenRouter({
apiKey: process.env.OPENROUTER_API_KEY,
});
const searchTool = tool({
name: 'web_search',
description: 'Search the web',
inputSchema: z.object({ query: z.string() }),
outputSchema: z.object({ results: z.array(z.string()) }),
execute: async ({ query }) => {
return { results: ['Result 1', 'Result 2'] };
},
});
const finishTool = tool({
name: 'finish',
description: 'Complete the task',
inputSchema: z.object({ answer: z.string() }),
execute: async ({ answer }) => ({ answer }),
});
const result = client.callModel({
model: 'openai/gpt-5-nano',
instructions: 'You are a research assistant.',
input: 'What are the latest AI developments?',
tools: [searchTool, finishTool],
stopWhen: [stepCountIs(10), hasToolCall('finish')],
});
const text = await result.getText();
import { OpenRouter } from '@openrouter/agent';
import { tool } from '@openrouter/agent/tool';
import { stepCountIs, hasToolCall } from '@openrouter/agent/stop-conditions';
import { z } from 'zod';
const client = new OpenRouter({
apiKey: process.env.OPENROUTER_API_KEY,
});
const searchTool = tool({
name: 'web_search',
description: 'Search the web',
inputSchema: z.object({ query: z.string() }),
outputSchema: z.object({ results: z.array(z.string()) }),
execute: async ({ query }) => {
return { results: ['Result 1', 'Result 2'] };
},
});
const finishTool = tool({
name: 'finish',
description: 'Complete the task',
inputSchema: z.object({ answer: z.string() }),
execute: async ({ answer }) => ({ answer }),
});
const result = client.callModel({
model: 'openai/gpt-5-nano',
instructions: 'You are a research assistant.',
input: 'What are the latest AI developments?',
tools: [searchTool, finishTool],
stopWhen: [stepCountIs(10), hasToolCall('finish')],
});
const text = await result.getText();
The only changes are the three import lines at the top.
Keep @openrouter/sdk installed if you use any of these non-agent features:
| Feature | Access |
|---------|--------|
| Model listing | client.models.list() |
| Chat completions | client.chat.send() |
| Legacy completions | client.completions.generate() |
| Usage analytics | client.analytics.getUserActivity() |
| Credit balance | client.credits.getCredits() |
| API key management | client.apiKeys.list(), .create(), etc. |
| OAuth PKCE flow | client.oAuth.createAuthCode(), .exchangeAuthCodeForAPIKey() |
For mixed projects, use @openrouter/sdk for these features and @openrouter/agent for agent features:
import OpenRouter from '@openrouter/sdk'; // SDK client for models, credits, etc.
import { OpenRouter as Agent } from '@openrouter/agent'; // Agent client for callModel
import { tool } from '@openrouter/agent/tool';
import { stepCountIs } from '@openrouter/agent/stop-conditions';
// Use SDK client for non-agent features
const sdkClient = new OpenRouter({ apiKey: process.env.OPENROUTER_API_KEY });
const models = await sdkClient.models.list();
const credits = await sdkClient.credits.getCredits();
// Use Agent client for callModel
const agent = new Agent({ apiKey: process.env.OPENROUTER_API_KEY });
const result = agent.callModel({
model: 'openai/gpt-5-nano',
input: 'Hello!',
tools: [myTool],
stopWhen: stepCountIs(5),
});
These features are only available in @openrouter/agent, not in @openrouter/sdk:
Type-safe shared state across all tools in a conversation:
import { OpenRouter } from '@openrouter/agent';
import { z } from 'zod';
const client = new OpenRouter({ apiKey: '...' });
const result = client.callModel({
model: 'openai/gpt-5-nano',
input: 'Process this data',
sharedContextSchema: z.object({
userId: z.string(),
sessionData: z.record(z.unknown()),
}),
context: {
shared: { userId: '123', sessionData: {} },
},
tools: [myTool],
});
Tools can declare their own typed context and access shared context:
import { tool } from '@openrouter/agent/tool';
import { z } from 'zod';
const myTool = tool({
name: 'my_tool',
description: 'A tool with context',
inputSchema: z.object({ query: z.string() }),
contextSchema: z.object({ apiKey: z.string() }),
execute: async (params, context) => {
// context.local — this tool's own context
// context.shared — shared context across all tools
// context.setContext({ ... }) — update this tool's context
// context.setSharedContext({ ... }) — update shared context
return { result: 'done' };
},
});
Require user approval before tool execution:
const dangerousTool = tool({
name: 'delete_file',
description: 'Delete a file',
inputSchema: z.object({ path: z.string() }),
requireApproval: true, // or a function: (toolCall, context) => boolean
execute: async ({ path }) => { /* ... */ },
});
const result = client.callModel({
model: 'openai/gpt-5-nano',
input: 'Complex task',
tools: [myTool],
onTurnStart: async (context) => {
console.log(`Starting turn ${context.numberOfTurns}`);
},
onTurnEnd: async (context, response) => {
console.log(`Turn ${context.numberOfTurns} complete`);
},
});
@openrouter/agent provides granular subpath imports:
| Subpath | Exports |
|---------|---------|
| @openrouter/agent | Barrel: all exports below |
| @openrouter/agent/client | OpenRouter class |
| @openrouter/agent/call-model | callModel standalone function |
| @openrouter/agent/tool | tool() factory function |
| @openrouter/agent/tool-types | Tool, ToolWithExecute, ToolWithGenerator, ManualTool, type guards |
| @openrouter/agent/stop-conditions | stepCountIs, hasToolCall, maxCost, maxTokensUsed, finishReasonIs |
| @openrouter/agent/model-result | ModelResult response wrapper |
| @openrouter/agent/async-params | CallModelInput, hasAsyncFunctions, resolveAsyncFunctions |
| @openrouter/agent/anthropic-compat | fromClaudeMessages, toClaudeMessage |
| @openrouter/agent/chat-compat | fromChatMessages, toChatMessage |
| @openrouter/agent/conversation-state | createInitialState, updateState, appendToMessages |
| @openrouter/agent/next-turn-params | nextTurnParams utilities |
| @openrouter/agent/stream-transformers | extractUnsupportedContent, getUnsupportedContentSummary |
| @openrouter/agent/tool-context | buildToolExecuteContext, ToolContextStore |
| @openrouter/agent/tool-event-broadcaster | ToolEventBroadcaster |
| @openrouter/agent/turn-context | buildTurnContext |
development
Transcribe speech to text using OpenRouter's speech-to-text API. Use when the user asks to transcribe audio, convert speech to text, extract a transcript from a recording or meeting, caption a video's audio, or mentions STT, speech-to-text, ASR, or transcription.
tools
Generate videos from text prompts (and optional reference or frame images) using OpenRouter's asynchronous video generation API. Use when the user asks to create, generate, or make a video or animation from a description, animate an existing image, or turn a prompt into a short video clip.
tools
Generate speech audio from text using OpenRouter's text-to-speech API. Use when the user asks to synthesize speech, narrate text, create a voiceover, generate an audiobook clip, read text aloud, convert text to an audio file, or mentions TTS, text-to-speech, or voice synthesis.
tools
Scaffolds a headless agent in TypeScript using @openrouter/agent and Bun — for CLI tools, API servers, queue workers, and pipelines. No terminal UI. Use when building a headless agent, programmatic agent, CLI tool that uses AI, batch agent, pipeline agent, API agent, agent without a UI, or agent service.