diff --git a/js/plugins/anthropic/.gitignore b/js/plugins/anthropic/.gitignore
new file mode 100644
index 0000000000..d83aca04ae
--- /dev/null
+++ b/js/plugins/anthropic/.gitignore
@@ -0,0 +1,3 @@
+lib/
+node_modules/
+coverage/
diff --git a/js/plugins/anthropic/.npmignore b/js/plugins/anthropic/.npmignore
new file mode 100644
index 0000000000..d265d4ab73
--- /dev/null
+++ b/js/plugins/anthropic/.npmignore
@@ -0,0 +1,17 @@
+# typescript source files
+src/
+tests/
+tsconfig.json
+tsup.common.ts
+tsup.config.ts
+
+# GitHub files
+.github/
+.gitignore
+.npmignore
+CODE_OF_CONDUCT.md
+CONTRIBUTING.md
+
+# Developer related files
+.devcontainer/
+.vscode/
diff --git a/js/plugins/anthropic/NOTICE b/js/plugins/anthropic/NOTICE
new file mode 100644
index 0000000000..dc335bb090
--- /dev/null
+++ b/js/plugins/anthropic/NOTICE
@@ -0,0 +1,8 @@
+This project includes code derived from the Firebase Genkit Anthropic community plugin
+(https://github.com/BloomLabsInc/genkit-plugins/tree/main/plugins/anthropic).
+
+Copyright 2024 Bloom Labs Inc.
+Copyright 2025 Google LLC.
+
+Licensed under the Apache License, Version 2.0.
+See the LICENSE file distributed with this project for the full license text.
diff --git a/js/plugins/anthropic/README.md b/js/plugins/anthropic/README.md
new file mode 100644
index 0000000000..ec9c9115c0
--- /dev/null
+++ b/js/plugins/anthropic/README.md
@@ -0,0 +1,195 @@
+# Firebase Genkit + Anthropic AI
+
+
Firebase Genkit <> Anthropic AI Plugin
+
+Anthropic AI plugin for Google Firebase Genkit
+
+`@genkit-ai/anthropic` is the official Anthropic plugin for [Firebase Genkit](https://github.com/firebase/genkit). It supersedes the earlier community package `genkitx-anthropic` and is now maintained by Google.
+
+## Supported models
+
+The plugin supports the most recent Anthropic models: **Claude Sonnet 4.5**, **Claude Opus 4.1**, **Claude Haiku 4.5**, **Claude Sonnet 4**, **Claude Opus 4**, **Claude 3.5 Haiku**, and **Claude 3 Haiku**.
+
+## Installation
+
+Install the plugin in your project with your favorite package manager:
+
+- `npm install @genkit-ai/anthropic`
+- `yarn add @genkit-ai/anthropic`
+- `pnpm add @genkit-ai/anthropic`
+
+## Usage
+
+### Initialize
+
+```typescript
+import { genkit } from 'genkit';
+import { anthropic } from '@genkit-ai/anthropic';
+
+const ai = genkit({
+ plugins: [anthropic({ apiKey: process.env.ANTHROPIC_API_KEY })],
+ // specify a default model for generate here if you wish:
+ model: anthropic.model('claude-sonnet-4-5'),
+});
+```
+
+### Basic examples
+
+The simplest way to generate text is by using the `generate` method:
+
+```typescript
+const response = await ai.generate({
+ model: anthropic.model('claude-3-haiku'),
+ prompt: 'Tell me a joke.',
+});
+
+console.log(response.text);
+```
+
+### Multi-modal prompt
+
+```typescript
+// ...initialize Genkit instance (as shown above)...
+
+const response = await ai.generate({
+ prompt: [
+ { text: 'What animal is in the photo?' },
+ { media: { url: imageUrl } },
+ ],
+ config: {
+ // control of the level of visual detail when processing image embeddings
+ // Low detail level also decreases the token usage
+ visualDetailLevel: 'low',
+ },
+});
+console.log(response.text);
+```
+
+### Extended thinking
+
+Claude 4 models can expose their internal reasoning. Enable it per-request with the Anthropic thinking config and read the reasoning from the response:
+
+```typescript
+const response = await ai.generate({
+ prompt: 'Walk me through your reasoning for Fermat’s little theorem.',
+ config: {
+ thinking: {
+ enabled: true,
+ budgetTokens: 4096, // Must be >= 1024 and less than max_tokens
+ },
+ },
+});
+
+console.log(response.text); // Final assistant answer
+console.log(response.reasoning); // Summarized thinking steps
+```
+
+When thinking is enabled, request bodies sent through the plugin include the `thinking` payload (`{ type: 'enabled', budget_tokens: … }`) that Anthropic's API expects, and streamed responses deliver `reasoning` parts as they arrive so you can render the chain-of-thought incrementally.
+
+### Beta API Limitations
+
+The beta API surface provides access to experimental features, but some server-managed tool blocks are not yet supported by this plugin. The following beta API features will cause an error if encountered:
+
+- `web_fetch_tool_result`
+- `code_execution_tool_result`
+- `bash_code_execution_tool_result`
+- `text_editor_code_execution_tool_result`
+- `mcp_tool_result`
+- `mcp_tool_use`
+- `container_upload`
+
+Note that `server_tool_use` and `web_search_tool_result` ARE supported and work with both stable and beta APIs.
+
+### Within a flow
+
+```typescript
+import { z } from 'genkit';
+
+// ...initialize Genkit instance (as shown above)...
+
+export const jokeFlow = ai.defineFlow(
+ {
+ name: 'jokeFlow',
+ inputSchema: z.string(),
+ outputSchema: z.string(),
+ },
+ async (subject) => {
+ const llmResponse = await ai.generate({
+ prompt: `tell me a joke about ${subject}`,
+ });
+ return llmResponse.text;
+ }
+);
+```
+
+### Direct model usage (without Genkit instance)
+
+The plugin supports Genkit Plugin API v2, which allows you to use models directly without initializing the full Genkit framework:
+
+```typescript
+import { anthropic } from '@genkit-ai/anthropic';
+
+// Create a model reference directly
+const claude = anthropic.model('claude-sonnet-4-5');
+
+// Use the model directly
+const response = await claude({
+ messages: [
+ {
+ role: 'user',
+ content: [{ text: 'Tell me a joke.' }],
+ },
+ ],
+});
+
+console.log(response);
+```
+
+You can also create model references using the plugin's `model()` method:
+
+```typescript
+import { anthropic } from '@genkit-ai/anthropic';
+
+// Create model references
+const claudeSonnet45 = anthropic.model('claude-sonnet-4-5');
+const claudeOpus41 = anthropic.model('claude-opus-4-1');
+const claude35Haiku = anthropic.model('claude-3-5-haiku');
+
+// Use the model reference directly
+const response = await claudeSonnet45({
+ messages: [
+ {
+ role: 'user',
+ content: [{ text: 'Hello!' }],
+ },
+ ],
+});
+```
+
+This approach is useful for:
+
+- Framework developers who need raw model access
+- Testing models in isolation
+- Using Genkit models in non-Genkit applications
+
+## Acknowledgements
+
+This plugin builds on the community work published as [`genkitx-anthropic`](https://github.com/BloomLabsInc/genkit-plugins/blob/main/plugins/anthropic/README.md) by Bloom Labs Inc. Their Apache 2.0–licensed implementation provided the foundation for this maintained package.
+
+## Contributing
+
+Want to contribute to the project? That's awesome! Head over to our [Contribution Guidelines](CONTRIBUTING.md).
+
+## Need support?
+
+> [!NOTE]
+> This repository depends on Google's Firebase Genkit. For issues and questions related to Genkit, please refer to instructions available in [Genkit's repository](https://github.com/firebase/genkit).
+
+
+## Credits
+
+This plugin is maintained by Google with acknowledgement to the community contributions from [Bloom Labs Inc](https://github.com/BloomLabsInc).
+
+## License
+
+This project is licensed under the [Apache 2.0 License](https://github.com/BloomLabsInc/genkit-plugins/blob/main/LICENSE).
diff --git a/js/plugins/anthropic/package.json b/js/plugins/anthropic/package.json
new file mode 100644
index 0000000000..32b1c4ba87
--- /dev/null
+++ b/js/plugins/anthropic/package.json
@@ -0,0 +1,70 @@
+{
+ "name": "@genkit-ai/anthropic",
+ "description": "Genkit AI framework plugin for Anthropic APIs.",
+ "keywords": [
+ "genkit",
+ "genkit-plugin",
+ "genkit-model",
+ "anthropic",
+ "anthropic-ai",
+ "claude-4",
+ "haiku-4",
+ "opus",
+ "haiku",
+ "sonnet",
+ "ai",
+ "genai",
+ "generative-ai"
+ ],
+ "version": "1.23.0",
+ "type": "commonjs",
+ "repository": {
+ "type": "git",
+ "url": "https://github.com/firebase/genkit.git",
+ "directory": "js/plugins/anthropic"
+ },
+ "author": "genkit",
+ "license": "Apache-2.0",
+ "peerDependencies": {
+ "genkit": "workspace:^"
+ },
+ "dependencies": {
+ "@anthropic-ai/sdk": "^0.68.0"
+ },
+ "devDependencies": {
+ "@types/node": "^20.11.16",
+ "check-node-version": "^4.2.1",
+ "genkit": "workspace:*",
+ "npm-run-all": "^4.1.5",
+ "rimraf": "^6.0.1",
+ "tsup": "^8.3.5",
+ "tsx": "^4.19.2",
+ "typescript": "^4.9.0"
+ },
+ "types": "./lib/index.d.ts",
+ "exports": {
+ ".": {
+ "types": "./lib/index.d.ts",
+ "require": "./lib/index.js",
+ "import": "./lib/index.mjs",
+ "default": "./lib/index.js"
+ }
+ },
+ "files": [
+ "lib"
+ ],
+ "publishConfig": {
+ "provenance": true,
+ "access": "public"
+ },
+ "scripts": {
+ "check": "tsc",
+ "compile": "tsup-node",
+ "build:clean": "rimraf ./lib",
+ "build": "npm-run-all build:clean check compile",
+ "build:watch": "tsup-node --watch",
+ "test": "tsx --test tests/*_test.ts",
+ "test:file": "tsx --test",
+ "test:coverage": "check-node-version --node '>=22' && tsx --test --experimental-test-coverage --test-coverage-include='src/**/*.ts' ./tests/**/*_test.ts"
+ }
+}
diff --git a/js/plugins/anthropic/src/index.ts b/js/plugins/anthropic/src/index.ts
new file mode 100644
index 0000000000..d5a0fef9cb
--- /dev/null
+++ b/js/plugins/anthropic/src/index.ts
@@ -0,0 +1,155 @@
+/**
+ * Copyright 2024 Bloom Labs Inc
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import Anthropic from '@anthropic-ai/sdk';
+import { genkitPluginV2, type GenkitPluginV2 } from 'genkit/plugin';
+
+import { ActionMetadata, ModelReference, z } from 'genkit';
+import { ModelAction } from 'genkit/model';
+import { ActionType } from 'genkit/registry';
+import { listActions } from './list.js';
+import {
+ AnthropicConfigSchemaType,
+ ClaudeConfig,
+ ClaudeModelName,
+ KNOWN_CLAUDE_MODELS,
+ KnownClaudeModels,
+ claudeModel,
+ claudeModelReference,
+} from './models.js';
+import { InternalPluginOptions, PluginOptions, __testClient } from './types.js';
+
+const PROMPT_CACHING_BETA_HEADER_VALUE = 'prompt-caching-2024-07-31';
+
+/**
+ * Gets or creates an Anthropic client instance.
+ * Supports test client injection for internal testing.
+ */
+function getAnthropicClient(options?: PluginOptions): Anthropic {
+ // Check for test client injection first (internal use only)
+ const internalOptions = options as InternalPluginOptions | undefined;
+ if (internalOptions?.[__testClient]) {
+ return internalOptions[__testClient];
+ }
+
+ // Production path: create real client
+ const apiKey = options?.apiKey || process.env.ANTHROPIC_API_KEY;
+ if (!apiKey) {
+ throw new Error(
+ 'Please pass in the API key or set the ANTHROPIC_API_KEY environment variable'
+ );
+ }
+ const defaultHeaders: Record = {};
+ if (options?.cacheSystemPrompt) {
+ defaultHeaders['anthropic-beta'] = PROMPT_CACHING_BETA_HEADER_VALUE;
+ }
+ return new Anthropic({ apiKey, defaultHeaders });
+}
+
+/**
+ * This module provides an interface to the Anthropic AI models through the Genkit plugin system.
+ * It allows users to interact with various Claude models by providing an API key and optional configuration.
+ *
+ * The main export is the `anthropic` plugin, which can be configured with an API key either directly or through
+ * environment variables. It initializes the Anthropic client and makes available the Claude models for use.
+ *
+ * Exports:
+ * - anthropic: The main plugin function to interact with the Anthropic AI.
+ *
+ * Usage:
+ * To use the Claude models, initialize the anthropic plugin inside `genkit()` and pass the configuration options. If no API key is provided in the options, the environment variable `ANTHROPIC_API_KEY` must be set. If you want to cache the system prompt, set `cacheSystemPrompt` to `true`. **Note:** Prompt caching is in beta and may change. To learn more, see https://docs.anthropic.com/en/docs/prompt-caching.
+ *
+ * Example:
+ * ```
+ * import { anthropic } from '@genkit-ai/anthropic';
+ * import { genkit } from 'genkit';
+ *
+ * const ai = genkit({
+ * plugins: [
+ * anthropic({ apiKey: 'your-api-key', cacheSystemPrompt: false })
+ * ... // other plugins
+ * ]
+ * });
+ *
+ * // Access models via the plugin's model() method:
+ * const model = anthropic.model('claude-sonnet-4');
+ * ```
+ */
+function anthropicPlugin(options?: PluginOptions): GenkitPluginV2 {
+ const client = getAnthropicClient(options);
+ const defaultApiVersion = options?.apiVersion;
+
+ let listActionsCache: ActionMetadata[] | null = null;
+
+ return genkitPluginV2({
+ name: 'anthropic',
+ init: async () => {
+ const actions: ModelAction[] = [];
+ for (const name of Object.keys(KNOWN_CLAUDE_MODELS)) {
+ const action = claudeModel({
+ name,
+ client,
+ cacheSystemPrompt: options?.cacheSystemPrompt,
+ defaultApiVersion,
+ });
+ actions.push(action);
+ }
+ return actions;
+ },
+ resolve: (actionType: ActionType, name: string) => {
+ if (actionType === 'model') {
+ // Strip the 'anthropic/' namespace prefix if present
+ const modelName = name.startsWith('anthropic/') ? name.slice(10) : name;
+ return claudeModel({
+ name: modelName,
+ client,
+ cacheSystemPrompt: options?.cacheSystemPrompt,
+ defaultApiVersion,
+ });
+ }
+ return undefined;
+ },
+ list: async () => {
+ if (listActionsCache) return listActionsCache;
+ listActionsCache = await listActions(client);
+ return listActionsCache;
+ },
+ });
+}
+
+export type AnthropicPlugin = {
+ (pluginOptions?: PluginOptions): GenkitPluginV2;
+ model(
+ name: KnownClaudeModels | (ClaudeModelName & {}),
+ config?: ClaudeConfig
+ ): ModelReference;
+ model(name: string, config?: any): ModelReference;
+};
+
+/**
+ * Anthropic AI plugin for Genkit.
+ * Includes Claude models (3, 3.5, and 4 series).
+ */
+export const anthropic = anthropicPlugin as AnthropicPlugin;
+(anthropic as any).model = (
+ name: string,
+ config?: any
+): ModelReference => {
+ return claudeModelReference(name, config);
+};
+
+export default anthropic;
diff --git a/js/plugins/anthropic/src/list.ts b/js/plugins/anthropic/src/list.ts
new file mode 100644
index 0000000000..6124fd4392
--- /dev/null
+++ b/js/plugins/anthropic/src/list.ts
@@ -0,0 +1,66 @@
+/**
+ * Copyright 2024 Bloom Labs Inc
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import Anthropic from '@anthropic-ai/sdk';
+import { modelActionMetadata } from 'genkit/plugin';
+
+import { ActionMetadata } from 'genkit';
+import { claudeModelReference } from './models.js';
+
+/**
+ * Retrieves available Anthropic models from the API and converts them into Genkit action metadata.
+ *
+ * This function queries the Anthropic API for the list of available models and generates metadata
+ * for all discovered models.
+ *
+ * @param client - The Anthropic API client instance
+ * @returns A promise that resolves to an array of action metadata for all discovered models
+ */
+export async function listActions(
+ client: Anthropic
+): Promise {
+ const clientModels = (await client.models.list()).data;
+ const seenNames = new Set();
+
+ return clientModels
+ .filter((modelInfo) => {
+ const modelId = modelInfo.id;
+ if (!modelId) {
+ return false;
+ }
+
+ const ref = claudeModelReference(modelId);
+ const name = ref.name;
+
+ // Deduplicate by name
+ if (seenNames.has(name)) {
+ return false;
+ }
+ seenNames.add(name);
+ return true;
+ })
+ .map((modelInfo) => {
+ const modelId = modelInfo.id!;
+ const ref = claudeModelReference(modelId);
+
+ return modelActionMetadata({
+ name: ref.name,
+ info: ref.info,
+ configSchema: ref.configSchema,
+ });
+ });
+}
diff --git a/js/plugins/anthropic/src/models.ts b/js/plugins/anthropic/src/models.ts
new file mode 100644
index 0000000000..2ee33d933c
--- /dev/null
+++ b/js/plugins/anthropic/src/models.ts
@@ -0,0 +1,241 @@
+/**
+ * Copyright 2024 Bloom Labs Inc
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import type {
+ GenerateRequest,
+ GenerateResponseData,
+ ModelReference,
+ StreamingCallback,
+} from 'genkit';
+import { z } from 'genkit';
+import type { GenerateResponseChunkData, ModelAction } from 'genkit/model';
+import { modelRef } from 'genkit/model';
+import { model } from 'genkit/plugin';
+
+import type { ModelInfo } from 'genkit/model';
+import { BetaRunner, Runner } from './runner/index.js';
+import {
+ AnthropicBaseConfigSchema,
+ AnthropicBaseConfigSchemaType,
+ AnthropicConfigSchema,
+ AnthropicThinkingConfigSchema,
+ resolveBetaEnabled,
+ type ClaudeModelParams,
+ type ClaudeRunnerParams,
+} from './types.js';
+
+// This contains all the Anthropic config schema types
+type ConfigSchemaType =
+ | AnthropicBaseConfigSchemaType
+ | AnthropicThinkingConfigSchemaType;
+
+/**
+ * Creates a model reference for a Claude model.
+ */
+function commonRef(
+ name: string,
+ configSchema: ConfigSchemaType = AnthropicConfigSchema,
+ info?: ModelInfo
+): ModelReference {
+ return modelRef({
+ name: `anthropic/${name}`,
+ configSchema,
+ info: info ?? {
+ supports: {
+ multiturn: true,
+ tools: true,
+ media: true,
+ systemRole: true,
+ output: ['text'],
+ },
+ },
+ });
+}
+
+export const KNOWN_CLAUDE_MODELS: Record<
+ string,
+ ModelReference<
+ AnthropicBaseConfigSchemaType | AnthropicThinkingConfigSchemaType
+ >
+> = {
+ 'claude-3-haiku': commonRef('claude-3-haiku', AnthropicBaseConfigSchema),
+ 'claude-3-5-haiku': commonRef('claude-3-5-haiku', AnthropicBaseConfigSchema),
+ 'claude-sonnet-4': commonRef(
+ 'claude-sonnet-4',
+ AnthropicThinkingConfigSchema
+ ),
+ 'claude-opus-4': commonRef('claude-opus-4', AnthropicThinkingConfigSchema),
+ 'claude-sonnet-4-5': commonRef(
+ 'claude-sonnet-4-5',
+ AnthropicThinkingConfigSchema
+ ),
+ 'claude-haiku-4-5': commonRef(
+ 'claude-haiku-4-5',
+ AnthropicThinkingConfigSchema
+ ),
+ 'claude-opus-4-1': commonRef(
+ 'claude-opus-4-1',
+ AnthropicThinkingConfigSchema
+ ),
+};
+
+/**
+ * Gets the un-prefixed model name from a modelReference.
+ */
+export function extractVersion(
+ model: ModelReference | undefined,
+ modelName: string
+): string {
+ // Extract from model name (remove 'anthropic/' prefix if present)
+ return modelName.replace(/^anthropic\//, '');
+}
+
+/**
+ * Generic Claude model info for unknown/unsupported models.
+ * Used when a model name is not in KNOWN_CLAUDE_MODELS.
+ */
+export const GENERIC_CLAUDE_MODEL_INFO = {
+ supports: {
+ multiturn: true,
+ tools: true,
+ media: true,
+ systemRole: true,
+ output: ['text'],
+ },
+};
+
+export type KnownClaudeModels = keyof typeof KNOWN_CLAUDE_MODELS;
+export type ClaudeModelName = string;
+export type AnthropicConfigSchemaType = typeof AnthropicConfigSchema;
+export type AnthropicThinkingConfigSchemaType =
+ typeof AnthropicThinkingConfigSchema;
+export type ClaudeConfig = z.infer;
+
+/**
+ * Creates the runner used by Genkit to interact with the Claude model.
+ * @param params Configuration for the Claude runner.
+ * @param configSchema The config schema for this model (used for type inference).
+ * @returns The runner that Genkit will call when the model is invoked.
+ */
+export function claudeRunner(
+ params: ClaudeRunnerParams,
+ configSchema: TConfigSchema
+) {
+ const { defaultApiVersion, ...runnerParams } = params;
+
+ if (!runnerParams.client) {
+ throw new Error('Anthropic client is required to create a runner');
+ }
+
+ let stableRunner: Runner | null = null;
+ let betaRunner: BetaRunner | null = null;
+
+ return async (
+ request: GenerateRequest,
+ {
+ streamingRequested,
+ sendChunk,
+ abortSignal,
+ }: {
+ streamingRequested: boolean;
+ sendChunk: StreamingCallback;
+ abortSignal: AbortSignal;
+ }
+ ): Promise => {
+ // Cast to AnthropicConfigSchema for internal runner which expects the full schema
+ const normalizedRequest = request as unknown as GenerateRequest<
+ typeof AnthropicConfigSchema
+ >;
+ const isBeta = resolveBetaEnabled(
+ normalizedRequest.config,
+ defaultApiVersion
+ );
+ const runner = isBeta
+ ? (betaRunner ??= new BetaRunner(runnerParams))
+ : (stableRunner ??= new Runner(runnerParams));
+ return runner.run(normalizedRequest, {
+ streamingRequested,
+ sendChunk,
+ abortSignal,
+ });
+ };
+}
+
+/**
+ * Strips the 'anthropic/' namespace prefix if present.
+ */
+function checkModelName(name: string): string {
+ return name.startsWith('anthropic/') ? name.slice(10) : name;
+}
+
+/**
+ * Creates a model reference for a Claude model.
+ * This allows referencing models without initializing the plugin.
+ */
+export function claudeModelReference(
+ name: string,
+ config: z.infer = {}
+): ModelReference {
+ const modelName = checkModelName(name);
+ return modelRef({
+ name: `anthropic/${modelName}`,
+ config: config,
+ configSchema: AnthropicConfigSchema,
+ info: {
+ ...GENERIC_CLAUDE_MODEL_INFO,
+ },
+ });
+}
+
+/**
+ * Defines a Claude model with the given name and Anthropic client.
+ * Accepts any model name and lets the API validate it. If the model is in KNOWN_CLAUDE_MODELS, uses that modelRef
+ * for better defaults; otherwise creates a generic model reference.
+ */
+export function claudeModel(
+ params: ClaudeModelParams
+): ModelAction {
+ const {
+ name,
+ client: runnerClient,
+ cacheSystemPrompt: cachePrompt,
+ defaultApiVersion: apiVersion,
+ } = params;
+ // Use supported model ref if available, otherwise create generic model ref
+ const modelRef = KNOWN_CLAUDE_MODELS[name];
+ const modelInfo = modelRef ? modelRef.info : GENERIC_CLAUDE_MODEL_INFO;
+ const configSchema = modelRef?.configSchema ?? AnthropicConfigSchema;
+
+ return model<
+ AnthropicBaseConfigSchemaType | AnthropicThinkingConfigSchemaType
+ >(
+ {
+ name: `anthropic/${name}`,
+ ...modelInfo,
+ configSchema: configSchema,
+ },
+ claudeRunner(
+ {
+ name,
+ client: runnerClient,
+ cacheSystemPrompt: cachePrompt,
+ defaultApiVersion: apiVersion,
+ },
+ configSchema
+ )
+ );
+}
diff --git a/js/plugins/anthropic/src/runner/base.ts b/js/plugins/anthropic/src/runner/base.ts
new file mode 100644
index 0000000000..e6b7132e28
--- /dev/null
+++ b/js/plugins/anthropic/src/runner/base.ts
@@ -0,0 +1,550 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { Anthropic } from '@anthropic-ai/sdk';
+import type { DocumentBlockParam } from '@anthropic-ai/sdk/resources/messages';
+import type {
+ GenerateRequest,
+ GenerateResponseChunkData,
+ GenerateResponseData,
+ MessageData,
+ Part,
+ Role,
+} from 'genkit';
+import { Message as GenkitMessage } from 'genkit';
+import type { ToolDefinition } from 'genkit/model';
+
+import {
+ AnthropicConfigSchema,
+ Media,
+ MediaSchema,
+ MediaType,
+ MediaTypeSchema,
+ type ClaudeRunnerParams,
+ type ThinkingConfig,
+} from '../types.js';
+
+import {
+ RunnerContentBlockParam,
+ RunnerMessage,
+ RunnerMessageParam,
+ RunnerRequestBody,
+ RunnerStream,
+ RunnerStreamEvent,
+ RunnerStreamingRequestBody,
+ RunnerTool,
+ RunnerToolResponseContent,
+ RunnerTypes,
+} from './types.js';
+
+const ANTHROPIC_THINKING_CUSTOM_KEY = 'anthropicThinking';
+
+/**
+ * Shared runner logic for Anthropic SDK integrations.
+ *
+ * Concrete subclasses pass in their SDK-specific type bundle via `RunnerTypes`,
+ * letting this base class handle message/tool translation once for both the
+ * stable and beta APIs that share the same conceptual surface.
+ */
+export abstract class BaseRunner {
+ protected name: string;
+ protected client: Anthropic;
+ protected cacheSystemPrompt?: boolean;
+
+ /**
+ * Default maximum output tokens for Claude models when not specified in the request.
+ */
+ protected readonly DEFAULT_MAX_OUTPUT_TOKENS = 4096;
+
+ constructor(params: ClaudeRunnerParams) {
+ this.name = params.name;
+ this.client = params.client;
+ this.cacheSystemPrompt = params.cacheSystemPrompt;
+ }
+
+ /**
+ * Converts a Genkit role to the corresponding Anthropic role.
+ */
+ protected toAnthropicRole(
+ role: Role,
+ toolMessageType?: 'tool_use' | 'tool_result'
+ ): 'user' | 'assistant' {
+ if (role === 'user') {
+ return 'user';
+ }
+ if (role === 'model') {
+ return 'assistant';
+ }
+ if (role === 'tool') {
+ return toolMessageType === 'tool_use' ? 'assistant' : 'user';
+ }
+ throw new Error(`Unsupported genkit role: ${role}`);
+ }
+
+ protected isMediaType(value: string): value is MediaType {
+ return MediaTypeSchema.safeParse(value).success;
+ }
+
+ protected isMediaObject(obj: unknown): obj is Media {
+ return MediaSchema.safeParse(obj).success;
+ }
+
+ /**
+ * Checks if a URL is a data URL (starts with 'data:').
+ */
+ protected isDataUrl(url: string): boolean {
+ return url.startsWith('data:');
+ }
+
+ protected extractDataFromBase64Url(
+ url: string
+ ): { data: string; contentType: string } | null {
+ const match = url.match(/^data:([^;]+);base64,(.+)$/);
+ return (
+ match && {
+ contentType: match[1],
+ data: match[2],
+ }
+ );
+ }
+
+ /**
+ * Both the stable and beta Anthropic SDKs accept the same JSON shape for PDF
+ * document sources (either `type: 'base64'` with a base64 payload or `type: 'url'`
+ * with a public URL). Even though the return type references the stable SDK
+ * union, TypeScript’s structural typing lets the beta runner reuse this helper.
+ */
+ protected toPdfDocumentSource(media: Media): DocumentBlockParam['source'] {
+ if (media.contentType !== 'application/pdf') {
+ throw new Error(
+ `PDF contentType mismatch: expected application/pdf, got ${media.contentType}`
+ );
+ }
+ const url = media.url;
+ if (this.isDataUrl(url)) {
+ const extracted = this.extractDataFromBase64Url(url);
+ if (!extracted) {
+ throw new Error(
+ `Invalid PDF data URL format: ${url.substring(0, 50)}...`
+ );
+ }
+ const { data, contentType } = extracted;
+ if (contentType !== 'application/pdf') {
+ throw new Error(
+ `PDF contentType mismatch: expected application/pdf, got ${contentType}`
+ );
+ }
+ return {
+ type: 'base64',
+ media_type: 'application/pdf',
+ data,
+ };
+ }
+ return {
+ type: 'url',
+ url,
+ };
+ }
+
+ /**
+ * Normalizes Genkit `Media` into either a base64 payload or a remote URL
+ * accepted by the Anthropic SDK. Anthropic supports both `data:` URLs (which
+ * we forward as base64) and remote `https` URLs without additional handling.
+ */
+ protected toImageSource(
+ media: Media
+ ):
+ | { kind: 'base64'; data: string; mediaType: MediaType }
+ | { kind: 'url'; url: string } {
+ if (this.isDataUrl(media.url)) {
+ const extracted = this.extractDataFromBase64Url(media.url);
+ const { data, contentType } = extracted ?? {};
+ if (!data || !contentType) {
+ throw new Error(
+ `Invalid genkit part media provided to toAnthropicMessageContent: ${JSON.stringify(
+ media
+ )}.`
+ );
+ }
+
+ const resolvedMediaType = contentType;
+ if (!resolvedMediaType) {
+ throw new Error('Media type is required but was not provided');
+ }
+ if (!this.isMediaType(resolvedMediaType)) {
+ // Provide helpful error message for text files
+ if (resolvedMediaType === 'text/plain') {
+ throw new Error(
+ `Unsupported media type: ${resolvedMediaType}. Text files should be sent as text content in the message, not as media. For example, use { text: '...' } instead of { media: { url: '...', contentType: 'text/plain' } }`
+ );
+ }
+ throw new Error(`Unsupported media type: ${resolvedMediaType}`);
+ }
+ return {
+ kind: 'base64',
+ data,
+ mediaType: resolvedMediaType,
+ };
+ }
+
+ if (!media.url) {
+ throw new Error('Media url is required but was not provided');
+ }
+
+ // For non-data URLs, use the provided contentType or default to a generic type
+ // Note: Anthropic will validate the actual content when fetching from URL
+ if (media.contentType) {
+ if (!this.isMediaType(media.contentType)) {
+ // Provide helpful error message for text files
+ if (media.contentType === 'text/plain') {
+ throw new Error(
+ `Unsupported media type: ${media.contentType}. Text files should be sent as text content in the message, not as media. For example, use { text: '...' } instead of { media: { url: '...', contentType: 'text/plain' } }`
+ );
+ }
+ throw new Error(`Unsupported media type: ${media.contentType}`);
+ }
+ }
+
+ return {
+ kind: 'url',
+ url: media.url,
+ };
+ }
+
+ /**
+ * Converts tool response output to the appropriate Anthropic content format.
+ * Handles Media objects, data URLs, strings, and other outputs.
+ */
+ protected toAnthropicToolResponseContent(
+ part: Part
+ ): RunnerToolResponseContent {
+ const output = part.toolResponse?.output ?? {};
+
+ // Handle Media objects (images returned by tools)
+ if (this.isMediaObject(output)) {
+ const { data, contentType } =
+ this.extractDataFromBase64Url(output.url) ?? {};
+ if (data && contentType) {
+ if (!this.isMediaType(contentType)) {
+ // Provide helpful error message for text files
+ if (contentType === 'text/plain') {
+ throw new Error(
+ `Unsupported media type: ${contentType}. Text files should be sent as text content, not as media.`
+ );
+ }
+ throw new Error(`Unsupported media type: ${contentType}`);
+ }
+ return {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data,
+ media_type: contentType,
+ },
+ };
+ }
+ }
+
+ // Handle string outputs - check if it's a data URL
+ if (typeof output === 'string') {
+ // Check if string is a data URL (e.g., "data:image/gif;base64,...")
+ if (this.isDataUrl(output)) {
+ const { data, contentType } =
+ this.extractDataFromBase64Url(output) ?? {};
+ if (data && contentType) {
+ if (!this.isMediaType(contentType)) {
+ // Provide helpful error message for text files
+ if (contentType === 'text/plain') {
+ throw new Error(
+ `Unsupported media type: ${contentType}. Text files should be sent as text content, not as media.`
+ );
+ }
+ throw new Error(`Unsupported media type: ${contentType}`);
+ }
+ return {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data,
+ media_type: contentType,
+ },
+ };
+ }
+ }
+ // Regular string output
+ return {
+ type: 'text',
+ text: output,
+ };
+ }
+
+ // Handle other outputs by stringifying
+ return {
+ type: 'text',
+ text: JSON.stringify(output),
+ };
+ }
+
+ protected createThinkingPart(thinking: string, signature?: string): Part {
+ const custom =
+ signature !== undefined
+ ? {
+ [ANTHROPIC_THINKING_CUSTOM_KEY]: { signature },
+ }
+ : undefined;
+ return custom
+ ? {
+ reasoning: thinking,
+ custom,
+ }
+ : {
+ reasoning: thinking,
+ };
+ }
+
+ protected getThinkingSignature(part: Part): string | undefined {
+ const custom = part.custom as Record | undefined;
+ const thinkingValue = custom?.[ANTHROPIC_THINKING_CUSTOM_KEY];
+ if (
+ typeof thinkingValue === 'object' &&
+ thinkingValue !== null &&
+ 'signature' in thinkingValue &&
+ typeof (thinkingValue as { signature: unknown }).signature === 'string'
+ ) {
+ return (thinkingValue as { signature: string }).signature;
+ }
+ return undefined;
+ }
+
+ protected getRedactedThinkingData(part: Part): string | undefined {
+ const custom = part.custom as Record | undefined;
+ const redacted = custom?.redactedThinking;
+ return typeof redacted === 'string' ? redacted : undefined;
+ }
+
+ protected toAnthropicThinkingConfig(
+ config: ThinkingConfig | undefined
+ ):
+ | { type: 'enabled'; budget_tokens: number }
+ | { type: 'disabled' }
+ | undefined {
+ if (!config) return undefined;
+
+ const { enabled, budgetTokens } = config;
+
+ if (enabled === true) {
+ if (budgetTokens === undefined) {
+ return undefined;
+ }
+ return { type: 'enabled', budget_tokens: budgetTokens };
+ }
+
+ if (enabled === false) {
+ return { type: 'disabled' };
+ }
+
+ if (budgetTokens !== undefined) {
+ return { type: 'enabled', budget_tokens: budgetTokens };
+ }
+
+ return undefined;
+ }
+
+ protected toWebSearchToolResultPart(params: {
+ toolUseId: string;
+ content: unknown;
+ type: string;
+ }): Part {
+ const { toolUseId, content, type } = params;
+ return {
+ text: `[Anthropic server tool result ${toolUseId}] ${JSON.stringify(content)}`,
+ custom: {
+ anthropicServerToolResult: {
+ type,
+ toolUseId,
+ content,
+ },
+ },
+ };
+ }
+
+ /**
+ * Converts a Genkit Part to the corresponding Anthropic content block.
+ * Each runner implements this to return its specific API type.
+ */
+ protected abstract toAnthropicMessageContent(
+ part: Part
+ ): RunnerContentBlockParam;
+
+ /**
+ * Converts Genkit messages to Anthropic format.
+ * Extracts system message and converts remaining messages using the runner's
+ * toAnthropicMessageContent implementation.
+ */
+ protected toAnthropicMessages(messages: MessageData[]): {
+ system?: string;
+ messages: RunnerMessageParam[];
+ } {
+ let system: string | undefined;
+
+ if (messages[0]?.role === 'system') {
+ const systemMessage = messages[0];
+ const textParts: string[] = [];
+
+ for (const part of systemMessage.content ?? []) {
+ if (part.text) {
+ textParts.push(part.text);
+ } else if (part.media || part.toolRequest || part.toolResponse) {
+ throw new Error(
+ 'System messages can only contain text content. Media, tool requests, and tool responses are not supported in system messages.'
+ );
+ }
+ }
+
+ // Concatenate multiple text parts into a single string.
+ // Note: The Anthropic SDK supports system as string | Array,
+ // so we could alternatively preserve the multi-part structure as:
+ // system = textParts.map(text => ({ type: 'text', text }))
+ // However, concatenation is simpler and maintains semantic equivalence while
+ // keeping the cache control logic straightforward in the concrete runners.
+ system = textParts.length > 0 ? textParts.join('\n\n') : undefined;
+ }
+
+ const messagesToIterate =
+ system !== undefined ? messages.slice(1) : messages;
+ const anthropicMsgs: RunnerMessageParam[] = [];
+
+ for (const message of messagesToIterate) {
+ const msg = new GenkitMessage(message);
+
+ // Detect tool message kind from Genkit Parts (no SDK typing needed)
+ const hadToolUse = msg.content.some((p) => !!p.toolRequest);
+ const hadToolResult = msg.content.some((p) => !!p.toolResponse);
+
+ const toolMessageType = hadToolUse
+ ? ('tool_use' as const)
+ : hadToolResult
+ ? ('tool_result' as const)
+ : undefined;
+
+ const role = this.toAnthropicRole(message.role, toolMessageType);
+
+ const content = msg.content.map((part) =>
+ this.toAnthropicMessageContent(part)
+ );
+
+ anthropicMsgs.push({ role, content });
+ }
+
+ return { system, messages: anthropicMsgs };
+ }
+
+ /**
+ * Converts a Genkit ToolDefinition to an Anthropic Tool object.
+ */
+ protected toAnthropicTool(tool: ToolDefinition): RunnerTool {
+ return {
+ name: tool.name,
+ description: tool.description,
+ input_schema: tool.inputSchema,
+ } as RunnerTool;
+ }
+
+ /**
+ * Converts an Anthropic request to a non-streaming Anthropic API request body.
+ * @param modelName The name of the Anthropic model to use.
+ * @param request The Genkit GenerateRequest to convert.
+ * @param cacheSystemPrompt Whether to cache the system prompt.
+ * @returns The converted Anthropic API non-streaming request body.
+ * @throws An error if an unsupported output format is requested.
+ */
+ protected abstract toAnthropicRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): RunnerRequestBody;
+
+ /**
+ * Converts an Anthropic request to a streaming Anthropic API request body.
+ * @param modelName The name of the Anthropic model to use.
+ * @param request The Genkit GenerateRequest to convert.
+ * @param cacheSystemPrompt Whether to cache the system prompt.
+ * @returns The converted Anthropic API streaming request body.
+ * @throws An error if an unsupported output format is requested.
+ */
+ protected abstract toAnthropicStreamingRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): RunnerStreamingRequestBody;
+
+ protected abstract createMessage(
+ body: RunnerRequestBody,
+ abortSignal: AbortSignal
+ ): Promise>;
+
+ protected abstract streamMessages(
+ body: RunnerStreamingRequestBody,
+ abortSignal: AbortSignal
+ ): RunnerStream;
+
+ protected abstract toGenkitResponse(
+ message: RunnerMessage
+ ): GenerateResponseData;
+
+ protected abstract toGenkitPart(
+ event: RunnerStreamEvent
+ ): Part | undefined;
+
+ public async run(
+ request: GenerateRequest,
+ options: {
+ streamingRequested: boolean;
+ sendChunk: (chunk: GenerateResponseChunkData) => void;
+ abortSignal: AbortSignal;
+ }
+ ): Promise {
+ const { streamingRequested, sendChunk, abortSignal } = options;
+
+ if (streamingRequested) {
+ const body = this.toAnthropicStreamingRequestBody(
+ this.name,
+ request,
+ this.cacheSystemPrompt
+ );
+ const stream = this.streamMessages(body, abortSignal);
+ for await (const event of stream) {
+ const part = this.toGenkitPart(event);
+ if (part) {
+ sendChunk({
+ index: 0,
+ content: [part],
+ });
+ }
+ }
+ const finalMessage = await stream.finalMessage();
+ return this.toGenkitResponse(finalMessage);
+ }
+
+ const body = this.toAnthropicRequestBody(
+ this.name,
+ request,
+ this.cacheSystemPrompt
+ );
+ const response = await this.createMessage(body, abortSignal);
+ return this.toGenkitResponse(response);
+ }
+}
diff --git a/js/plugins/anthropic/src/runner/beta.ts b/js/plugins/anthropic/src/runner/beta.ts
new file mode 100644
index 0000000000..6a71fa71d5
--- /dev/null
+++ b/js/plugins/anthropic/src/runner/beta.ts
@@ -0,0 +1,494 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { BetaMessageStream } from '@anthropic-ai/sdk/lib/BetaMessageStream.js';
+import type {
+ BetaContentBlock,
+ BetaImageBlockParam,
+ BetaMessage,
+ MessageCreateParams as BetaMessageCreateParams,
+ MessageCreateParamsNonStreaming as BetaMessageCreateParamsNonStreaming,
+ MessageCreateParamsStreaming as BetaMessageCreateParamsStreaming,
+ BetaMessageParam,
+ BetaRawMessageStreamEvent,
+ BetaRedactedThinkingBlockParam,
+ BetaRequestDocumentBlock,
+ BetaStopReason,
+ BetaTextBlockParam,
+ BetaThinkingBlockParam,
+ BetaTool,
+ BetaToolResultBlockParam,
+ BetaToolUseBlockParam,
+} from '@anthropic-ai/sdk/resources/beta/messages';
+
+import type {
+ GenerateRequest,
+ GenerateResponseData,
+ ModelResponseData,
+ Part,
+} from 'genkit';
+import { logger } from 'genkit/logging';
+
+import { KNOWN_CLAUDE_MODELS, extractVersion } from '../models.js';
+import { AnthropicConfigSchema, type ClaudeRunnerParams } from '../types.js';
+import { BaseRunner } from './base.js';
+import { RunnerTypes } from './types.js';
+
+/**
+ * Server-managed tool blocks emitted by the beta API that Genkit cannot yet
+ * interpret. We fail fast on these so callers do not accidentally treat them as
+ * locally executable tool invocations.
+ */
+/**
+ * Server tool types that exist in beta but are not yet supported.
+ * Note: server_tool_use and web_search_tool_result ARE supported (same as stable API).
+ */
+const BETA_UNSUPPORTED_SERVER_TOOL_BLOCK_TYPES = new Set([
+ 'web_fetch_tool_result',
+ 'code_execution_tool_result',
+ 'bash_code_execution_tool_result',
+ 'text_editor_code_execution_tool_result',
+ 'mcp_tool_result',
+ 'mcp_tool_use',
+ 'container_upload',
+]);
+
+const unsupportedServerToolError = (blockType: string): string =>
+ `Anthropic beta runner does not yet support server-managed tool block '${blockType}'. Please retry against the stable API or wait for dedicated support.`;
+
+interface BetaRunnerTypes extends RunnerTypes {
+ Message: BetaMessage;
+ Stream: BetaMessageStream;
+ StreamEvent: BetaRawMessageStreamEvent;
+ RequestBody: BetaMessageCreateParamsNonStreaming;
+ StreamingRequestBody: BetaMessageCreateParamsStreaming;
+ Tool: BetaTool;
+ MessageParam: BetaMessageParam;
+ ToolResponseContent: BetaTextBlockParam | BetaImageBlockParam;
+ ContentBlockParam:
+ | BetaTextBlockParam
+ | BetaImageBlockParam
+ | BetaRequestDocumentBlock
+ | BetaToolUseBlockParam
+ | BetaToolResultBlockParam
+ | BetaThinkingBlockParam
+ | BetaRedactedThinkingBlockParam;
+}
+
+/**
+ * Runner for the Anthropic Beta API.
+ */
+export class BetaRunner extends BaseRunner {
+ constructor(params: ClaudeRunnerParams) {
+ super(params);
+ }
+
+ /**
+ * Map a Genkit Part -> Anthropic beta content block param.
+ * Supports: text, images (base64 data URLs), PDFs (document source),
+ * tool_use (client tool request), tool_result (client tool response).
+ */
+ protected toAnthropicMessageContent(
+ part: Part
+ ):
+ | BetaTextBlockParam
+ | BetaImageBlockParam
+ | BetaRequestDocumentBlock
+ | BetaToolUseBlockParam
+ | BetaToolResultBlockParam
+ | BetaThinkingBlockParam
+ | BetaRedactedThinkingBlockParam {
+ if (part.reasoning) {
+ const signature = this.getThinkingSignature(part);
+ if (!signature) {
+ throw new Error(
+ 'Anthropic thinking parts require a signature when sending back to the API. Preserve the `custom.anthropicThinking.signature` value from the original response.'
+ );
+ }
+ return {
+ type: 'thinking',
+ thinking: part.reasoning,
+ signature,
+ };
+ }
+
+ const redactedThinking = this.getRedactedThinkingData(part);
+ if (redactedThinking !== undefined) {
+ return {
+ type: 'redacted_thinking',
+ data: redactedThinking,
+ };
+ }
+
+ // Text
+ if (part.text) {
+ return { type: 'text', text: part.text };
+ }
+
+ // Media
+ if (part.media) {
+ if (part.media.contentType === 'application/pdf') {
+ return {
+ type: 'document',
+ source: this.toPdfDocumentSource(part.media),
+ };
+ }
+
+ const source = this.toImageSource(part.media);
+ if (source.kind === 'base64') {
+ return {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data: source.data,
+ media_type: source.mediaType,
+ },
+ };
+ }
+ return {
+ type: 'image',
+ source: {
+ type: 'url',
+ url: source.url,
+ },
+ };
+ }
+
+ // Tool request (client tool use)
+ if (part.toolRequest) {
+ if (!part.toolRequest.ref) {
+ throw new Error(
+ `Tool request ref is required for Anthropic API. Part: ${JSON.stringify(
+ part.toolRequest
+ )}`
+ );
+ }
+ return {
+ type: 'tool_use',
+ id: part.toolRequest.ref,
+ name: part.toolRequest.name,
+ input: part.toolRequest.input,
+ };
+ }
+
+ // Tool response (client tool result)
+ if (part.toolResponse) {
+ if (!part.toolResponse.ref) {
+ throw new Error(
+ `Tool response ref is required for Anthropic API. Part: ${JSON.stringify(
+ part.toolResponse
+ )}`
+ );
+ }
+ const betaResult: BetaToolResultBlockParam = {
+ type: 'tool_result',
+ tool_use_id: part.toolResponse.ref,
+ content: [this.toAnthropicToolResponseContent(part)],
+ };
+ return betaResult;
+ }
+
+ throw new Error(
+ `Unsupported genkit part fields encountered for current message role: ${JSON.stringify(
+ part
+ )}.`
+ );
+ }
+
+ protected createMessage(
+ body: BetaMessageCreateParamsNonStreaming,
+ abortSignal: AbortSignal
+ ): Promise {
+ return this.client.beta.messages.create(body, { signal: abortSignal });
+ }
+
+ protected streamMessages(
+ body: BetaMessageCreateParamsStreaming,
+ abortSignal: AbortSignal
+ ): BetaMessageStream {
+ return this.client.beta.messages.stream(body, { signal: abortSignal });
+ }
+
+ /**
+ * Build non-streaming request body.
+ */
+ protected toAnthropicRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): BetaMessageCreateParamsNonStreaming {
+ const model = KNOWN_CLAUDE_MODELS[modelName];
+ const { system, messages } = this.toAnthropicMessages(request.messages);
+ const mappedModelName =
+ request.config?.version ?? extractVersion(model, modelName);
+
+ let betaSystem: BetaMessageCreateParamsNonStreaming['system'];
+
+ if (system !== undefined) {
+ betaSystem = cacheSystemPrompt
+ ? [
+ {
+ type: 'text' as const,
+ text: system,
+ cache_control: { type: 'ephemeral' as const },
+ },
+ ]
+ : system;
+ }
+
+ const body: BetaMessageCreateParamsNonStreaming = {
+ model: mappedModelName,
+ max_tokens:
+ request.config?.maxOutputTokens ?? this.DEFAULT_MAX_OUTPUT_TOKENS,
+ messages,
+ };
+
+ if (betaSystem !== undefined) body.system = betaSystem;
+ if (request.config?.stopSequences !== undefined)
+ body.stop_sequences = request.config.stopSequences;
+ if (request.config?.temperature !== undefined)
+ body.temperature = request.config.temperature;
+ if (request.config?.topK !== undefined) body.top_k = request.config.topK;
+ if (request.config?.topP !== undefined) body.top_p = request.config.topP;
+ if (request.config?.tool_choice !== undefined) {
+ body.tool_choice = request.config
+ .tool_choice as BetaMessageCreateParams['tool_choice'];
+ }
+ if (request.config?.metadata !== undefined) {
+ body.metadata = request.config
+ .metadata as BetaMessageCreateParams['metadata'];
+ }
+ if (request.tools) {
+ body.tools = request.tools.map((tool) => this.toAnthropicTool(tool));
+ }
+ const thinkingConfig = this.toAnthropicThinkingConfig(
+ request.config?.thinking
+ );
+ if (thinkingConfig) {
+ body.thinking = thinkingConfig as BetaMessageCreateParams['thinking'];
+ }
+
+ if (request.output?.format && request.output.format !== 'text') {
+ throw new Error(
+ `Only text output format is supported for Claude models currently`
+ );
+ }
+
+ return body;
+ }
+
+ /**
+ * Build streaming request body.
+ */
+ protected toAnthropicStreamingRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): BetaMessageCreateParamsStreaming {
+ const model = KNOWN_CLAUDE_MODELS[modelName];
+ const { system, messages } = this.toAnthropicMessages(request.messages);
+ const mappedModelName =
+ request.config?.version ?? extractVersion(model, modelName);
+
+ const betaSystem =
+ system === undefined
+ ? undefined
+ : cacheSystemPrompt
+ ? [
+ {
+ type: 'text' as const,
+ text: system,
+ cache_control: { type: 'ephemeral' as const },
+ },
+ ]
+ : system;
+
+ const body: BetaMessageCreateParamsStreaming = {
+ model: mappedModelName,
+ max_tokens:
+ request.config?.maxOutputTokens ?? this.DEFAULT_MAX_OUTPUT_TOKENS,
+ messages,
+ stream: true,
+ };
+
+ if (betaSystem !== undefined) body.system = betaSystem;
+ if (request.config?.stopSequences !== undefined)
+ body.stop_sequences = request.config.stopSequences;
+ if (request.config?.temperature !== undefined)
+ body.temperature = request.config.temperature;
+ if (request.config?.topK !== undefined) body.top_k = request.config.topK;
+ if (request.config?.topP !== undefined) body.top_p = request.config.topP;
+ if (request.config?.tool_choice !== undefined) {
+ body.tool_choice = request.config
+ .tool_choice as BetaMessageCreateParams['tool_choice'];
+ }
+ if (request.config?.metadata !== undefined) {
+ body.metadata = request.config
+ .metadata as BetaMessageCreateParams['metadata'];
+ }
+ if (request.tools) {
+ body.tools = request.tools.map((tool) => this.toAnthropicTool(tool));
+ }
+ const thinkingConfig = this.toAnthropicThinkingConfig(
+ request.config?.thinking
+ );
+ if (thinkingConfig) {
+ body.thinking = thinkingConfig as BetaMessageCreateParams['thinking'];
+ }
+
+ if (request.output?.format && request.output.format !== 'text') {
+ throw new Error(
+ `Only text output format is supported for Claude models currently`
+ );
+ }
+
+ return body;
+ }
+
+ protected toGenkitResponse(message: BetaMessage): GenerateResponseData {
+ return {
+ candidates: [
+ {
+ index: 0,
+ finishReason: this.fromBetaStopReason(message.stop_reason),
+ message: {
+ role: 'model',
+ content: message.content.map((block) =>
+ this.fromBetaContentBlock(block)
+ ),
+ },
+ },
+ ],
+ usage: {
+ inputTokens: message.usage.input_tokens,
+ outputTokens: message.usage.output_tokens,
+ },
+ custom: message,
+ };
+ }
+
+ protected toGenkitPart(event: BetaRawMessageStreamEvent): Part | undefined {
+ if (event.type === 'content_block_start') {
+ const blockType = (event.content_block as { type?: string }).type;
+ if (
+ blockType &&
+ BETA_UNSUPPORTED_SERVER_TOOL_BLOCK_TYPES.has(blockType)
+ ) {
+ throw new Error(unsupportedServerToolError(blockType));
+ }
+ return this.fromBetaContentBlock(event.content_block);
+ }
+ if (event.type === 'content_block_delta') {
+ if (event.delta.type === 'text_delta') {
+ return { text: event.delta.text };
+ }
+ if (event.delta.type === 'thinking_delta') {
+ return { reasoning: event.delta.thinking };
+ }
+ // server/client tool input_json_delta not supported yet
+ return undefined;
+ }
+ return undefined;
+ }
+
+ private fromBetaContentBlock(contentBlock: BetaContentBlock): Part {
+ switch (contentBlock.type) {
+ case 'tool_use': {
+ return {
+ toolRequest: {
+ ref: contentBlock.id,
+ name: contentBlock.name ?? 'unknown_tool',
+ input: contentBlock.input,
+ },
+ };
+ }
+
+ case 'mcp_tool_use':
+ throw new Error(unsupportedServerToolError(contentBlock.type));
+
+ case 'server_tool_use': {
+ const baseName = contentBlock.name ?? 'unknown_tool';
+ const serverToolName =
+ 'server_name' in contentBlock && contentBlock.server_name
+ ? `${contentBlock.server_name}/${baseName}`
+ : baseName;
+ return {
+ text: `[Anthropic server tool ${serverToolName}] input: ${JSON.stringify(contentBlock.input)}`,
+ custom: {
+ anthropicServerToolUse: {
+ id: contentBlock.id,
+ name: serverToolName,
+ input: contentBlock.input,
+ },
+ },
+ };
+ }
+
+ case 'web_search_tool_result':
+ return this.toWebSearchToolResultPart({
+ type: contentBlock.type,
+ toolUseId: contentBlock.tool_use_id,
+ content: contentBlock.content,
+ });
+
+ case 'text':
+ return { text: contentBlock.text };
+
+ case 'thinking':
+ return this.createThinkingPart(
+ contentBlock.thinking,
+ contentBlock.signature
+ );
+
+ case 'redacted_thinking':
+ return { custom: { redactedThinking: contentBlock.data } };
+
+ default: {
+ if (BETA_UNSUPPORTED_SERVER_TOOL_BLOCK_TYPES.has(contentBlock.type)) {
+ throw new Error(unsupportedServerToolError(contentBlock.type));
+ }
+ const unknownType = (contentBlock as { type: string }).type;
+ logger.warn(
+ `Unexpected Anthropic beta content block type: ${unknownType}. Returning empty text. Content block: ${JSON.stringify(
+ contentBlock
+ )}`
+ );
+ return { text: '' };
+ }
+ }
+ }
+
+ private fromBetaStopReason(
+ reason: BetaStopReason | null
+ ): ModelResponseData['finishReason'] {
+ switch (reason) {
+ case 'max_tokens':
+ case 'model_context_window_exceeded':
+ return 'length';
+ case 'end_turn':
+ case 'stop_sequence':
+ case 'tool_use':
+ case 'pause_turn':
+ return 'stop';
+ case null:
+ return 'unknown';
+ case 'refusal':
+ return 'other';
+ default:
+ return 'other';
+ }
+ }
+}
diff --git a/js/plugins/anthropic/src/runner/index.ts b/js/plugins/anthropic/src/runner/index.ts
new file mode 100644
index 0000000000..ce7e3c6fdd
--- /dev/null
+++ b/js/plugins/anthropic/src/runner/index.ts
@@ -0,0 +1,19 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+export { BaseRunner } from './base.js';
+export { BetaRunner } from './beta.js';
+export { Runner } from './stable.js';
diff --git a/js/plugins/anthropic/src/runner/stable.ts b/js/plugins/anthropic/src/runner/stable.ts
new file mode 100644
index 0000000000..0c8f7ffc4f
--- /dev/null
+++ b/js/plugins/anthropic/src/runner/stable.ts
@@ -0,0 +1,514 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { MessageStream } from '@anthropic-ai/sdk/lib/MessageStream.js';
+import type {
+ ContentBlock,
+ DocumentBlockParam,
+ ImageBlockParam,
+ Message,
+ MessageCreateParams,
+ MessageCreateParamsNonStreaming,
+ MessageCreateParamsStreaming,
+ MessageParam,
+ MessageStreamEvent,
+ RedactedThinkingBlockParam,
+ TextBlockParam,
+ ThinkingBlockParam,
+ Tool,
+ ToolResultBlockParam,
+ ToolUseBlockParam,
+} from '@anthropic-ai/sdk/resources/messages';
+import type {
+ GenerateRequest,
+ GenerateResponseData,
+ ModelResponseData,
+ Part,
+} from 'genkit';
+import { logger } from 'genkit/logging';
+
+import { KNOWN_CLAUDE_MODELS, extractVersion } from '../models.js';
+import { AnthropicConfigSchema, type ClaudeRunnerParams } from '../types.js';
+import { BaseRunner } from './base.js';
+import { RunnerTypes as BaseRunnerTypes } from './types.js';
+interface RunnerTypes extends BaseRunnerTypes {
+ Message: Message;
+ Stream: MessageStream;
+ StreamEvent: MessageStreamEvent;
+ RequestBody: MessageCreateParamsNonStreaming;
+ StreamingRequestBody: MessageCreateParamsStreaming;
+ Tool: Tool;
+ MessageParam: MessageParam;
+ ToolResponseContent: TextBlockParam | ImageBlockParam;
+ ContentBlockParam:
+ | TextBlockParam
+ | ImageBlockParam
+ | DocumentBlockParam
+ | ToolUseBlockParam
+ | ToolResultBlockParam
+ | ThinkingBlockParam
+ | RedactedThinkingBlockParam;
+}
+
+export class Runner extends BaseRunner {
+ constructor(params: ClaudeRunnerParams) {
+ super(params);
+ }
+
+ protected toAnthropicMessageContent(
+ part: Part
+ ):
+ | TextBlockParam
+ | ImageBlockParam
+ | DocumentBlockParam
+ | ToolUseBlockParam
+ | ToolResultBlockParam
+ | ThinkingBlockParam
+ | RedactedThinkingBlockParam {
+ if (part.reasoning) {
+ const signature = this.getThinkingSignature(part);
+ if (!signature) {
+ throw new Error(
+ 'Anthropic thinking parts require a signature when sending back to the API. Preserve the `custom.anthropicThinking.signature` value from the original response.'
+ );
+ }
+ return {
+ type: 'thinking',
+ thinking: part.reasoning,
+ signature,
+ };
+ }
+
+ const redactedThinking = this.getRedactedThinkingData(part);
+ if (redactedThinking !== undefined) {
+ return {
+ type: 'redacted_thinking',
+ data: redactedThinking,
+ };
+ }
+
+ if (part.text) {
+ return {
+ type: 'text',
+ text: part.text,
+ citations: null,
+ };
+ }
+
+ if (part.media) {
+ if (part.media.contentType === 'application/pdf') {
+ return {
+ type: 'document',
+ source: this.toPdfDocumentSource(part.media),
+ };
+ }
+
+ const source = this.toImageSource(part.media);
+ if (source.kind === 'base64') {
+ return {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data: source.data,
+ media_type: source.mediaType,
+ },
+ };
+ }
+ return {
+ type: 'image',
+ source: {
+ type: 'url',
+ url: source.url,
+ },
+ };
+ }
+
+ if (part.toolRequest) {
+ if (!part.toolRequest.ref) {
+ throw new Error(
+ `Tool request ref is required for Anthropic API. Part: ${JSON.stringify(
+ part.toolRequest
+ )}`
+ );
+ }
+ return {
+ type: 'tool_use',
+ id: part.toolRequest.ref,
+ name: part.toolRequest.name,
+ input: part.toolRequest.input,
+ };
+ }
+
+ if (part.toolResponse) {
+ if (!part.toolResponse.ref) {
+ throw new Error(
+ `Tool response ref is required for Anthropic API. Part: ${JSON.stringify(
+ part.toolResponse
+ )}`
+ );
+ }
+ return {
+ type: 'tool_result',
+ tool_use_id: part.toolResponse.ref,
+ content: [this.toAnthropicToolResponseContent(part)],
+ };
+ }
+
+ throw new Error(
+ `Unsupported genkit part fields encountered for current message role: ${JSON.stringify(
+ part
+ )}.`
+ );
+ }
+
+ protected toAnthropicRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): MessageCreateParamsNonStreaming {
+ const model = KNOWN_CLAUDE_MODELS[modelName];
+ const { system, messages } = this.toAnthropicMessages(request.messages);
+ const mappedModelName =
+ request.config?.version ?? extractVersion(model, modelName);
+
+ const systemValue =
+ system === undefined
+ ? undefined
+ : cacheSystemPrompt
+ ? [
+ {
+ type: 'text' as const,
+ text: system,
+ cache_control: { type: 'ephemeral' as const },
+ },
+ ]
+ : system;
+
+ const body: MessageCreateParamsNonStreaming = {
+ model: mappedModelName,
+ max_tokens:
+ request.config?.maxOutputTokens ?? this.DEFAULT_MAX_OUTPUT_TOKENS,
+ messages,
+ };
+
+ if (systemValue !== undefined) {
+ body.system = systemValue;
+ }
+
+ if (request.tools) {
+ body.tools = request.tools.map((tool) => this.toAnthropicTool(tool));
+ }
+ if (request.config?.topK !== undefined) {
+ body.top_k = request.config.topK;
+ }
+ if (request.config?.topP !== undefined) {
+ body.top_p = request.config.topP;
+ }
+ if (request.config?.temperature !== undefined) {
+ body.temperature = request.config.temperature;
+ }
+ if (request.config?.stopSequences !== undefined) {
+ body.stop_sequences = request.config.stopSequences;
+ }
+ if (request.config?.metadata !== undefined) {
+ body.metadata = request.config.metadata;
+ }
+ if (request.config?.tool_choice !== undefined) {
+ body.tool_choice = request.config.tool_choice;
+ }
+ const thinkingConfig = this.toAnthropicThinkingConfig(
+ request.config?.thinking
+ );
+ if (thinkingConfig) {
+ body.thinking = thinkingConfig as MessageCreateParams['thinking'];
+ }
+
+ if (request.output?.format && request.output.format !== 'text') {
+ throw new Error(
+ `Only text output format is supported for Claude models currently`
+ );
+ }
+ return body;
+ }
+
+ protected toAnthropicStreamingRequestBody(
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ): MessageCreateParamsStreaming {
+ const model = KNOWN_CLAUDE_MODELS[modelName];
+ const { system, messages } = this.toAnthropicMessages(request.messages);
+ const mappedModelName =
+ request.config?.version ?? extractVersion(model, modelName);
+
+ const systemValue =
+ system === undefined
+ ? undefined
+ : cacheSystemPrompt
+ ? [
+ {
+ type: 'text' as const,
+ text: system,
+ cache_control: { type: 'ephemeral' as const },
+ },
+ ]
+ : system;
+
+ const body: MessageCreateParamsStreaming = {
+ model: mappedModelName,
+ max_tokens:
+ request.config?.maxOutputTokens ?? this.DEFAULT_MAX_OUTPUT_TOKENS,
+ messages,
+ stream: true,
+ };
+
+ if (systemValue !== undefined) {
+ body.system = systemValue;
+ }
+
+ if (request.tools) {
+ body.tools = request.tools.map((tool) => this.toAnthropicTool(tool));
+ }
+ if (request.config?.topK !== undefined) {
+ body.top_k = request.config.topK;
+ }
+ if (request.config?.topP !== undefined) {
+ body.top_p = request.config.topP;
+ }
+ if (request.config?.temperature !== undefined) {
+ body.temperature = request.config.temperature;
+ }
+ if (request.config?.stopSequences !== undefined) {
+ body.stop_sequences = request.config.stopSequences;
+ }
+ if (request.config?.metadata !== undefined) {
+ body.metadata = request.config.metadata;
+ }
+ if (request.config?.tool_choice !== undefined) {
+ body.tool_choice = request.config.tool_choice;
+ }
+ const thinkingConfig = this.toAnthropicThinkingConfig(
+ request.config?.thinking
+ );
+ if (thinkingConfig) {
+ body.thinking =
+ thinkingConfig as MessageCreateParamsStreaming['thinking'];
+ }
+
+ if (request.output?.format && request.output.format !== 'text') {
+ throw new Error(
+ `Only text output format is supported for Claude models currently`
+ );
+ }
+ return body;
+ }
+
+ protected async createMessage(
+ body: MessageCreateParamsNonStreaming,
+ abortSignal: AbortSignal
+ ): Promise {
+ return await this.client.messages.create(body, { signal: abortSignal });
+ }
+
+ protected streamMessages(
+ body: MessageCreateParamsStreaming,
+ abortSignal: AbortSignal
+ ): MessageStream {
+ return this.client.messages.stream(body, { signal: abortSignal });
+ }
+
+ protected toGenkitResponse(message: Message): GenerateResponseData {
+ return this.fromAnthropicResponse(message);
+ }
+
+ protected toGenkitPart(event: MessageStreamEvent): Part | undefined {
+ return this.fromAnthropicContentBlockChunk(event);
+ }
+
+ protected fromAnthropicContentBlockChunk(
+ event: MessageStreamEvent
+ ): Part | undefined {
+ // Handle content_block_delta events
+ if (event.type === 'content_block_delta') {
+ const delta = event.delta;
+
+ if (delta.type === 'input_json_delta') {
+ throw new Error(
+ 'Anthropic streaming tool input (input_json_delta) is not yet supported. Please disable streaming or upgrade this plugin.'
+ );
+ }
+
+ if (delta.type === 'text_delta') {
+ return { text: delta.text };
+ }
+
+ if (delta.type === 'thinking_delta') {
+ return { reasoning: delta.thinking };
+ }
+
+ // signature_delta - ignore
+ return undefined;
+ }
+
+ // Handle content_block_start events
+ if (event.type === 'content_block_start') {
+ const block = event.content_block;
+
+ switch (block.type) {
+ case 'server_tool_use':
+ return {
+ text: `[Anthropic server tool ${block.name}] input: ${JSON.stringify(block.input)}`,
+ custom: {
+ anthropicServerToolUse: {
+ id: block.id,
+ name: block.name,
+ input: block.input,
+ },
+ },
+ };
+
+ case 'web_search_tool_result':
+ return this.toWebSearchToolResultPart({
+ type: block.type,
+ toolUseId: block.tool_use_id,
+ content: block.content,
+ });
+
+ case 'text':
+ return { text: block.text };
+
+ case 'thinking':
+ return this.createThinkingPart(block.thinking, block.signature);
+
+ case 'redacted_thinking':
+ return { custom: { redactedThinking: block.data } };
+
+ case 'tool_use':
+ return {
+ toolRequest: {
+ ref: block.id,
+ name: block.name,
+ input: block.input,
+ },
+ };
+
+ default: {
+ const unknownType = (block as { type: string }).type;
+ logger.warn(
+ `Unexpected Anthropic content block type in stream: ${unknownType}. Returning undefined. Content block: ${JSON.stringify(block)}`
+ );
+ return undefined;
+ }
+ }
+ }
+
+ // Other event types (message_start, message_delta, etc.) - ignore
+ return undefined;
+ }
+
+ protected fromAnthropicContentBlock(contentBlock: ContentBlock): Part {
+ switch (contentBlock.type) {
+ case 'server_tool_use':
+ return {
+ text: `[Anthropic server tool ${contentBlock.name}] input: ${JSON.stringify(contentBlock.input)}`,
+ custom: {
+ anthropicServerToolUse: {
+ id: contentBlock.id,
+ name: contentBlock.name,
+ input: contentBlock.input,
+ },
+ },
+ };
+
+ case 'web_search_tool_result':
+ return this.toWebSearchToolResultPart({
+ type: contentBlock.type,
+ toolUseId: contentBlock.tool_use_id,
+ content: contentBlock.content,
+ });
+
+ case 'tool_use':
+ return {
+ toolRequest: {
+ ref: contentBlock.id,
+ name: contentBlock.name,
+ input: contentBlock.input,
+ },
+ };
+
+ case 'text':
+ return { text: contentBlock.text };
+
+ case 'thinking':
+ return this.createThinkingPart(
+ contentBlock.thinking,
+ contentBlock.signature
+ );
+
+ case 'redacted_thinking':
+ return { custom: { redactedThinking: contentBlock.data } };
+
+ default: {
+ const unknownType = (contentBlock as { type: string }).type;
+ logger.warn(
+ `Unexpected Anthropic content block type: ${unknownType}. Returning empty text. Content block: ${JSON.stringify(contentBlock)}`
+ );
+ return { text: '' };
+ }
+ }
+ }
+
+ protected fromAnthropicStopReason(
+ reason: Message['stop_reason']
+ ): ModelResponseData['finishReason'] {
+ switch (reason) {
+ case 'max_tokens':
+ return 'length';
+ case 'end_turn':
+ // fall through
+ case 'stop_sequence':
+ // fall through
+ case 'tool_use':
+ return 'stop';
+ case null:
+ return 'unknown';
+ default:
+ return 'other';
+ }
+ }
+
+ protected fromAnthropicResponse(response: Message): GenerateResponseData {
+ return {
+ candidates: [
+ {
+ index: 0,
+ finishReason: this.fromAnthropicStopReason(response.stop_reason),
+ message: {
+ role: 'model',
+ content: response.content.map((block) =>
+ this.fromAnthropicContentBlock(block)
+ ),
+ },
+ },
+ ],
+ usage: {
+ inputTokens: response.usage.input_tokens,
+ outputTokens: response.usage.output_tokens,
+ },
+ custom: response,
+ };
+ }
+}
diff --git a/js/plugins/anthropic/src/runner/types.ts b/js/plugins/anthropic/src/runner/types.ts
new file mode 100644
index 0000000000..5fd04c6911
--- /dev/null
+++ b/js/plugins/anthropic/src/runner/types.ts
@@ -0,0 +1,78 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+/**
+ * Type contract that each Anthropic runner passes into the generic `BaseRunner`.
+ *
+ * The concrete runners (stable vs. beta SDKs) bind these slots to their SDK’s
+ * concrete interfaces so the shared logic in `BaseRunner` can stay strongly typed
+ * without knowing which SDK variant it is talking to.
+ *
+ * Properties are `unknown` by default, so every subclass must plug in the
+ * correct Anthropic types to keep the generic plumbing sound.
+ */
+type RunnerTypes = {
+ Message: unknown;
+ Stream: AsyncIterable & { finalMessage(): Promise };
+ StreamEvent: unknown;
+ RequestBody: unknown;
+ StreamingRequestBody: unknown;
+ Tool: unknown;
+ MessageParam: unknown;
+ ContentBlockParam: unknown;
+ ToolResponseContent: unknown;
+};
+
+type RunnerMessage = ApiTypes['Message'];
+
+/** Streaming handle that yields Anthropic events and exposes the final message. */
+type RunnerStream = ApiTypes['Stream'];
+
+/** Discrete event emitted by the Anthropic stream (delta, block start, etc.). */
+type RunnerStreamEvent = ApiTypes['StreamEvent'];
+
+/** Non-streaming request payload shape for create-message calls. */
+type RunnerRequestBody = ApiTypes['RequestBody'];
+type RunnerStreamingRequestBody =
+ ApiTypes['StreamingRequestBody'];
+
+/** Tool definition compatible with the target Anthropic SDK. */
+type RunnerTool = ApiTypes['Tool'];
+
+/** Anthropic message param shape used when sending history to the API. */
+type RunnerMessageParam =
+ ApiTypes['MessageParam'];
+
+/** Content block that the runner sends to Anthropic for a single part. */
+type RunnerContentBlockParam =
+ ApiTypes['ContentBlockParam'];
+
+/** Tool response block that Anthropic expects when returning tool output. */
+type RunnerToolResponseContent =
+ ApiTypes['ToolResponseContent'];
+
+export {
+ RunnerContentBlockParam,
+ RunnerMessage,
+ RunnerMessageParam,
+ RunnerRequestBody,
+ RunnerStream,
+ RunnerStreamEvent,
+ RunnerStreamingRequestBody,
+ RunnerTool,
+ RunnerToolResponseContent,
+ RunnerTypes,
+};
diff --git a/js/plugins/anthropic/src/types.ts b/js/plugins/anthropic/src/types.ts
new file mode 100644
index 0000000000..9796d71c36
--- /dev/null
+++ b/js/plugins/anthropic/src/types.ts
@@ -0,0 +1,166 @@
+/**
+ * Copyright 2024 Bloom Labs Inc
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import type Anthropic from '@anthropic-ai/sdk';
+import { z } from 'genkit';
+import { GenerationCommonConfigSchema } from 'genkit/model';
+
+/**
+ * Internal symbol for dependency injection in tests.
+ * Not part of the public API.
+ * @internal
+ */
+export const __testClient = Symbol('testClient');
+
+/**
+ * Plugin configuration options for the Anthropic plugin.
+ */
+export interface PluginOptions {
+ apiKey?: string;
+ cacheSystemPrompt?: boolean;
+ /** Default API surface for all requests unless overridden per-request. */
+ apiVersion?: 'stable' | 'beta';
+}
+
+/**
+ * Internal plugin options that include test client injection.
+ * @internal
+ */
+export interface InternalPluginOptions extends PluginOptions {
+ [__testClient]?: Anthropic;
+}
+
+/**
+ * Shared parameters required to construct Claude helpers.
+ */
+interface ClaudeHelperParamsBase {
+ name: string;
+ client: Anthropic;
+ cacheSystemPrompt?: boolean;
+ defaultApiVersion?: 'stable' | 'beta';
+}
+
+/**
+ * Parameters for creating a Claude model action.
+ */
+export interface ClaudeModelParams extends ClaudeHelperParamsBase {}
+
+/**
+ * Parameters for creating a Claude runner.
+ */
+export interface ClaudeRunnerParams extends ClaudeHelperParamsBase {}
+
+export const AnthropicBaseConfigSchema = GenerationCommonConfigSchema.extend({
+ tool_choice: z
+ .union([
+ z.object({
+ type: z.literal('auto'),
+ }),
+ z.object({
+ type: z.literal('any'),
+ }),
+ z.object({
+ type: z.literal('tool'),
+ name: z.string(),
+ }),
+ ])
+ .optional(),
+ metadata: z
+ .object({
+ user_id: z.string().optional(),
+ })
+ .optional(),
+ /** Optional shorthand to pick API surface for this request. */
+ apiVersion: z.enum(['stable', 'beta']).optional(),
+});
+
+export type AnthropicBaseConfigSchemaType = typeof AnthropicBaseConfigSchema;
+
+export const ThinkingConfigSchema = z
+ .object({
+ enabled: z.boolean().optional(),
+ budgetTokens: z.number().int().min(1_024).optional(),
+ })
+ .superRefine((value, ctx) => {
+ if (value.enabled && value.budgetTokens === undefined) {
+ ctx.addIssue({
+ code: z.ZodIssueCode.custom,
+ path: ['budgetTokens'],
+ message: 'budgetTokens is required when thinking is enabled',
+ });
+ }
+ });
+
+export const AnthropicThinkingConfigSchema = AnthropicBaseConfigSchema.extend({
+ thinking: ThinkingConfigSchema.optional(),
+});
+
+export const AnthropicConfigSchema = AnthropicThinkingConfigSchema;
+
+export type ThinkingConfig = z.infer;
+export type AnthropicBaseConfig = z.infer;
+export type AnthropicThinkingConfig = z.infer<
+ typeof AnthropicThinkingConfigSchema
+>;
+export type ClaudeConfig = AnthropicThinkingConfig | AnthropicBaseConfig;
+
+/**
+ * Media object representation with URL and optional content type.
+ */
+export interface Media {
+ url: string;
+ contentType?: string;
+}
+
+export const MediaSchema = z.object({
+ url: z.string(),
+ contentType: z.string().optional(),
+});
+
+export const MediaTypeSchema = z.enum([
+ 'image/jpeg',
+ 'image/png',
+ 'image/gif',
+ 'image/webp',
+]);
+
+export type MediaType = z.infer;
+
+export const MEDIA_TYPES = {
+ JPEG: 'image/jpeg',
+ PNG: 'image/png',
+ GIF: 'image/gif',
+ WEBP: 'image/webp',
+} as const satisfies Record;
+
+/**
+ * Resolve whether beta API should be used for this call.
+ * Priority:
+ * 1. request.config.apiVersion (per-request override - explicit stable or beta)
+ * 2. pluginDefaultApiVersion (plugin-wide default)
+ * 3. otherwise stable
+ */
+export function resolveBetaEnabled(
+ cfg: AnthropicThinkingConfig | AnthropicBaseConfig | undefined,
+ pluginDefaultApiVersion?: 'stable' | 'beta'
+): boolean {
+ if (cfg?.apiVersion !== undefined) {
+ return cfg.apiVersion === 'beta';
+ }
+ if (pluginDefaultApiVersion === 'beta') return true;
+ return false;
+}
diff --git a/js/plugins/anthropic/tests/beta_runner_test.ts b/js/plugins/anthropic/tests/beta_runner_test.ts
new file mode 100644
index 0000000000..655bfc599e
--- /dev/null
+++ b/js/plugins/anthropic/tests/beta_runner_test.ts
@@ -0,0 +1,804 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import * as assert from 'assert';
+import type { Part } from 'genkit';
+import { describe, it } from 'node:test';
+
+import { BetaRunner } from '../src/runner/beta.js';
+import { createMockAnthropicClient } from './mocks/anthropic-client.js';
+
+describe('BetaRunner.toAnthropicMessageContent', () => {
+ function createRunner() {
+ return new BetaRunner({
+ name: 'anthropic/claude-3-5-haiku',
+ client: createMockAnthropicClient(),
+ cacheSystemPrompt: false,
+ });
+ }
+
+ it('converts PDF media parts into document blocks', () => {
+ const runner = createRunner();
+ const part: Part = {
+ media: {
+ contentType: 'application/pdf',
+ url: 'data:application/pdf;base64,UEsDBAoAAAAAAD',
+ },
+ };
+
+ const result = (runner as any).toAnthropicMessageContent(part);
+
+ assert.strictEqual(result.type, 'document');
+ assert.ok(result.source);
+ assert.strictEqual(result.source.type, 'base64');
+ assert.strictEqual(result.source.media_type, 'application/pdf');
+ assert.ok(result.source.data);
+ });
+
+ it('throws when tool request ref is missing', () => {
+ const runner = createRunner();
+ const part: Part = {
+ toolRequest: {
+ name: 'do_something',
+ input: { foo: 'bar' },
+ },
+ };
+
+ assert.throws(() => {
+ (runner as any).toAnthropicMessageContent(part);
+ }, /Tool request ref is required/);
+ });
+
+ it('maps tool request with ref into tool_use block', () => {
+ const runner = createRunner();
+ const part: Part = {
+ toolRequest: {
+ ref: 'tool-123',
+ name: 'do_something',
+ input: { foo: 'bar' },
+ },
+ };
+
+ const result = (runner as any).toAnthropicMessageContent(part);
+
+ assert.strictEqual(result.type, 'tool_use');
+ assert.strictEqual(result.id, 'tool-123');
+ assert.strictEqual(result.name, 'do_something');
+ assert.deepStrictEqual(result.input, { foo: 'bar' });
+ });
+
+ it('throws when tool response ref is missing', () => {
+ const runner = createRunner();
+ const part: Part = {
+ toolResponse: {
+ name: 'do_something',
+ output: 'done',
+ },
+ };
+
+ assert.throws(() => {
+ (runner as any).toAnthropicMessageContent(part);
+ }, /Tool response ref is required/);
+ });
+
+ it('maps tool response into tool_result block containing text response', () => {
+ const runner = createRunner();
+ const part: Part = {
+ toolResponse: {
+ name: 'do_something',
+ ref: 'tool-abc',
+ output: 'done',
+ },
+ };
+
+ const result = (runner as any).toAnthropicMessageContent(part);
+
+ assert.strictEqual(result.type, 'tool_result');
+ assert.strictEqual(result.tool_use_id, 'tool-abc');
+ assert.deepStrictEqual(result.content, [{ type: 'text', text: 'done' }]);
+ });
+
+ it('should handle WEBP image data URLs', () => {
+ const runner = createRunner();
+ const part: Part = {
+ media: {
+ contentType: 'image/webp',
+ url: '',
+ },
+ };
+
+ const result = (runner as any).toAnthropicMessageContent(part);
+
+ assert.strictEqual(result.type, 'image');
+ assert.strictEqual(result.source.type, 'base64');
+ assert.strictEqual(result.source.media_type, 'image/webp');
+ assert.strictEqual(result.source.data, 'AAA');
+ });
+
+ it('should prefer data URL content type over media.contentType for WEBP', () => {
+ const runner = createRunner();
+ const part: Part = {
+ media: {
+ // Even if contentType says PNG, data URL says WEBP - should use WEBP
+ contentType: 'image/png',
+ url: '',
+ },
+ };
+
+ const result = (runner as any).toAnthropicMessageContent(part);
+
+ assert.strictEqual(result.type, 'image');
+ assert.strictEqual(result.source.type, 'base64');
+ // Key fix: should use data URL type (webp), not contentType (png)
+ assert.strictEqual(result.source.media_type, 'image/webp');
+ assert.strictEqual(result.source.data, 'AAA');
+ });
+
+ it('should throw helpful error for text/plain in toAnthropicMessageContent', () => {
+ const runner = createRunner();
+ const part: Part = {
+ media: {
+ contentType: 'text/plain',
+ url: 'data:text/plain;base64,AAA',
+ },
+ };
+
+ assert.throws(
+ () => {
+ (runner as any).toAnthropicMessageContent(part);
+ },
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ }
+ );
+ });
+
+ it('should throw helpful error for text/plain with remote URL', () => {
+ const runner = createRunner();
+ const part: Part = {
+ media: {
+ contentType: 'text/plain',
+ url: 'https://example.com/file.txt',
+ },
+ };
+
+ assert.throws(
+ () => {
+ (runner as any).toAnthropicMessageContent(part);
+ },
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ }
+ );
+ });
+
+ it('should throw helpful error for text/plain in tool response', () => {
+ const runner = createRunner();
+ const part: Part = {
+ toolResponse: {
+ ref: 'call_123',
+ name: 'get_file',
+ output: {
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'text/plain',
+ },
+ },
+ };
+
+ assert.throws(
+ () => {
+ (runner as any).toAnthropicToolResponseContent(part);
+ },
+ (error: Error) => {
+ return error.message.includes(
+ 'Text files should be sent as text content'
+ );
+ }
+ );
+ });
+});
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { Anthropic } from '@anthropic-ai/sdk';
+import { mock } from 'node:test';
+
+describe('BetaRunner', () => {
+ it('should map all supported Part shapes to beta content blocks', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+
+ const exposed = runner as any;
+
+ const textPart = exposed.toAnthropicMessageContent({
+ text: 'Hello',
+ } as any);
+ assert.deepStrictEqual(textPart, { type: 'text', text: 'Hello' });
+
+ const pdfPart = exposed.toAnthropicMessageContent({
+ media: {
+ url: 'data:application/pdf;base64,JVBERi0xLjQKJ',
+ contentType: 'application/pdf',
+ },
+ } as any);
+ assert.strictEqual(pdfPart.type, 'document');
+
+ const imagePart = exposed.toAnthropicMessageContent({
+ media: {
+ url: '',
+ contentType: 'image/png',
+ },
+ } as any);
+ assert.strictEqual(imagePart.type, 'image');
+
+ const toolUsePart = exposed.toAnthropicMessageContent({
+ toolRequest: {
+ ref: 'tool1',
+ name: 'get_weather',
+ input: { city: 'NYC' },
+ },
+ } as any);
+ assert.deepStrictEqual(toolUsePart, {
+ type: 'tool_use',
+ id: 'tool1',
+ name: 'get_weather',
+ input: { city: 'NYC' },
+ });
+
+ const toolResultPart = exposed.toAnthropicMessageContent({
+ toolResponse: {
+ ref: 'tool1',
+ name: 'get_weather',
+ output: 'Sunny',
+ },
+ } as any);
+ assert.strictEqual(toolResultPart.type, 'tool_result');
+
+ assert.throws(() => exposed.toAnthropicMessageContent({} as any));
+ });
+
+ it('should convert beta stream events to Genkit Parts', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+
+ const exposed = runner as any;
+ const textPart = exposed.toGenkitPart({
+ type: 'content_block_start',
+ index: 0,
+ content_block: { type: 'text', text: 'hi' },
+ } as any);
+ assert.deepStrictEqual(textPart, { text: 'hi' });
+
+ const serverToolEvent = {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'server_tool_use',
+ id: 'toolu_test',
+ name: 'myTool',
+ input: { foo: 'bar' },
+ server_name: 'srv',
+ },
+ } as any;
+ const toolPart = exposed.toGenkitPart(serverToolEvent);
+ assert.deepStrictEqual(toolPart, {
+ text: '[Anthropic server tool srv/myTool] input: {"foo":"bar"}',
+ custom: {
+ anthropicServerToolUse: {
+ id: 'toolu_test',
+ name: 'srv/myTool',
+ input: { foo: 'bar' },
+ },
+ },
+ });
+
+ const deltaPart = exposed.toGenkitPart({
+ type: 'content_block_delta',
+ index: 0,
+ delta: { type: 'thinking_delta', thinking: 'hmm' },
+ } as any);
+ assert.deepStrictEqual(deltaPart, { reasoning: 'hmm' });
+
+ const ignored = exposed.toGenkitPart({ type: 'message_stop' } as any);
+ assert.strictEqual(ignored, undefined);
+ });
+
+ it('should throw on unsupported mcp tool stream events', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+
+ const exposed = runner as any;
+ assert.throws(
+ () =>
+ exposed.toGenkitPart({
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'mcp_tool_use',
+ id: 'toolu_unsupported',
+ input: {},
+ },
+ }),
+ /server-managed tool block 'mcp_tool_use'/
+ );
+ });
+
+ it('should map beta stop reasons correctly', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+
+ const finishReason = runner['fromBetaStopReason'](
+ 'model_context_window_exceeded'
+ );
+ assert.strictEqual(finishReason, 'length');
+
+ const pauseReason = runner['fromBetaStopReason']('pause_turn');
+ assert.strictEqual(pauseReason, 'stop');
+ });
+
+ it('should execute streaming calls and surface errors', async () => {
+ const streamError = new Error('stream failed');
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ {
+ type: 'content_block_start',
+ index: 0,
+ content_block: { type: 'text', text: 'hi' },
+ } as any,
+ ],
+ streamErrorAfterChunk: 1,
+ streamError,
+ });
+
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+ const sendChunk = mock.fn();
+ await assert.rejects(async () =>
+ runner.run({ messages: [] } as any, {
+ streamingRequested: true,
+ sendChunk,
+ abortSignal: new AbortController().signal,
+ })
+ );
+ assert.strictEqual(sendChunk.mock.calls.length, 1);
+
+ const abortController = new AbortController();
+ abortController.abort();
+ await assert.rejects(async () =>
+ runner.run({ messages: [] } as any, {
+ streamingRequested: true,
+ sendChunk: () => {},
+ abortSignal: abortController.signal,
+ })
+ );
+ });
+
+ it('should throw when tool refs are missing in message content', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+ const exposed = runner as any;
+
+ assert.throws(() =>
+ exposed.toAnthropicMessageContent({
+ toolRequest: {
+ name: 'get_weather',
+ input: {},
+ },
+ } as any)
+ );
+
+ assert.throws(() =>
+ exposed.toAnthropicMessageContent({
+ toolResponse: {
+ name: 'get_weather',
+ output: 'ok',
+ },
+ } as any)
+ );
+
+ assert.throws(() =>
+ exposed.toAnthropicMessageContent({
+ media: {
+ url: 'data:image/png;base64,',
+ contentType: undefined,
+ },
+ } as any)
+ );
+ });
+
+ it('should build request bodies with optional config fields', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ cacheSystemPrompt: true,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [{ text: 'You are helpful.' }],
+ },
+ {
+ role: 'user',
+ content: [{ text: 'Tell me a joke' }],
+ },
+ ],
+ config: {
+ maxOutputTokens: 128,
+ topK: 4,
+ topP: 0.65,
+ temperature: 0.55,
+ stopSequences: ['DONE'],
+ metadata: { user_id: 'beta-user' },
+ tool_choice: { type: 'tool', name: 'get_weather' },
+ thinking: { enabled: true, budgetTokens: 2048 },
+ },
+ tools: [
+ {
+ name: 'get_weather',
+ description: 'Returns the weather',
+ inputSchema: { type: 'object' },
+ },
+ ],
+ } satisfies any;
+
+ const body = runner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+
+ assert.strictEqual(body.model, 'claude-3-5-haiku');
+ assert.ok(Array.isArray(body.system));
+ assert.strictEqual(body.max_tokens, 128);
+ assert.strictEqual(body.top_k, 4);
+ assert.strictEqual(body.top_p, 0.65);
+ assert.strictEqual(body.temperature, 0.55);
+ assert.deepStrictEqual(body.stop_sequences, ['DONE']);
+ assert.deepStrictEqual(body.metadata, { user_id: 'beta-user' });
+ assert.deepStrictEqual(body.tool_choice, {
+ type: 'tool',
+ name: 'get_weather',
+ });
+ assert.strictEqual(body.tools?.length, 1);
+ assert.deepStrictEqual(body.thinking, {
+ type: 'enabled',
+ budget_tokens: 2048,
+ });
+
+ const streamingBody = runner.toAnthropicStreamingRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+ assert.strictEqual(streamingBody.stream, true);
+ assert.ok(Array.isArray(streamingBody.system));
+ assert.deepStrictEqual(streamingBody.thinking, {
+ type: 'enabled',
+ budget_tokens: 2048,
+ });
+
+ const disabledBody = runner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ {
+ messages: [],
+ config: {
+ thinking: { enabled: false },
+ },
+ } satisfies any,
+ false
+ );
+ assert.deepStrictEqual(disabledBody.thinking, { type: 'disabled' });
+ });
+
+ it('should concatenate multiple text parts in system message', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { text: 'Always be concise.' },
+ { text: 'Use proper grammar.' },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ } satisfies any;
+
+ const body = runner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ false
+ );
+
+ assert.strictEqual(
+ body.system,
+ 'You are a helpful assistant.\n\nAlways be concise.\n\nUse proper grammar.'
+ );
+ });
+
+ it('should concatenate multiple text parts in system message with caching', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { text: 'Always be concise.' },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ } satisfies any;
+
+ const body = runner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+
+ assert.ok(Array.isArray(body.system));
+ assert.deepStrictEqual(body.system, [
+ {
+ type: 'text',
+ text: 'You are a helpful assistant.\n\nAlways be concise.',
+ cache_control: { type: 'ephemeral' },
+ },
+ ]);
+ });
+
+ it('should throw error if system message contains media', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ {
+ media: {
+ url: '',
+ contentType: 'image/png',
+ },
+ },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ } satisfies any;
+
+ assert.throws(
+ () => runner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+
+ it('should throw error if system message contains tool requests', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { toolRequest: { name: 'getTool', input: {}, ref: '123' } },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ } satisfies any;
+
+ assert.throws(
+ () => runner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+
+ it('should throw error if system message contains tool responses', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-3-5-haiku',
+ client: mockClient as Anthropic,
+ }) as any;
+
+ const request = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { toolResponse: { name: 'getTool', output: {}, ref: '123' } },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ } satisfies any;
+
+ assert.throws(
+ () => runner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+
+ it('should throw for unsupported mcp tool use blocks', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+ const exposed = runner as any;
+
+ assert.throws(
+ () =>
+ exposed.fromBetaContentBlock({
+ type: 'mcp_tool_use',
+ id: 'toolu_unknown',
+ input: {},
+ }),
+ /server-managed tool block 'mcp_tool_use'/
+ );
+ });
+
+ it('should convert additional beta content block types', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+
+ const thinkingPart = (runner as any).fromBetaContentBlock({
+ type: 'thinking',
+ thinking: 'pondering',
+ signature: 'sig_456',
+ });
+ assert.deepStrictEqual(thinkingPart, {
+ reasoning: 'pondering',
+ custom: { anthropicThinking: { signature: 'sig_456' } },
+ });
+
+ const redactedPart = (runner as any).fromBetaContentBlock({
+ type: 'redacted_thinking',
+ data: '[redacted]',
+ });
+ assert.deepStrictEqual(redactedPart, {
+ custom: { redactedThinking: '[redacted]' },
+ });
+
+ const toolPart = (runner as any).fromBetaContentBlock({
+ type: 'tool_use',
+ id: 'toolu_x',
+ name: 'plainTool',
+ input: { value: 1 },
+ });
+ assert.deepStrictEqual(toolPart, {
+ toolRequest: {
+ ref: 'toolu_x',
+ name: 'plainTool',
+ input: { value: 1 },
+ },
+ });
+
+ const serverToolPart = (runner as any).fromBetaContentBlock({
+ type: 'server_tool_use',
+ id: 'srv_tool_1',
+ name: 'serverTool',
+ input: { arg: 'value' },
+ server_name: 'srv',
+ });
+ assert.deepStrictEqual(serverToolPart, {
+ text: '[Anthropic server tool srv/serverTool] input: {"arg":"value"}',
+ custom: {
+ anthropicServerToolUse: {
+ id: 'srv_tool_1',
+ name: 'srv/serverTool',
+ input: { arg: 'value' },
+ },
+ },
+ });
+
+ const warnMock = mock.method(console, 'warn', () => {});
+ const fallbackPart = (runner as any).fromBetaContentBlock({
+ type: 'mystery',
+ });
+ assert.deepStrictEqual(fallbackPart, { text: '' });
+ assert.strictEqual(warnMock.mock.calls.length, 1);
+ warnMock.mock.restore();
+ });
+
+ it('should map additional stop reasons', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new BetaRunner({
+ name: 'claude-test',
+ client: mockClient as Anthropic,
+ });
+ const exposed = runner as any;
+
+ const refusal = exposed.fromBetaStopReason('refusal');
+ assert.strictEqual(refusal, 'other');
+
+ const unknown = exposed.fromBetaStopReason('something-new');
+ assert.strictEqual(unknown, 'other');
+
+ const nullReason = exposed.fromBetaStopReason(null);
+ assert.strictEqual(nullReason, 'unknown');
+ });
+});
diff --git a/js/plugins/anthropic/tests/execution_test.ts b/js/plugins/anthropic/tests/execution_test.ts
new file mode 100644
index 0000000000..069d2d2dcd
--- /dev/null
+++ b/js/plugins/anthropic/tests/execution_test.ts
@@ -0,0 +1,358 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import type { GenerateRequest, ModelAction } from '@genkit-ai/ai/model';
+import * as assert from 'assert';
+import { describe, mock, test } from 'node:test';
+import { anthropic } from '../src/index.js';
+import { __testClient } from '../src/types.js';
+import {
+ createMockAnthropicClient,
+ createMockAnthropicMessage,
+} from './mocks/anthropic-client.js';
+
+describe('Model Execution Integration Tests', () => {
+ test('should resolve and execute a model via plugin', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Hello from Claude!',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ // Resolve the model action via plugin
+ const modelAction = plugin.resolve('model', 'claude-3-5-haiku-20241022');
+ assert.ok(modelAction, 'Model should be resolved');
+ assert.strictEqual(
+ (modelAction as ModelAction).__action.name,
+ 'anthropic/claude-3-5-haiku-20241022'
+ );
+
+ // Execute the model
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'user',
+ content: [{ text: 'Hi there!' }],
+ },
+ ],
+ };
+
+ const response = await (modelAction as ModelAction)(request, {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ });
+
+ assert.ok(response, 'Response should be returned');
+ assert.ok(response.candidates, 'Response should have candidates');
+ assert.strictEqual(response.candidates.length, 1);
+ assert.strictEqual(response.candidates[0].message.role, 'model');
+ assert.strictEqual(response.candidates[0].message.content.length, 1);
+ assert.strictEqual(
+ response.candidates[0].message.content[0].text,
+ 'Hello from Claude!'
+ );
+
+ // Verify API was called
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ });
+
+ test('should handle multi-turn conversations', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'The capital of France is Paris.',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'user',
+ content: [{ text: 'What is your name?' }],
+ },
+ {
+ role: 'model',
+ content: [{ text: 'I am Claude, an AI assistant.' }],
+ },
+ {
+ role: 'user',
+ content: [{ text: 'What is the capital of France?' }],
+ },
+ ],
+ };
+
+ const response = await modelAction(request, {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ });
+
+ assert.ok(response, 'Response should be returned');
+ assert.strictEqual(
+ response.candidates[0].message.content[0].text,
+ 'The capital of France is Paris.'
+ );
+
+ // Verify API was called with multi-turn conversation
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const apiRequest = createStub.mock.calls[0].arguments[0];
+ assert.strictEqual(apiRequest.messages.length, 3);
+ });
+
+ test('should handle system messages', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Arr matey!',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [{ text: 'You are a pirate. Respond like a pirate.' }],
+ },
+ {
+ role: 'user',
+ content: [{ text: 'Hello!' }],
+ },
+ ],
+ };
+
+ const response = await modelAction(request, {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ });
+
+ assert.ok(response, 'Response should be returned');
+
+ // Verify system message was passed to API
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const apiRequest = createStub.mock.calls[0].arguments[0];
+ assert.ok(apiRequest.system, 'System prompt should be set');
+ assert.strictEqual(
+ apiRequest.system,
+ 'You are a pirate. Respond like a pirate.'
+ );
+ assert.strictEqual(
+ apiRequest.messages.length,
+ 1,
+ 'System message should not be in messages array'
+ );
+ });
+
+ test('should return usage metadata', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Response',
+ usage: {
+ input_tokens: 100,
+ output_tokens: 50,
+ },
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ },
+ {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response.usage, 'Usage should be returned');
+ assert.strictEqual(response.usage?.inputTokens, 100);
+ assert.strictEqual(response.usage?.outputTokens, 50);
+ });
+
+ test('should handle different stop reasons', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'This is a partial response',
+ stopReason: 'max_tokens',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Tell me a story' }] }],
+ },
+ {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response, 'Response should be returned');
+ assert.strictEqual(response.candidates[0].finishReason, 'length');
+ });
+
+ test('should resolve model without anthropic prefix', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Response',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ // Resolve without prefix
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+ assert.ok(modelAction, 'Model should be resolved without prefix');
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hi' }] }],
+ },
+ {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response, 'Response should be returned');
+ });
+
+ test('should resolve model with anthropic prefix', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Response',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ // Resolve with prefix
+ const modelAction = plugin.resolve(
+ 'model',
+ 'anthropic/claude-3-5-haiku-20241022'
+ ) as ModelAction;
+ assert.ok(modelAction, 'Model should be resolved with prefix');
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hi' }] }],
+ },
+ {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response, 'Response should be returned');
+ });
+
+ test('should handle unknown model names', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Response from future model',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ });
+
+ // Resolve unknown model (passes through to API)
+ const modelAction = plugin.resolve(
+ 'model',
+ 'claude-99-experimental-12345'
+ ) as ModelAction;
+ assert.ok(modelAction, 'Unknown model should still be resolved');
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hi' }] }],
+ },
+ {
+ streamingRequested: false,
+ sendChunk: mock.fn(),
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response, 'Response should be returned for unknown model');
+ assert.strictEqual(
+ response.candidates[0].message.content[0].text,
+ 'Response from future model'
+ );
+ });
+});
diff --git a/js/plugins/anthropic/tests/index_test.ts b/js/plugins/anthropic/tests/index_test.ts
new file mode 100644
index 0000000000..62ef06b5fc
--- /dev/null
+++ b/js/plugins/anthropic/tests/index_test.ts
@@ -0,0 +1,286 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import * as assert from 'assert';
+import { genkit, type ActionMetadata } from 'genkit';
+import type { ModelInfo } from 'genkit/model';
+import { describe, it } from 'node:test';
+import anthropic from '../src/index.js';
+import { KNOWN_CLAUDE_MODELS } from '../src/models.js';
+import { PluginOptions, __testClient } from '../src/types.js';
+import { createMockAnthropicClient } from './mocks/anthropic-client.js';
+
+function getModelInfo(
+ metadata: ActionMetadata | undefined
+): ModelInfo | undefined {
+ return metadata?.metadata?.model as ModelInfo | undefined;
+}
+
+describe('Anthropic Plugin', () => {
+ it('should register all supported Claude models', async () => {
+ const mockClient = createMockAnthropicClient();
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ for (const modelName of Object.keys(KNOWN_CLAUDE_MODELS)) {
+ const modelPath = `/model/anthropic/${modelName}`;
+ const expectedBaseName = `anthropic/${modelName}`;
+ const model = await ai.registry.lookupAction(modelPath);
+ assert.ok(model, `${modelName} should be registered at ${modelPath}`);
+ assert.strictEqual(model?.__action.name, expectedBaseName);
+ }
+ });
+
+ it('should throw error when API key is missing', () => {
+ // Save original env var if it exists
+ const originalApiKey = process.env.ANTHROPIC_API_KEY;
+ delete process.env.ANTHROPIC_API_KEY;
+
+ try {
+ assert.throws(() => {
+ anthropic({} as PluginOptions);
+ }, /Please pass in the API key or set the ANTHROPIC_API_KEY environment variable/);
+ } finally {
+ // Restore original env var
+ if (originalApiKey !== undefined) {
+ process.env.ANTHROPIC_API_KEY = originalApiKey;
+ }
+ }
+ });
+
+ it('should use API key from environment variable', () => {
+ // Save original env var if it exists
+ const originalApiKey = process.env.ANTHROPIC_API_KEY;
+ const testApiKey = 'test-api-key-from-env';
+
+ try {
+ // Set test API key
+ process.env.ANTHROPIC_API_KEY = testApiKey;
+
+ // Plugin should initialize without throwing
+ const plugin = anthropic({} as PluginOptions);
+ assert.ok(plugin);
+ assert.strictEqual(plugin.name, 'anthropic');
+ } finally {
+ // Restore original env var
+ if (originalApiKey !== undefined) {
+ process.env.ANTHROPIC_API_KEY = originalApiKey;
+ } else {
+ delete process.env.ANTHROPIC_API_KEY;
+ }
+ }
+ });
+
+ it('should resolve models dynamically via resolve function', async () => {
+ const mockClient = createMockAnthropicClient();
+ const plugin = anthropic({ [__testClient]: mockClient } as PluginOptions);
+
+ assert.ok(plugin.resolve, 'Plugin should have resolve method');
+
+ // Test resolving a valid model
+ const validModel = plugin.resolve!('model', 'anthropic/claude-3-5-haiku');
+ assert.ok(validModel, 'Should resolve valid model');
+ assert.strictEqual(typeof validModel, 'function');
+
+ // Test resolving an unknown model name - should return a model action
+ // (following Google GenAI pattern: accept any model name, let API validate)
+ const unknownModel = plugin.resolve!(
+ 'model',
+ 'anthropic/unknown-model-xyz'
+ );
+ assert.ok(unknownModel, 'Should resolve unknown model name');
+ assert.strictEqual(
+ typeof unknownModel,
+ 'function',
+ 'Should return a model action'
+ );
+
+ // Test resolving with invalid action type (using 'tool' as invalid for this context)
+ const invalidActionType = plugin.resolve!(
+ 'tool',
+ 'anthropic/claude-3-5-haiku'
+ );
+ assert.strictEqual(
+ invalidActionType,
+ undefined,
+ 'Should return undefined for invalid action type'
+ );
+ });
+
+ it('should list available models from API', async () => {
+ const mockClient = createMockAnthropicClient({
+ modelList: [
+ { id: 'claude-3-5-haiku-20241022', display_name: 'Claude 3.5 Haiku' },
+ {
+ id: 'claude-3-5-haiku-latest',
+ display_name: 'Claude 3.5 Haiku Latest',
+ },
+ { id: 'claude-3-5-sonnet-20241022', display_name: 'Claude 3.5 Sonnet' },
+ { id: 'claude-sonnet-4-20250514', display_name: 'Claude 4 Sonnet' },
+ { id: 'claude-new-5-20251212', display_name: 'Claude New 5' },
+ { id: 'claude-experimental-latest' },
+ ],
+ });
+
+ const plugin = anthropic({ [__testClient]: mockClient } as PluginOptions);
+ assert.ok(plugin.list, 'Plugin should have list method');
+
+ const models = await plugin.list!();
+
+ assert.ok(Array.isArray(models), 'Should return an array');
+ assert.ok(models.length > 0, 'Should return at least one model');
+
+ const names = models.map((model) => model.name).sort();
+ // Models are listed with their full IDs from the API (no normalization)
+ assert.ok(
+ names.includes('anthropic/claude-3-5-haiku-20241022'),
+ 'Known model should be listed with full model ID from API'
+ );
+ assert.ok(
+ names.includes('anthropic/claude-3-5-haiku-latest'),
+ 'Latest variant should be listed separately'
+ );
+ assert.ok(
+ names.includes('anthropic/claude-3-5-sonnet-20241022'),
+ 'Unknown Claude 3.5 Sonnet should be listed with full model ID'
+ );
+ assert.ok(
+ names.includes('anthropic/claude-sonnet-4-20250514'),
+ 'Known Claude Sonnet 4 model should be listed with full model ID'
+ );
+ assert.ok(
+ names.includes('anthropic/claude-new-5-20251212'),
+ 'Unknown model IDs should surface as-is'
+ );
+ assert.ok(
+ names.includes('anthropic/claude-experimental-latest'),
+ 'Latest-suffixed unknown models should be surfaced'
+ );
+
+ const haikuMetadata = models.find(
+ (model) => model.name === 'anthropic/claude-3-5-haiku-20241022'
+ );
+ assert.ok(haikuMetadata, 'Haiku metadata should exist');
+ const haikuInfo = getModelInfo(haikuMetadata);
+ assert.ok(haikuInfo, 'Haiku model info should exist');
+
+ const newModelMetadata = models.find(
+ (model) => model.name === 'anthropic/claude-new-5-20251212'
+ );
+ assert.ok(newModelMetadata, 'New model metadata should exist');
+
+ const experimentalMetadata = models.find(
+ (model) => model.name === 'anthropic/claude-experimental-latest'
+ );
+ assert.ok(experimentalMetadata, 'Experimental model metadata should exist');
+
+ // Verify mock was called
+ const listStub = mockClient.models.list as any;
+ assert.strictEqual(
+ listStub.mock.calls.length,
+ 1,
+ 'models.list should be called once'
+ );
+ });
+
+ it('should cache list results on subsequent calls?', async () => {
+ const mockClient = createMockAnthropicClient({
+ modelList: [
+ { id: 'claude-3-5-haiku-20241022', display_name: 'Claude 3.5 Haiku' },
+ ],
+ });
+
+ const plugin = anthropic({ [__testClient]: mockClient } as PluginOptions);
+ assert.ok(plugin.list, 'Plugin should have list method');
+
+ // First call
+ const firstResult = await plugin.list!();
+ assert.ok(firstResult, 'First call should return results');
+
+ // Second call
+ const secondResult = await plugin.list!();
+ assert.ok(secondResult, 'Second call should return results');
+
+ // Verify both results are the same (reference equality for cache)
+ assert.strictEqual(
+ firstResult,
+ secondResult,
+ 'Results should be cached (same reference)'
+ );
+
+ // Verify models.list was only called once due to caching
+ const listStub = mockClient.models.list as any;
+ assert.strictEqual(
+ listStub.mock.calls.length,
+ 1,
+ 'models.list should only be called once due to caching'
+ );
+ });
+});
+
+describe('Anthropic resolve helpers', () => {
+ it('should resolve model names without anthropic/ prefix', () => {
+ const mockClient = createMockAnthropicClient();
+ const plugin = anthropic({ [__testClient]: mockClient } as PluginOptions);
+
+ const action = plugin.resolve?.('model', 'claude-3-5-haiku');
+ assert.ok(action, 'Should resolve model without prefix');
+ assert.strictEqual(typeof action, 'function');
+ });
+
+ it('anthropic.model should return model reference with config', () => {
+ const reference = anthropic.model('claude-3-5-haiku', {
+ temperature: 0.25,
+ });
+
+ const referenceAny = reference as any;
+ assert.ok(referenceAny, 'Model reference should be created');
+ assert.ok(referenceAny.name.includes('claude-3-5-haiku'));
+ assert.strictEqual(referenceAny.config?.temperature, 0.25);
+ });
+
+ it('should apply system prompt caching when cacheSystemPrompt is true', async () => {
+ const mockClient = createMockAnthropicClient();
+ const plugin = anthropic({
+ cacheSystemPrompt: true,
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const action = plugin.resolve?.('model', 'anthropic/claude-3-5-haiku');
+ assert.ok(action, 'Action should be resolved');
+
+ const abortSignal = new AbortController().signal;
+ await (action as any)(
+ {
+ messages: [
+ {
+ role: 'system',
+ content: [{ text: 'You are helpful.' }],
+ },
+ ],
+ },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const requestBody = createStub.mock.calls[0].arguments[0];
+ assert.ok(Array.isArray(requestBody.system));
+ assert.strictEqual(requestBody.system[0].cache_control.type, 'ephemeral');
+ });
+});
diff --git a/js/plugins/anthropic/tests/integration_test.ts b/js/plugins/anthropic/tests/integration_test.ts
new file mode 100644
index 0000000000..209a455870
--- /dev/null
+++ b/js/plugins/anthropic/tests/integration_test.ts
@@ -0,0 +1,542 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import * as assert from 'assert';
+import { genkit, z } from 'genkit';
+import { describe, it } from 'node:test';
+import { anthropic } from '../src/index.js';
+import { __testClient } from '../src/types.js';
+import {
+ createMockAnthropicClient,
+ createMockAnthropicMessage,
+ mockContentBlockStart,
+ mockMessageWithContent,
+ mockMessageWithToolUse,
+ mockTextChunk,
+} from './mocks/anthropic-client.js';
+
+import { PluginOptions } from '../src/types.js';
+
+describe('Anthropic Integration', () => {
+ it('should successfully generate a response', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Hello',
+ });
+
+ assert.strictEqual(result.text, 'Hello! How can I help you today?');
+ });
+
+ it('should handle tool calling workflow (call tool, receive result, generate final response)', async () => {
+ const mockClient = createMockAnthropicClient({
+ sequentialResponses: [
+ // First response: tool use request
+ mockMessageWithToolUse('get_weather', { city: 'NYC' }),
+ // Second response: final text after tool result
+ createMockAnthropicMessage({
+ text: 'The weather in NYC is sunny, 72°F',
+ }),
+ ],
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ // Define the tool
+ ai.defineTool(
+ {
+ name: 'get_weather',
+ description: 'Get the weather for a city',
+ inputSchema: z.object({
+ city: z.string(),
+ }),
+ },
+ async (input: { city: string }) => {
+ return `The weather in ${input.city} is sunny, 72°F`;
+ }
+ );
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'What is the weather in NYC?',
+ tools: ['get_weather'],
+ });
+
+ assert.ok(
+ result.text.includes('NYC') ||
+ result.text.includes('sunny') ||
+ result.text.includes('72')
+ );
+ });
+
+ it('should handle multi-turn conversations', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ // First turn
+ const response1 = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'My name is Alice',
+ });
+
+ // Second turn with conversation history
+ const response2 = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: "What's my name?",
+ messages: response1.messages,
+ });
+
+ // Verify conversation history is maintained
+ assert.ok(
+ response2.messages.length >= 2,
+ 'Should have conversation history'
+ );
+ assert.strictEqual(response2.messages[0].role, 'user');
+ assert.ok(
+ response2.messages[0].content[0].text?.includes('Alice') ||
+ response2.messages[0].content[0].text?.includes('name')
+ );
+ });
+
+ it('should stream responses with streaming callback', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockContentBlockStart('Hello'),
+ mockTextChunk(' world'),
+ mockTextChunk('!'),
+ ],
+ messageResponse: {
+ content: [{ type: 'text', text: 'Hello world!', citations: null }],
+ usage: {
+ input_tokens: 5,
+ output_tokens: 15,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ cache_creation: null,
+ server_tool_use: null,
+ service_tier: null,
+ },
+ },
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const chunks: any[] = [];
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Say hello world',
+ streamingCallback: (chunk) => {
+ chunks.push(chunk);
+ },
+ });
+
+ assert.ok(chunks.length > 0, 'Should have received streaming chunks');
+ assert.ok(result.text, 'Should have final response text');
+ });
+
+ it('should handle media/image inputs', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ messages: [
+ {
+ role: 'user',
+ content: [
+ { text: 'Describe this image:' },
+ {
+ media: {
+ url: '',
+ contentType: 'image/png',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ assert.ok(result.text, 'Should generate response for image input');
+ });
+
+ it('should handle WEBP image inputs', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ messages: [
+ {
+ role: 'user',
+ content: [
+ { text: 'Describe this image:' },
+ {
+ media: {
+ url: '',
+ contentType: 'image/webp',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ assert.ok(result.text, 'Should generate response for WEBP image input');
+ // Verify the request was made with correct media_type
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const requestBody = createStub.mock.calls[0].arguments[0];
+ const imageContent = requestBody.messages[0].content.find(
+ (c: any) => c.type === 'image'
+ );
+ assert.ok(imageContent, 'Should have image content in request');
+ assert.strictEqual(
+ imageContent.source.media_type,
+ 'image/webp',
+ 'Should use WEBP media type from data URL'
+ );
+ });
+
+ it('should handle WEBP image with mismatched contentType (prefers data URL)', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ media: {
+ // Data URL says WEBP, but contentType says PNG - should use WEBP
+ url: '',
+ contentType: 'image/png',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ assert.ok(result.text, 'Should generate response for WEBP image input');
+ // Verify the request was made with WEBP (from data URL), not PNG (from contentType)
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const requestBody = createStub.mock.calls[0].arguments[0];
+ const imageContent = requestBody.messages[0].content.find(
+ (c: any) => c.type === 'image'
+ );
+ assert.ok(imageContent, 'Should have image content in request');
+ assert.strictEqual(
+ imageContent.source.media_type,
+ 'image/webp',
+ 'Should prefer data URL content type (webp) over contentType (png)'
+ );
+ });
+
+ it('should throw helpful error for text/plain media', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ await assert.rejects(
+ async () => {
+ await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ media: {
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'text/plain',
+ },
+ },
+ ],
+ },
+ ],
+ });
+ },
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ },
+ 'Should throw helpful error for text/plain media'
+ );
+ });
+
+ it('should forward thinking config and surface reasoning in responses', async () => {
+ const thinkingContent = [
+ {
+ type: 'thinking' as const,
+ thinking: 'Let me analyze the problem carefully.',
+ signature: 'sig_reasoning_123',
+ },
+ {
+ type: 'text' as const,
+ text: 'The answer is 42.',
+ citations: null,
+ },
+ ];
+ const mockClient = createMockAnthropicClient({
+ messageResponse: mockMessageWithContent(thinkingContent),
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const thinkingConfig = { enabled: true, budgetTokens: 2048 };
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'What is the meaning of life?',
+ config: { thinking: thinkingConfig },
+ });
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const requestBody = createStub.mock.calls[0].arguments[0];
+ assert.deepStrictEqual(requestBody.thinking, {
+ type: 'enabled',
+ budget_tokens: 2048,
+ });
+
+ assert.strictEqual(
+ result.reasoning,
+ 'Let me analyze the problem carefully.'
+ );
+ const assistantMessage = result.messages[result.messages.length - 1];
+ const reasoningPart = assistantMessage.content.find(
+ (part) => part.reasoning
+ );
+ assert.ok(reasoningPart, 'Expected reasoning part in assistant message');
+ assert.strictEqual(
+ reasoningPart?.custom?.anthropicThinking?.signature,
+ 'sig_reasoning_123'
+ );
+ });
+
+ it('should propagate API errors correctly', async () => {
+ const apiError = new Error('API Error: 401 Unauthorized');
+ const mockClient = createMockAnthropicClient({
+ shouldError: apiError,
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ await assert.rejects(
+ async () => {
+ await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Hello',
+ });
+ },
+ (error: Error) => {
+ assert.strictEqual(error.message, 'API Error: 401 Unauthorized');
+ return true;
+ }
+ );
+ });
+
+ it('should respect abort signals for cancellation', async () => {
+ // Note: Detailed abort signal handling is tested in converters_test.ts
+ // This test verifies that errors (including abort errors) are properly propagated at the integration layer
+ const mockClient = createMockAnthropicClient({
+ shouldError: new Error('AbortError'),
+ });
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ await assert.rejects(
+ async () => {
+ await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Hello',
+ });
+ },
+ (error: Error) => {
+ // Should propagate the error
+ assert.ok(
+ error.message.includes('AbortError'),
+ 'Should propagate errors'
+ );
+ return true;
+ }
+ );
+ });
+
+ it('should track token usage in responses', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ usage: {
+ input_tokens: 25,
+ output_tokens: 50,
+ cache_creation_input_tokens: 5,
+ cache_read_input_tokens: 10,
+ cache_creation: null,
+ server_tool_use: null,
+ service_tier: null,
+ },
+ },
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Hello',
+ });
+
+ assert.ok(result.usage, 'Should have usage information');
+ assert.strictEqual(result.usage.inputTokens, 25);
+ assert.strictEqual(result.usage.outputTokens, 50);
+ });
+
+ it('should route requests through beta surface when plugin default is beta', async () => {
+ const mockClient = createMockAnthropicClient();
+ const ai = genkit({
+ plugins: [
+ anthropic({
+ apiVersion: 'beta',
+ [__testClient]: mockClient,
+ } as PluginOptions),
+ ],
+ });
+
+ await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Hello',
+ });
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(
+ betaCreateStub.mock.calls.length,
+ 1,
+ 'Beta API should be used'
+ );
+ const regularCreateStub = mockClient.messages.create as any;
+ assert.strictEqual(
+ regularCreateStub.mock.calls.length,
+ 0,
+ 'Stable API should not be used'
+ );
+ });
+
+ it('should stream thinking deltas as reasoning chunks', async () => {
+ const thinkingConfig = { enabled: true, budgetTokens: 3072 };
+ const streamChunks = [
+ {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'thinking',
+ thinking: '',
+ signature: 'sig_stream_123',
+ },
+ } as any,
+ {
+ type: 'content_block_delta',
+ index: 0,
+ delta: {
+ type: 'thinking_delta',
+ thinking: 'Analyzing intermediate steps.',
+ },
+ } as any,
+ {
+ type: 'content_block_start',
+ index: 1,
+ content_block: {
+ type: 'text',
+ text: '',
+ },
+ } as any,
+ mockTextChunk('Final streamed response.'),
+ ];
+ const finalMessage = mockMessageWithContent([
+ {
+ type: 'thinking',
+ thinking: 'Analyzing intermediate steps.',
+ signature: 'sig_stream_123',
+ },
+ {
+ type: 'text',
+ text: 'Final streamed response.',
+ citations: null,
+ },
+ ]);
+ const mockClient = createMockAnthropicClient({
+ streamChunks,
+ messageResponse: finalMessage,
+ });
+
+ const ai = genkit({
+ plugins: [anthropic({ [__testClient]: mockClient } as PluginOptions)],
+ });
+
+ const chunks: any[] = [];
+ const result = await ai.generate({
+ model: 'anthropic/claude-3-5-haiku',
+ prompt: 'Explain how you reason.',
+ streamingCallback: (chunk) => chunks.push(chunk),
+ config: { thinking: thinkingConfig },
+ });
+
+ const streamStub = mockClient.messages.stream as any;
+ assert.strictEqual(streamStub.mock.calls.length, 1);
+ const streamRequest = streamStub.mock.calls[0].arguments[0];
+ assert.deepStrictEqual(streamRequest.thinking, {
+ type: 'enabled',
+ budget_tokens: 3072,
+ });
+
+ const hasReasoningChunk = chunks.some((chunk) =>
+ (chunk.content || []).some(
+ (part: any) => part.reasoning === 'Analyzing intermediate steps.'
+ )
+ );
+ assert.ok(
+ hasReasoningChunk,
+ 'Expected reasoning chunk in streaming callback'
+ );
+ assert.strictEqual(result.reasoning, 'Analyzing intermediate steps.');
+ });
+});
diff --git a/js/plugins/anthropic/tests/mocks/anthropic-client.ts b/js/plugins/anthropic/tests/mocks/anthropic-client.ts
new file mode 100644
index 0000000000..321df8f24f
--- /dev/null
+++ b/js/plugins/anthropic/tests/mocks/anthropic-client.ts
@@ -0,0 +1,389 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import type Anthropic from '@anthropic-ai/sdk';
+import type {
+ BetaMessage,
+ BetaRawMessageStreamEvent,
+} from '@anthropic-ai/sdk/resources/beta/messages.mjs';
+import type {
+ Message,
+ MessageStreamEvent,
+} from '@anthropic-ai/sdk/resources/messages.mjs';
+import { mock } from 'node:test';
+
+export interface MockAnthropicClientOptions {
+ messageResponse?: Partial;
+ sequentialResponses?: Partial[]; // For tool calling - multiple responses
+ streamChunks?: MessageStreamEvent[];
+ modelList?: Array<{ id: string; display_name?: string }>;
+ shouldError?: Error;
+ streamErrorAfterChunk?: number; // Throw error after this many chunks
+ streamError?: Error; // Error to throw during streaming
+ abortSignal?: AbortSignal; // Abort signal to check
+}
+
+/**
+ * Creates a mock Anthropic client for testing
+ */
+export function createMockAnthropicClient(
+ options: MockAnthropicClientOptions = {}
+): Anthropic {
+ const messageResponse = {
+ ...mockDefaultMessage(),
+ ...options.messageResponse,
+ };
+ const betaMessageResponse = toBetaMessage(messageResponse);
+
+ // Support sequential responses for tool calling workflows
+ let callCount = 0;
+ const createStub = options.shouldError
+ ? mock.fn(async () => {
+ throw options.shouldError;
+ })
+ : options.sequentialResponses
+ ? mock.fn(async () => {
+ const response =
+ options.sequentialResponses![callCount] || messageResponse;
+ callCount++;
+ return {
+ ...mockDefaultMessage(),
+ ...response,
+ };
+ })
+ : mock.fn(async () => messageResponse);
+
+ let betaCallCount = 0;
+ const betaCreateStub = options.shouldError
+ ? mock.fn(async () => {
+ throw options.shouldError;
+ })
+ : options.sequentialResponses
+ ? mock.fn(async () => {
+ const response =
+ options.sequentialResponses![betaCallCount] || messageResponse;
+ betaCallCount++;
+ return toBetaMessage({
+ ...mockDefaultMessage(),
+ ...response,
+ });
+ })
+ : mock.fn(async () => betaMessageResponse);
+
+ const streamStub = options.shouldError
+ ? mock.fn(() => {
+ throw options.shouldError;
+ })
+ : mock.fn((_body: any, opts?: { signal?: AbortSignal }) => {
+ // Check abort signal before starting stream
+ if (opts?.signal?.aborted) {
+ throw new Error('AbortError');
+ }
+ return createMockStream(
+ options.streamChunks || [],
+ messageResponse as Message,
+ options.streamErrorAfterChunk,
+ options.streamError,
+ opts?.signal
+ );
+ });
+
+ const betaStreamStub = options.shouldError
+ ? mock.fn(() => {
+ throw options.shouldError;
+ })
+ : mock.fn((_body: any, opts?: { signal?: AbortSignal }) => {
+ if (opts?.signal?.aborted) {
+ throw new Error('AbortError');
+ }
+ const betaChunks = (options.streamChunks || []).map((chunk) =>
+ toBetaStreamEvent(chunk)
+ );
+ return createMockStream(
+ betaChunks,
+ toBetaMessage(messageResponse),
+ options.streamErrorAfterChunk,
+ options.streamError,
+ opts?.signal
+ );
+ });
+
+ const listStub = options.shouldError
+ ? mock.fn(async () => {
+ throw options.shouldError;
+ })
+ : mock.fn(async () => ({
+ data: options.modelList || mockDefaultModels(),
+ }));
+
+ return {
+ messages: {
+ create: createStub,
+ stream: streamStub,
+ },
+ models: {
+ list: listStub,
+ },
+ beta: {
+ messages: {
+ create: betaCreateStub,
+ stream: betaStreamStub,
+ },
+ },
+ } as unknown as Anthropic;
+}
+
+/**
+ * Creates a mock async iterable stream for streaming responses
+ */
+function createMockStream(
+ chunks: TEventType[],
+ finalMsg: TMessageType,
+ errorAfterChunk?: number,
+ streamError?: Error,
+ abortSignal?: AbortSignal
+) {
+ let index = 0;
+ return {
+ [Symbol.asyncIterator]() {
+ return {
+ async next() {
+ // Check abort signal
+ if (abortSignal?.aborted) {
+ const error = new Error('AbortError');
+ error.name = 'AbortError';
+ throw error;
+ }
+
+ // Check if we should throw an error after this chunk
+ if (
+ errorAfterChunk !== undefined &&
+ streamError &&
+ index >= errorAfterChunk
+ ) {
+ throw streamError;
+ }
+
+ if (index < chunks.length) {
+ return { value: chunks[index++] as TEventType, done: false };
+ }
+ return { value: undefined as unknown as TEventType, done: true };
+ },
+ };
+ },
+ async finalMessage() {
+ // Check abort signal before returning final message
+ if (abortSignal?.aborted) {
+ const error = new Error('AbortError');
+ error.name = 'AbortError';
+ throw error;
+ }
+ return finalMsg as TMessageType;
+ },
+ };
+}
+
+export interface CreateMockAnthropicMessageOptions {
+ id?: string;
+ text?: string;
+ toolUse?: {
+ id?: string;
+ name: string;
+ input: any;
+ };
+ stopReason?: Message['stop_reason'];
+ usage?: Partial;
+}
+
+/**
+ * Creates a customizable mock Anthropic Message response
+ *
+ * @example
+ * // Simple text response
+ * createMockAnthropicMessage({ text: 'Hi there!' })
+ *
+ * // Tool use response
+ * createMockAnthropicMessage({
+ * toolUse: { name: 'get_weather', input: { city: 'NYC' } }
+ * })
+ *
+ * // Custom usage
+ * createMockAnthropicMessage({ usage: { input_tokens: 5, output_tokens: 15 } })
+ */
+export function createMockAnthropicMessage(
+ options: CreateMockAnthropicMessageOptions = {}
+): Message {
+ const content: Message['content'] = [];
+
+ if (options.toolUse) {
+ content.push({
+ type: 'tool_use',
+ id: options.toolUse.id || 'toolu_test123',
+ name: options.toolUse.name,
+ input: options.toolUse.input,
+ });
+ } else {
+ content.push({
+ type: 'text',
+ text: options.text || 'Hello! How can I help you today?',
+ citations: null,
+ });
+ }
+
+ const usage: Message['usage'] = {
+ cache_creation: null,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ input_tokens: 10,
+ output_tokens: 20,
+ server_tool_use: null,
+ service_tier: null,
+ ...(options.usage ?? {}),
+ };
+
+ return {
+ id: options.id || 'msg_test123',
+ type: 'message',
+ role: 'assistant',
+ model: 'claude-3-5-sonnet-20241022',
+ content,
+ stop_reason:
+ options.stopReason || (options.toolUse ? 'tool_use' : 'end_turn'),
+ stop_sequence: null,
+ usage,
+ };
+}
+
+/**
+ * Creates a default mock Message response
+ */
+export function mockDefaultMessage(): Message {
+ return createMockAnthropicMessage();
+}
+
+/**
+ * Creates a mock text content block chunk event
+ */
+export function mockTextChunk(text: string): MessageStreamEvent {
+ return {
+ type: 'content_block_delta',
+ index: 0,
+ delta: {
+ type: 'text_delta',
+ text,
+ },
+ } as MessageStreamEvent;
+}
+
+/**
+ * Creates a mock content block start event with text
+ */
+export function mockContentBlockStart(text: string): MessageStreamEvent {
+ return {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'text',
+ text,
+ },
+ } as MessageStreamEvent;
+}
+
+/**
+ * Creates a mock tool use content block
+ */
+export function mockToolUseChunk(
+ id: string,
+ name: string,
+ input: any
+): MessageStreamEvent {
+ return {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'tool_use',
+ id,
+ name,
+ input,
+ },
+ } as MessageStreamEvent;
+}
+
+/**
+ * Creates a default list of mock models
+ */
+export function mockDefaultModels() {
+ return [
+ { id: 'claude-3-5-sonnet-20241022', display_name: 'Claude 3.5 Sonnet' },
+ { id: 'claude-3-5-haiku-20241022', display_name: 'Claude 3.5 Haiku' },
+ { id: 'claude-3-opus-20240229', display_name: 'Claude 3 Opus' },
+ ];
+}
+
+/**
+ * Creates a mock Message with tool use
+ */
+export function mockMessageWithToolUse(
+ toolName: string,
+ toolInput: any
+): Partial {
+ return {
+ content: [
+ {
+ type: 'tool_use',
+ id: 'toolu_test123',
+ name: toolName,
+ input: toolInput,
+ },
+ ],
+ stop_reason: 'tool_use',
+ };
+}
+
+/**
+ * Creates a mock Message with custom content
+ */
+export function mockMessageWithContent(
+ content: Message['content']
+): Partial {
+ return {
+ content,
+ stop_reason: 'end_turn',
+ };
+}
+
+function toBetaMessage(message: Message): BetaMessage {
+ return {
+ ...message,
+ container: null,
+ context_management: null,
+ usage: {
+ cache_creation: message.usage.cache_creation,
+ cache_creation_input_tokens: message.usage.cache_creation_input_tokens,
+ cache_read_input_tokens: message.usage.cache_read_input_tokens,
+ input_tokens: message.usage.input_tokens,
+ output_tokens: message.usage.output_tokens,
+ server_tool_use: message.usage.server_tool_use as any,
+ service_tier: message.usage.service_tier,
+ },
+ };
+}
+
+function toBetaStreamEvent(
+ event: MessageStreamEvent
+): BetaRawMessageStreamEvent {
+ return event as unknown as BetaRawMessageStreamEvent;
+}
diff --git a/js/plugins/anthropic/tests/stable_runner_test.ts b/js/plugins/anthropic/tests/stable_runner_test.ts
new file mode 100644
index 0000000000..9b60084b3e
--- /dev/null
+++ b/js/plugins/anthropic/tests/stable_runner_test.ts
@@ -0,0 +1,2340 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import type Anthropic from '@anthropic-ai/sdk';
+import type {
+ Message,
+ MessageCreateParams,
+ MessageParam,
+ MessageStreamEvent,
+} from '@anthropic-ai/sdk/resources/messages.mjs';
+import * as assert from 'assert';
+import type {
+ GenerateRequest,
+ GenerateResponseData,
+ MessageData,
+ Part,
+ Role,
+} from 'genkit';
+import type { CandidateData, ToolDefinition } from 'genkit/model';
+import { describe, it, mock } from 'node:test';
+
+import { claudeModel, claudeRunner } from '../src/models.js';
+import { Runner } from '../src/runner/stable.js';
+import { AnthropicConfigSchema } from '../src/types.js';
+import {
+ createMockAnthropicClient,
+ mockContentBlockStart,
+ mockTextChunk,
+} from './mocks/anthropic-client.js';
+
+// Test helper: Create a Runner instance for testing converter methods
+// Type interface to access protected methods in tests
+type RunnerProtectedMethods = {
+ toAnthropicRole: (
+ role: Role,
+ toolMessageType?: 'tool_use' | 'tool_result'
+ ) => 'user' | 'assistant';
+ toAnthropicToolResponseContent: (part: Part) => any;
+ toAnthropicMessageContent: (part: Part) => any;
+ toAnthropicMessages: (messages: MessageData[]) => {
+ system?: string;
+ messages: any[];
+ };
+ toAnthropicTool: (tool: ToolDefinition) => any;
+ toAnthropicRequestBody: (
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ) => any;
+ toAnthropicStreamingRequestBody: (
+ modelName: string,
+ request: GenerateRequest,
+ cacheSystemPrompt?: boolean
+ ) => any;
+ fromAnthropicContentBlockChunk: (
+ event: MessageStreamEvent
+ ) => Part | undefined;
+ fromAnthropicStopReason: (reason: Message['stop_reason']) => any;
+ fromAnthropicResponse: (message: Message) => GenerateResponseData;
+};
+
+const mockClient = createMockAnthropicClient();
+const testRunner = new Runner({
+ name: 'test-model',
+ client: mockClient,
+}) as Runner & RunnerProtectedMethods;
+
+const createUsage = (
+ overrides: Partial = {}
+): Message['usage'] => ({
+ cache_creation: null,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ input_tokens: 0,
+ output_tokens: 0,
+ server_tool_use: null,
+ service_tier: null,
+ ...overrides,
+});
+
+describe('toAnthropicRole', () => {
+ const testCases: {
+ genkitRole: Role;
+ toolMessageType?: 'tool_use' | 'tool_result';
+ expectedAnthropicRole: MessageParam['role'];
+ }[] = [
+ {
+ genkitRole: 'user',
+ expectedAnthropicRole: 'user',
+ },
+ {
+ genkitRole: 'model',
+ expectedAnthropicRole: 'assistant',
+ },
+ {
+ genkitRole: 'tool',
+ toolMessageType: 'tool_use',
+ expectedAnthropicRole: 'assistant',
+ },
+ {
+ genkitRole: 'tool',
+ toolMessageType: 'tool_result',
+ expectedAnthropicRole: 'user',
+ },
+ ];
+
+ for (const test of testCases) {
+ it(`should map Genkit "${test.genkitRole}" role to Anthropic "${test.expectedAnthropicRole}" role${
+ test.toolMessageType
+ ? ` when toolMessageType is "${test.toolMessageType}"`
+ : ''
+ }`, () => {
+ const actualOutput = testRunner.toAnthropicRole(
+ test.genkitRole,
+ test.toolMessageType
+ );
+ assert.strictEqual(actualOutput, test.expectedAnthropicRole);
+ });
+ }
+
+ it('should throw an error for unknown roles', () => {
+ assert.throws(
+ () => testRunner.toAnthropicRole('unknown' as Role),
+ /Unsupported genkit role: unknown/
+ );
+ });
+});
+
+describe('toAnthropicToolResponseContent', () => {
+ it('should not throw for parts without toolResponse', () => {
+ // toAnthropicToolResponseContent expects part.toolResponse to exist
+ // but will just return stringified undefined/empty object if not
+ const part: Part = { data: 'hi' } as Part;
+ const result = testRunner.toAnthropicToolResponseContent(part);
+ assert.ok(result);
+ assert.strictEqual(result.type, 'text');
+ });
+});
+
+describe('toAnthropicMessageContent', () => {
+ it('should throw if a media part contains invalid media', () => {
+ assert.throws(
+ () =>
+ testRunner.toAnthropicMessageContent({
+ media: {
+ url: '',
+ },
+ }),
+ /Media url is required but was not provided/
+ );
+ });
+
+ it('should throw if the provided part is invalid', () => {
+ assert.throws(
+ () => testRunner.toAnthropicMessageContent({ fake: 'part' } as Part),
+ /Unsupported genkit part fields encountered for current message role: {"fake":"part"}/
+ );
+ });
+
+ it('should treat remote URLs without explicit content type as image URLs', () => {
+ const result = testRunner.toAnthropicMessageContent({
+ media: {
+ url: 'https://example.com/image.png',
+ },
+ });
+
+ assert.deepStrictEqual(result, {
+ type: 'image',
+ source: {
+ type: 'url',
+ url: 'https://example.com/image.png',
+ },
+ });
+ });
+
+ it('should handle PDF with base64 data URL correctly', () => {
+ const result = testRunner.toAnthropicMessageContent({
+ media: {
+ url: 'data:application/pdf;base64,JVBERi0xLjQKJ',
+ contentType: 'application/pdf',
+ },
+ });
+
+ assert.deepStrictEqual(result, {
+ type: 'document',
+ source: {
+ type: 'base64',
+ media_type: 'application/pdf',
+ data: 'JVBERi0xLjQKJ',
+ },
+ });
+ });
+
+ it('should handle PDF with HTTP/HTTPS URL correctly', () => {
+ const result = testRunner.toAnthropicMessageContent({
+ media: {
+ url: 'https://example.com/document.pdf',
+ contentType: 'application/pdf',
+ },
+ });
+
+ assert.deepStrictEqual(result, {
+ type: 'document',
+ source: {
+ type: 'url',
+ url: 'https://example.com/document.pdf',
+ },
+ });
+ });
+});
+
+describe('toAnthropicMessages', () => {
+ const testCases: {
+ should: string;
+ inputMessages: MessageData[];
+ expectedOutput: {
+ messages: MessageParam[];
+ system?: string;
+ };
+ }[] = [
+ {
+ should: 'should transform tool request content correctly',
+ inputMessages: [
+ {
+ role: 'model',
+ content: [
+ {
+ toolRequest: {
+ ref: 'toolu_01A09q90qw90lq917835lq9',
+ name: 'tellAFunnyJoke',
+ input: { topic: 'bob' },
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ role: 'assistant',
+ content: [
+ {
+ type: 'tool_use',
+ id: 'toolu_01A09q90qw90lq917835lq9',
+ name: 'tellAFunnyJoke',
+ input: { topic: 'bob' },
+ },
+ ],
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform tool response text content correctly',
+ inputMessages: [
+ {
+ role: 'tool',
+ content: [
+ {
+ toolResponse: {
+ ref: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ name: 'tellAFunnyJoke',
+ output: 'Why did the bob cross the road?',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ type: 'tool_result',
+ tool_use_id: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ content: [
+ {
+ type: 'text',
+ text: 'Why did the bob cross the road?',
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform tool response media content correctly',
+ inputMessages: [
+ {
+ role: 'tool',
+ content: [
+ {
+ toolResponse: {
+ ref: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ name: 'tellAFunnyJoke',
+ output: {
+ url: '',
+ contentType: 'image/gif',
+ },
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ type: 'tool_result',
+ tool_use_id: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ content: [
+ {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data: 'R0lGODlhAQABAAAAACw=',
+ media_type: 'image/gif',
+ },
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should:
+ 'should transform tool response base64 image url content correctly',
+ inputMessages: [
+ {
+ role: 'tool',
+ content: [
+ {
+ toolResponse: {
+ ref: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ name: 'tellAFunnyJoke',
+ output: '',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ type: 'tool_result',
+ tool_use_id: 'call_SVDpFV2l2fW88QRFtv85FWwM',
+ content: [
+ {
+ type: 'image',
+ source: {
+ type: 'base64',
+ data: 'R0lGODlhAQABAAAAACw=',
+ media_type: 'image/gif',
+ },
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform text content correctly',
+ inputMessages: [
+ { role: 'user', content: [{ text: 'hi' }] },
+ { role: 'model', content: [{ text: 'how can I help you?' }] },
+ { role: 'user', content: [{ text: 'I am testing' }] },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ text: 'hi',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'user',
+ },
+ {
+ content: [
+ {
+ text: 'how can I help you?',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'assistant',
+ },
+ {
+ content: [
+ {
+ text: 'I am testing',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform initial system prompt correctly',
+ inputMessages: [
+ { role: 'system', content: [{ text: 'You are an helpful assistant' }] },
+ { role: 'user', content: [{ text: 'hi' }] },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ text: 'hi',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: 'You are an helpful assistant',
+ },
+ },
+ {
+ should: 'should transform multi-modal (text + media) content correctly',
+ inputMessages: [
+ {
+ role: 'user',
+ content: [
+ { text: 'describe the following image:' },
+ {
+ media: {
+ url: '',
+ contentType: 'image/gif',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ text: 'describe the following image:',
+ type: 'text',
+ citations: null,
+ },
+ {
+ source: {
+ type: 'base64',
+ data: 'R0lGODlhAQABAAAAACw=',
+ media_type: 'image/gif',
+ },
+ type: 'image',
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform PDF with base64 data URL correctly',
+ inputMessages: [
+ {
+ role: 'user',
+ content: [
+ {
+ media: {
+ url: 'data:application/pdf;base64,JVBERi0xLjQKJ',
+ contentType: 'application/pdf',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ type: 'document',
+ source: {
+ type: 'base64',
+ media_type: 'application/pdf',
+ data: 'JVBERi0xLjQKJ',
+ },
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform PDF with HTTP/HTTPS URL correctly',
+ inputMessages: [
+ {
+ role: 'user',
+ content: [
+ {
+ media: {
+ url: 'https://example.com/document.pdf',
+ contentType: 'application/pdf',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ type: 'document',
+ source: {
+ type: 'url',
+ url: 'https://example.com/document.pdf',
+ },
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: undefined,
+ },
+ },
+ {
+ should: 'should transform PDF alongside text and images correctly',
+ inputMessages: [
+ {
+ role: 'user',
+ content: [
+ { text: 'Analyze this PDF and image:' },
+ {
+ media: {
+ url: 'data:application/pdf;base64,JVBERi0xLjQKJ',
+ contentType: 'application/pdf',
+ },
+ },
+ {
+ media: {
+ url: '',
+ contentType: 'image/png',
+ },
+ },
+ ],
+ },
+ ],
+ expectedOutput: {
+ messages: [
+ {
+ content: [
+ {
+ text: 'Analyze this PDF and image:',
+ type: 'text',
+ citations: null,
+ },
+ {
+ type: 'document',
+ source: {
+ type: 'base64',
+ media_type: 'application/pdf',
+ data: 'JVBERi0xLjQKJ',
+ },
+ },
+ {
+ source: {
+ type: 'base64',
+ data: 'R0lGODlhAQABAAAAACw=',
+ media_type: 'image/png',
+ },
+ type: 'image',
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ system: undefined,
+ },
+ },
+ ];
+
+ for (const test of testCases) {
+ it(test.should, () => {
+ const actualOutput = testRunner.toAnthropicMessages(test.inputMessages);
+ assert.deepStrictEqual(actualOutput, test.expectedOutput);
+ });
+ }
+});
+
+describe('toAnthropicTool', () => {
+ it('should transform Genkit tool definition to an Anthropic tool', () => {
+ const tool: ToolDefinition = {
+ name: 'tellAJoke',
+ description: 'Tell a joke',
+ inputSchema: {
+ type: 'object',
+ properties: {
+ topic: { type: 'string' },
+ },
+ required: ['topic'],
+ },
+ };
+ const actualOutput = testRunner.toAnthropicTool(tool);
+ assert.deepStrictEqual(actualOutput, {
+ name: 'tellAJoke',
+ description: 'Tell a joke',
+ input_schema: {
+ type: 'object',
+ properties: {
+ topic: { type: 'string' },
+ },
+ required: ['topic'],
+ },
+ });
+ });
+});
+
+describe('fromAnthropicContentBlockChunk', () => {
+ const testCases: {
+ should: string;
+ event: MessageStreamEvent;
+ expectedOutput: Part | undefined;
+ }[] = [
+ {
+ should: 'should return text part from content_block_start event',
+ event: {
+ index: 0,
+ type: 'content_block_start',
+ content_block: {
+ type: 'text',
+ text: 'Hello, World!',
+ citations: null,
+ },
+ },
+ expectedOutput: { text: 'Hello, World!' },
+ },
+ {
+ should:
+ 'should return thinking part from content_block_start thinking event',
+ event: {
+ index: 0,
+ type: 'content_block_start',
+ content_block: {
+ type: 'thinking',
+ thinking: 'Let me reason through this.',
+ signature: 'sig_123',
+ },
+ },
+ expectedOutput: {
+ reasoning: 'Let me reason through this.',
+ custom: { anthropicThinking: { signature: 'sig_123' } },
+ },
+ },
+ {
+ should:
+ 'should return redacted thinking part from content_block_start event',
+ event: {
+ index: 0,
+ type: 'content_block_start',
+ content_block: {
+ type: 'redacted_thinking',
+ data: 'encrypted-data',
+ },
+ },
+ expectedOutput: { custom: { redactedThinking: 'encrypted-data' } },
+ },
+ {
+ should: 'should return text delta part from content_block_delta event',
+ event: {
+ index: 0,
+ type: 'content_block_delta',
+ delta: {
+ type: 'text_delta',
+ text: 'Hello, World!',
+ },
+ },
+ expectedOutput: { text: 'Hello, World!' },
+ },
+ {
+ should: 'should return thinking delta part as text content',
+ event: {
+ index: 0,
+ type: 'content_block_delta',
+ delta: {
+ type: 'thinking_delta',
+ thinking: 'Step by step...',
+ },
+ },
+ expectedOutput: { reasoning: 'Step by step...' },
+ },
+ {
+ should: 'should return tool use requests',
+ event: {
+ index: 0,
+ type: 'content_block_start',
+ content_block: {
+ type: 'tool_use',
+ id: 'abc123',
+ name: 'tellAJoke',
+ input: { topic: 'dogs' },
+ },
+ },
+ expectedOutput: {
+ toolRequest: {
+ name: 'tellAJoke',
+ input: { topic: 'dogs' },
+ ref: 'abc123',
+ },
+ },
+ },
+ {
+ should: 'should return undefined for any other event',
+ event: {
+ type: 'message_stop',
+ },
+ expectedOutput: undefined,
+ },
+ ];
+
+ for (const test of testCases) {
+ it(test.should, () => {
+ const actualOutput = testRunner.fromAnthropicContentBlockChunk(
+ test.event
+ );
+ assert.deepStrictEqual(actualOutput, test.expectedOutput);
+ });
+ }
+
+ it('should throw for unsupported tool input streaming deltas', () => {
+ assert.throws(
+ () =>
+ testRunner.fromAnthropicContentBlockChunk({
+ index: 0,
+ type: 'content_block_delta',
+ delta: {
+ type: 'input_json_delta',
+ partial_json: '{"foo":',
+ },
+ } as MessageStreamEvent),
+ /Anthropic streaming tool input \(input_json_delta\) is not yet supported/
+ );
+ });
+});
+
+describe('fromAnthropicStopReason', () => {
+ const testCases: {
+ inputStopReason: Message['stop_reason'];
+ expectedFinishReason: CandidateData['finishReason'];
+ }[] = [
+ {
+ inputStopReason: 'max_tokens',
+ expectedFinishReason: 'length',
+ },
+ {
+ inputStopReason: 'end_turn',
+ expectedFinishReason: 'stop',
+ },
+ {
+ inputStopReason: 'stop_sequence',
+ expectedFinishReason: 'stop',
+ },
+ {
+ inputStopReason: 'tool_use',
+ expectedFinishReason: 'stop',
+ },
+ {
+ inputStopReason: null,
+ expectedFinishReason: 'unknown',
+ },
+ {
+ inputStopReason: 'unknown' as any,
+ expectedFinishReason: 'other',
+ },
+ ];
+
+ for (const test of testCases) {
+ it(`should map Anthropic stop reason "${test.inputStopReason}" to Genkit finish reason "${test.expectedFinishReason}"`, () => {
+ const actualOutput = testRunner.fromAnthropicStopReason(
+ test.inputStopReason
+ );
+ assert.strictEqual(actualOutput, test.expectedFinishReason);
+ });
+ }
+});
+
+describe('fromAnthropicResponse', () => {
+ const testCases: {
+ should: string;
+ message: Message;
+ expectedOutput: Omit;
+ }[] = [
+ {
+ should: 'should work with text content',
+ message: {
+ id: 'abc123',
+ model: 'whatever',
+ type: 'message',
+ role: 'assistant',
+ stop_reason: 'max_tokens',
+ stop_sequence: null,
+ content: [
+ {
+ type: 'text',
+ text: 'Tell a joke about dogs.',
+ citations: null,
+ },
+ ],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: null,
+ cache_read_input_tokens: null,
+ }),
+ },
+ expectedOutput: {
+ candidates: [
+ {
+ index: 0,
+ finishReason: 'length',
+ message: {
+ role: 'model',
+ content: [{ text: 'Tell a joke about dogs.' }],
+ },
+ },
+ ],
+ usage: {
+ inputTokens: 10,
+ outputTokens: 20,
+ },
+ },
+ },
+ {
+ should: 'should work with tool use content',
+ message: {
+ id: 'abc123',
+ model: 'whatever',
+ type: 'message',
+ role: 'assistant',
+ stop_reason: 'tool_use',
+ stop_sequence: null,
+ content: [
+ {
+ type: 'tool_use',
+ id: 'abc123',
+ name: 'tellAJoke',
+ input: { topic: 'dogs' },
+ },
+ ],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: null,
+ cache_read_input_tokens: null,
+ }),
+ },
+ expectedOutput: {
+ candidates: [
+ {
+ index: 0,
+ finishReason: 'stop',
+ message: {
+ role: 'model',
+ content: [
+ {
+ toolRequest: {
+ name: 'tellAJoke',
+ input: { topic: 'dogs' },
+ ref: 'abc123',
+ },
+ },
+ ],
+ },
+ },
+ ],
+ usage: {
+ inputTokens: 10,
+ outputTokens: 20,
+ },
+ },
+ },
+ ];
+
+ for (const test of testCases) {
+ it(test.should, () => {
+ const actualOutput = testRunner.fromAnthropicResponse(test.message);
+ // Check custom field exists and is the message
+ assert.ok(actualOutput.custom);
+ assert.strictEqual(actualOutput.custom, test.message);
+ // Check the rest
+ assert.deepStrictEqual(
+ {
+ candidates: actualOutput.candidates,
+ usage: actualOutput.usage,
+ },
+ test.expectedOutput
+ );
+ });
+ }
+});
+
+describe('toAnthropicRequestBody', () => {
+ const testCases: {
+ should: string;
+ modelName: string;
+ genkitRequest: GenerateRequest;
+ expectedOutput: MessageCreateParams;
+ }[] = [
+ {
+ should: '(claude-3-5-haiku) handles request with text messages',
+ modelName: 'claude-3-5-haiku',
+ genkitRequest: {
+ messages: [
+ { role: 'user', content: [{ text: 'Tell a joke about dogs.' }] },
+ ],
+ output: { format: 'text' },
+ config: {
+ metadata: {
+ user_id: 'exampleUser123',
+ },
+ },
+ },
+ expectedOutput: {
+ max_tokens: 4096,
+ messages: [
+ {
+ content: [
+ {
+ text: 'Tell a joke about dogs.',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ model: 'claude-3-5-haiku',
+ metadata: {
+ user_id: 'exampleUser123',
+ },
+ },
+ },
+ {
+ should: '(claude-3-haiku) handles request with text messages',
+ modelName: 'claude-3-haiku',
+ genkitRequest: {
+ messages: [
+ { role: 'user', content: [{ text: 'Tell a joke about dogs.' }] },
+ ],
+ output: { format: 'text' },
+ config: {
+ metadata: {
+ user_id: 'exampleUser123',
+ },
+ },
+ },
+ expectedOutput: {
+ max_tokens: 4096,
+ messages: [
+ {
+ content: [
+ {
+ text: 'Tell a joke about dogs.',
+ type: 'text',
+ citations: null,
+ },
+ ],
+ role: 'user',
+ },
+ ],
+ model: 'claude-3-haiku',
+ metadata: {
+ user_id: 'exampleUser123',
+ },
+ },
+ },
+ ];
+ for (const test of testCases) {
+ it(test.should, () => {
+ const actualOutput = testRunner.toAnthropicRequestBody(
+ test.modelName,
+ test.genkitRequest
+ );
+ assert.deepStrictEqual(actualOutput, test.expectedOutput);
+ });
+ }
+
+ it('should accept any model name and use it directly', () => {
+ // Following Google GenAI pattern: accept any model name, let API validate
+ const result = testRunner.toAnthropicRequestBody('fake-model', {
+ messages: [],
+ } as GenerateRequest);
+
+ // Should not throw, and should use the model name directly
+ assert.strictEqual(result.model, 'fake-model');
+ });
+
+ it('should throw if output format is not text', () => {
+ assert.throws(
+ () =>
+ testRunner.toAnthropicRequestBody('claude-3-5-haiku', {
+ messages: [],
+ tools: [],
+ output: { format: 'media' },
+ } as GenerateRequest),
+ /Only text output format is supported for Claude models currently/
+ );
+ });
+
+ it('should apply system prompt caching when enabled', () => {
+ const request: GenerateRequest = {
+ messages: [
+ { role: 'system', content: [{ text: 'You are a helpful assistant' }] },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ // Test with caching enabled
+ const outputWithCaching = testRunner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+ assert.deepStrictEqual(outputWithCaching.system, [
+ {
+ type: 'text',
+ text: 'You are a helpful assistant',
+ cache_control: { type: 'ephemeral' },
+ },
+ ]);
+
+ // Test with caching disabled
+ const outputWithoutCaching = testRunner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ false
+ );
+ assert.strictEqual(
+ outputWithoutCaching.system,
+ 'You are a helpful assistant'
+ );
+ });
+
+ it('should concatenate multiple text parts in system message', () => {
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { text: 'Always be concise.' },
+ { text: 'Use proper grammar.' },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ const output = testRunner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ false
+ );
+
+ assert.strictEqual(
+ output.system,
+ 'You are a helpful assistant.\n\nAlways be concise.\n\nUse proper grammar.'
+ );
+ });
+
+ it('should concatenate multiple text parts in system message with caching', () => {
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { text: 'Always be concise.' },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ const output = testRunner.toAnthropicRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+
+ assert.deepStrictEqual(output.system, [
+ {
+ type: 'text',
+ text: 'You are a helpful assistant.\n\nAlways be concise.',
+ cache_control: { type: 'ephemeral' },
+ },
+ ]);
+ });
+
+ it('should throw error if system message contains media', () => {
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ {
+ media: {
+ url: '',
+ contentType: 'image/png',
+ },
+ },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ assert.throws(
+ () =>
+ testRunner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+
+ it('should throw error if system message contains tool requests', () => {
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { toolRequest: { name: 'getTool', input: {}, ref: '123' } },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ assert.throws(
+ () =>
+ testRunner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+
+ it('should throw error if system message contains tool responses', () => {
+ const request: GenerateRequest = {
+ messages: [
+ {
+ role: 'system',
+ content: [
+ { text: 'You are a helpful assistant.' },
+ { toolResponse: { name: 'getTool', output: {}, ref: '123' } },
+ ],
+ },
+ { role: 'user', content: [{ text: 'Hi' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ assert.throws(
+ () =>
+ testRunner.toAnthropicRequestBody('claude-3-5-haiku', request, false),
+ /System messages can only contain text content/
+ );
+ });
+});
+
+describe('toAnthropicStreamingRequestBody', () => {
+ it('should set stream to true', () => {
+ const request: GenerateRequest = {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ };
+
+ const output = testRunner.toAnthropicStreamingRequestBody(
+ 'claude-3-5-haiku',
+ request
+ );
+
+ assert.strictEqual(output.stream, true);
+ assert.strictEqual(output.model, 'claude-3-5-haiku');
+ assert.strictEqual(output.max_tokens, 4096);
+ });
+
+ it('should support system prompt caching in streaming mode', () => {
+ const request: GenerateRequest = {
+ messages: [
+ { role: 'system', content: [{ text: 'You are a helpful assistant' }] },
+ { role: 'user', content: [{ text: 'Hello' }] },
+ ],
+ output: { format: 'text' },
+ };
+
+ const outputWithCaching = testRunner.toAnthropicStreamingRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ true
+ );
+ assert.deepStrictEqual(outputWithCaching.system, [
+ {
+ type: 'text',
+ text: 'You are a helpful assistant',
+ cache_control: { type: 'ephemeral' },
+ },
+ ]);
+ assert.strictEqual(outputWithCaching.stream, true);
+
+ const outputWithoutCaching = testRunner.toAnthropicStreamingRequestBody(
+ 'claude-3-5-haiku',
+ request,
+ false
+ );
+ assert.strictEqual(
+ outputWithoutCaching.system,
+ 'You are a helpful assistant'
+ );
+ assert.strictEqual(outputWithoutCaching.stream, true);
+ });
+});
+
+describe('claudeRunner', () => {
+ it('should correctly run non-streaming requests', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ { messages: [] },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ assert.deepStrictEqual(createStub.mock.calls[0].arguments, [
+ {
+ model: 'claude-3-5-haiku',
+ max_tokens: 4096,
+ messages: [],
+ },
+ {
+ signal: abortSignal,
+ },
+ ]);
+ });
+
+ it('should correctly run streaming requests', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'text',
+ text: 'res',
+ },
+ } as MessageStreamEvent,
+ ],
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const streamingCallback = mock.fn();
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ { messages: [] },
+ { streamingRequested: true, sendChunk: streamingCallback, abortSignal }
+ );
+
+ const streamStub = mockClient.messages.stream as any;
+ assert.strictEqual(streamStub.mock.calls.length, 1);
+ assert.deepStrictEqual(streamStub.mock.calls[0].arguments, [
+ {
+ model: 'claude-3-5-haiku',
+ max_tokens: 4096,
+ messages: [],
+ stream: true,
+ },
+ {
+ signal: abortSignal,
+ },
+ ]);
+ });
+
+ it('should use beta API when apiVersion is beta', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ {
+ messages: [],
+ config: { apiVersion: 'beta' },
+ },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 1);
+
+ const regularCreateStub = mockClient.messages.create as any;
+ assert.strictEqual(regularCreateStub.mock.calls.length, 0);
+ });
+
+ it('should use beta API when defaultApiVersion is beta', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ defaultApiVersion: 'beta',
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ {
+ messages: [],
+ },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 1);
+
+ const regularCreateStub = mockClient.messages.create as any;
+ assert.strictEqual(regularCreateStub.mock.calls.length, 0);
+ });
+
+ it('should use request apiVersion over defaultApiVersion', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ // defaultApiVersion is 'stable', but request overrides to 'beta'
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ defaultApiVersion: 'stable',
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ {
+ messages: [],
+ config: { apiVersion: 'beta' },
+ },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 1);
+
+ const regularCreateStub = mockClient.messages.create as any;
+ assert.strictEqual(regularCreateStub.mock.calls.length, 0);
+ });
+
+ it('should use stable API when defaultApiVersion is beta but request overrides to stable', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: {
+ content: [{ type: 'text', text: 'response', citations: null }],
+ usage: createUsage({
+ input_tokens: 10,
+ output_tokens: 20,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ // defaultApiVersion is 'beta', but request overrides to 'stable'
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ defaultApiVersion: 'beta',
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ await runner(
+ {
+ messages: [],
+ config: { apiVersion: 'stable' },
+ },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 0);
+
+ const regularCreateStub = mockClient.messages.create as any;
+ assert.strictEqual(regularCreateStub.mock.calls.length, 1);
+ });
+});
+
+describe('claudeRunner param object', () => {
+ it('should run requests when constructed with params object', async () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ cacheSystemPrompt: true,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+
+ await runner(
+ { messages: [{ role: 'user', content: [{ text: 'hi' }] }] },
+ { streamingRequested: false, sendChunk: () => {}, abortSignal }
+ );
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ assert.strictEqual(
+ createStub.mock.calls[0].arguments[0].messages[0].content[0].text,
+ 'hi'
+ );
+ });
+
+ it('should route to beta runner when defaultApiVersion is beta', async () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ defaultApiVersion: 'beta',
+ },
+ AnthropicConfigSchema
+ );
+ await runner(
+ { messages: [] },
+ {
+ streamingRequested: false,
+ sendChunk: () => {},
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 1);
+ });
+
+ it('should throw when client is omitted from params object', () => {
+ assert.throws(() => {
+ claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: undefined as unknown as Anthropic,
+ },
+ AnthropicConfigSchema
+ );
+ }, /Anthropic client is required to create a runner/);
+ });
+});
+
+describe('claudeModel', () => {
+ it('should fall back to generic metadata for unknown models', async () => {
+ const mockClient = createMockAnthropicClient();
+ const modelAction = claudeModel({
+ name: 'unknown-model',
+ client: mockClient,
+ });
+
+ const abortSignal = new AbortController().signal;
+ await (modelAction as any)(
+ { messages: [{ role: 'user', content: [{ text: 'hi' }] }] },
+ {
+ streamingRequested: false,
+ sendChunk: () => {},
+ abortSignal,
+ }
+ );
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ const request = createStub.mock.calls[0].arguments[0];
+ assert.strictEqual(request.model, 'unknown-model');
+ });
+ it('should support params object configuration', async () => {
+ const mockClient = createMockAnthropicClient();
+ const modelAction = claudeModel({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ defaultApiVersion: 'beta',
+ cacheSystemPrompt: true,
+ });
+
+ const abortSignal = new AbortController().signal;
+ await (modelAction as any)(
+ { messages: [], config: { maxOutputTokens: 128 } },
+ {
+ streamingRequested: false,
+ sendChunk: () => {},
+ abortSignal,
+ }
+ );
+
+ const betaCreateStub = mockClient.beta.messages.create as any;
+ assert.strictEqual(betaCreateStub.mock.calls.length, 1);
+ assert.strictEqual(
+ betaCreateStub.mock.calls[0].arguments[0].max_tokens,
+ 128
+ );
+ });
+
+ it('should correctly define supported Claude models', () => {
+ const mockClient = createMockAnthropicClient();
+ const modelName = 'claude-3-5-haiku';
+ const modelAction = claudeModel({
+ name: modelName,
+ client: mockClient,
+ });
+
+ // Verify the model action is returned
+ assert.ok(modelAction);
+ assert.strictEqual(typeof modelAction, 'function');
+ });
+
+ it('should accept any model name and create a model action', () => {
+ // Following Google GenAI pattern: accept any model name, let API validate
+ const modelAction = claudeModel({
+ name: 'unsupported-model',
+ client: {} as Anthropic,
+ });
+ assert.ok(modelAction, 'Should create model action for any model name');
+ assert.strictEqual(typeof modelAction, 'function');
+ });
+
+ it('should handle streaming with multiple text chunks', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockContentBlockStart('Hello'),
+ mockTextChunk(' world'),
+ mockTextChunk('!'),
+ ],
+ messageResponse: {
+ content: [{ type: 'text', text: 'Hello world!', citations: null }],
+ usage: createUsage({
+ input_tokens: 5,
+ output_tokens: 10,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const chunks: any[] = [];
+ const streamingCallback = mock.fn((chunk: any) => {
+ chunks.push(chunk);
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+
+ const result = await runner(
+ { messages: [{ role: 'user', content: [{ text: 'Hi' }] }] },
+ { streamingRequested: true, sendChunk: streamingCallback, abortSignal }
+ );
+
+ // Verify we received all the streaming chunks
+ assert.ok(chunks.length > 0, 'Should have received streaming chunks');
+ assert.strictEqual(chunks.length, 3, 'Should have received 3 chunks');
+
+ // Verify the final result
+ assert.ok(result.candidates);
+ assert.strictEqual(
+ result.candidates[0].message.content[0].text,
+ 'Hello world!'
+ );
+ assert.ok(result.usage);
+ assert.strictEqual(result.usage.inputTokens, 5);
+ assert.strictEqual(result.usage.outputTokens, 10);
+ });
+
+ it('should handle tool use in streaming mode', async () => {
+ const streamChunks = [
+ {
+ type: 'content_block_start',
+ index: 0,
+ content_block: {
+ type: 'tool_use',
+ id: 'toolu_123',
+ name: 'get_weather',
+ input: { city: 'NYC' },
+ },
+ } as MessageStreamEvent,
+ ];
+ const mockClient = createMockAnthropicClient({
+ streamChunks,
+ messageResponse: {
+ content: [
+ {
+ type: 'tool_use',
+ id: 'toolu_123',
+ name: 'get_weather',
+ input: { city: 'NYC' },
+ },
+ ],
+ stop_reason: 'tool_use',
+ usage: createUsage({
+ input_tokens: 15,
+ output_tokens: 25,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const chunks: any[] = [];
+ const streamingCallback = mock.fn((chunk: any) => {
+ chunks.push(chunk);
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+
+ const result = await runner(
+ {
+ messages: [
+ { role: 'user', content: [{ text: 'What is the weather?' }] },
+ ],
+ tools: [
+ {
+ name: 'get_weather',
+ description: 'Get the weather for a city',
+ inputSchema: {
+ type: 'object',
+ properties: {
+ city: { type: 'string' },
+ },
+ required: ['city'],
+ },
+ },
+ ],
+ },
+ { streamingRequested: true, sendChunk: streamingCallback, abortSignal }
+ );
+
+ // Verify we received the tool use chunk
+ assert.ok(chunks.length > 0, 'Should have received chunks');
+
+ // Verify the final result contains tool use
+ assert.ok(result.candidates);
+ const toolRequest = result.candidates[0].message.content.find(
+ (p) => p.toolRequest
+ );
+ assert.ok(toolRequest, 'Should have a tool request');
+ assert.strictEqual(toolRequest.toolRequest?.name, 'get_weather');
+ assert.deepStrictEqual(toolRequest.toolRequest?.input, { city: 'NYC' });
+ });
+
+ it('should handle streaming errors and partial responses', async () => {
+ const streamError = new Error('Network error during streaming');
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [mockContentBlockStart('Hello'), mockTextChunk(' world')],
+ streamErrorAfterChunk: 1, // Throw error after first chunk
+ streamError: streamError,
+ messageResponse: {
+ content: [{ type: 'text', text: 'Hello world', citations: null }],
+ usage: createUsage({
+ input_tokens: 5,
+ output_tokens: 10,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortSignal = new AbortController().signal;
+ const chunks: any[] = [];
+ const sendChunk = (chunk: any) => {
+ chunks.push(chunk);
+ };
+
+ // Should throw error during streaming
+ await assert.rejects(
+ async () => {
+ await runner(
+ { messages: [{ role: 'user', content: [{ text: 'Hi' }] }] },
+ {
+ streamingRequested: true,
+ sendChunk,
+ abortSignal,
+ }
+ );
+ },
+ (error: Error) => {
+ // Verify error is propagated
+ assert.strictEqual(error.message, 'Network error during streaming');
+ // Verify we received at least one chunk before error
+ assert.ok(
+ chunks.length > 0,
+ 'Should have received some chunks before error'
+ );
+ return true;
+ }
+ );
+ });
+
+ it('should handle abort signal during streaming', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockContentBlockStart('Hello'),
+ mockTextChunk(' world'),
+ mockTextChunk('!'),
+ ],
+ messageResponse: {
+ content: [{ type: 'text', text: 'Hello world!', citations: null }],
+ usage: createUsage({
+ input_tokens: 5,
+ output_tokens: 15,
+ cache_creation_input_tokens: 0,
+ cache_read_input_tokens: 0,
+ }),
+ },
+ });
+
+ const runner = claudeRunner(
+ {
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ },
+ AnthropicConfigSchema
+ );
+ const abortController = new AbortController();
+ const chunks: any[] = [];
+ const sendChunk = (chunk: any) => {
+ chunks.push(chunk);
+ // Abort after first chunk
+ if (chunks.length === 1) {
+ abortController.abort();
+ }
+ };
+
+ // Should throw AbortError when signal is aborted
+ await assert.rejects(
+ async () => {
+ await runner(
+ { messages: [{ role: 'user', content: [{ text: 'Hi' }] }] },
+ {
+ streamingRequested: true,
+ sendChunk,
+ abortSignal: abortController.signal,
+ }
+ );
+ },
+ (error: Error) => {
+ // Verify abort error is thrown
+ assert.ok(
+ error.name === 'AbortError' || error.message.includes('AbortError'),
+ 'Should throw AbortError'
+ );
+ return true;
+ }
+ );
+ });
+
+ it('should handle unknown models using generic settings', async () => {
+ const mockClient = createMockAnthropicClient();
+ const modelAction = claudeModel({
+ name: 'unknown-model',
+ client: mockClient,
+ });
+
+ const abortSignal = new AbortController().signal;
+ await (modelAction as any)(
+ { messages: [{ role: 'user', content: [{ text: 'hi' }] }] },
+ {
+ streamingRequested: false,
+ sendChunk: () => {},
+ abortSignal,
+ }
+ );
+
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+ assert.strictEqual(
+ createStub.mock.calls[0].arguments[0].model,
+ 'unknown-model'
+ );
+ });
+});
+
+describe('BaseRunner helper utilities', () => {
+ it('should throw descriptive errors for invalid PDF data URLs', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ assert.throws(
+ () =>
+ runner['toPdfDocumentSource']({
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'application/pdf',
+ } as any),
+ /PDF contentType mismatch/
+ );
+ });
+
+ it('should stringify non-media tool responses', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ const result = runner['toAnthropicToolResponseContent']({
+ toolResponse: {
+ ref: 'call_1',
+ name: 'tool',
+ output: { value: 42 },
+ },
+ } as any);
+
+ assert.deepStrictEqual(result, {
+ type: 'text',
+ text: JSON.stringify({ value: 42 }),
+ });
+ });
+
+ it('should parse image data URLs', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ const source = runner['toImageSource']({
+ url: '',
+ contentType: 'image/png',
+ });
+
+ assert.strictEqual(source.kind, 'base64');
+ if (source.kind !== 'base64') {
+ throw new Error('Expected base64 image source');
+ }
+ assert.strictEqual(source.mediaType, 'image/png');
+ assert.strictEqual(source.data, 'AAA');
+ });
+
+ it('should pass through remote image URLs', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ const source = runner['toImageSource']({
+ url: 'https://example.com/image.png',
+ contentType: 'image/png',
+ });
+
+ assert.strictEqual(source.kind, 'url');
+ if (source.kind !== 'url') {
+ throw new Error('Expected url image source');
+ }
+ assert.strictEqual(source.url, 'https://example.com/image.png');
+ });
+
+ it('should parse WEBP image data URLs with matching contentType', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ const source = runner['toImageSource']({
+ url: '',
+ contentType: 'image/webp',
+ });
+
+ assert.strictEqual(source.kind, 'base64');
+ if (source.kind !== 'base64') {
+ throw new Error('Expected base64 image source');
+ }
+ assert.strictEqual(source.mediaType, 'image/webp');
+ assert.strictEqual(source.data, 'AAA');
+ });
+
+ it('should prefer data URL content type over media.contentType for WEBP', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ // Even if contentType says PNG, data URL says WEBP - should use WEBP
+ const source = runner['toImageSource']({
+ url: '',
+ contentType: 'image/png',
+ });
+
+ assert.strictEqual(source.kind, 'base64');
+ if (source.kind !== 'base64') {
+ throw new Error('Expected base64 image source');
+ }
+ // Key fix: should use data URL type (webp), not contentType (png)
+ assert.strictEqual(source.mediaType, 'image/webp');
+ assert.strictEqual(source.data, 'AAA');
+ });
+
+ it('should handle WEBP via toAnthropicMessageContent', () => {
+ const result = testRunner.toAnthropicMessageContent({
+ media: {
+ url: '',
+ contentType: 'image/webp',
+ },
+ });
+
+ assert.strictEqual(result.type, 'image');
+ assert.strictEqual(result.source.type, 'base64');
+ assert.strictEqual(result.source.media_type, 'image/webp');
+ assert.strictEqual(result.source.data, 'AAA');
+ });
+
+ it('should handle WEBP in tool response content', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ const result = runner['toAnthropicToolResponseContent']({
+ toolResponse: {
+ ref: 'call_123',
+ name: 'get_image',
+ output: {
+ url: '',
+ contentType: 'image/webp',
+ },
+ },
+ } as any);
+
+ assert.strictEqual(result.type, 'image');
+ assert.strictEqual(result.source.type, 'base64');
+ assert.strictEqual(result.source.media_type, 'image/webp');
+ assert.strictEqual(result.source.data, 'AAA');
+ });
+
+ it('should throw helpful error for text/plain in toImageSource', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ assert.throws(
+ () =>
+ runner['toImageSource']({
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'text/plain',
+ }),
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ }
+ );
+ });
+
+ it('should throw helpful error for text/plain in toAnthropicMessageContent', () => {
+ assert.throws(
+ () =>
+ testRunner.toAnthropicMessageContent({
+ media: {
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'text/plain',
+ },
+ }),
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ }
+ );
+ });
+
+ it('should throw helpful error for text/plain in tool response', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ assert.throws(
+ () =>
+ runner['toAnthropicToolResponseContent']({
+ toolResponse: {
+ ref: 'call_123',
+ name: 'get_file',
+ output: {
+ url: 'data:text/plain;base64,AAA',
+ contentType: 'text/plain',
+ },
+ },
+ } as any),
+ (error: Error) => {
+ return error.message.includes(
+ 'Text files should be sent as text content'
+ );
+ }
+ );
+ });
+
+ it('should throw helpful error for text/plain with remote URL', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ });
+
+ assert.throws(
+ () =>
+ runner['toImageSource']({
+ url: 'https://example.com/file.txt',
+ contentType: 'text/plain',
+ }),
+ (error: Error) => {
+ return (
+ error.message.includes('Text files should be sent as text content') &&
+ error.message.includes('text:')
+ );
+ }
+ );
+ });
+});
+
+describe('Runner request bodies and error branches', () => {
+ it('should include optional config fields in non-streaming request body', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ cacheSystemPrompt: true,
+ }) as Runner & RunnerProtectedMethods;
+
+ const body = runner['toAnthropicRequestBody'](
+ 'claude-3-5-haiku',
+ {
+ messages: [
+ {
+ role: 'system',
+ content: [{ text: 'You are helpful.' }],
+ },
+ {
+ role: 'user',
+ content: [{ text: 'Tell me a joke' }],
+ },
+ ],
+ config: {
+ maxOutputTokens: 256,
+ topK: 3,
+ topP: 0.75,
+ temperature: 0.6,
+ stopSequences: ['END'],
+ metadata: { user_id: 'user-xyz' },
+ tool_choice: { type: 'auto' },
+ thinking: { enabled: true, budgetTokens: 2048 },
+ },
+ tools: [
+ {
+ name: 'get_weather',
+ description: 'Returns the weather',
+ inputSchema: { type: 'object' },
+ },
+ ],
+ } as unknown as GenerateRequest,
+ true
+ );
+
+ assert.strictEqual(body.model, 'claude-3-5-haiku');
+ assert.ok(Array.isArray(body.system));
+ assert.strictEqual(body.system?.[0].cache_control?.type, 'ephemeral');
+ assert.strictEqual(body.max_tokens, 256);
+ assert.strictEqual(body.top_k, 3);
+ assert.strictEqual(body.top_p, 0.75);
+ assert.strictEqual(body.temperature, 0.6);
+ assert.deepStrictEqual(body.stop_sequences, ['END']);
+ assert.deepStrictEqual(body.metadata, { user_id: 'user-xyz' });
+ assert.deepStrictEqual(body.tool_choice, { type: 'auto' });
+ assert.strictEqual(body.tools?.length, 1);
+ assert.deepStrictEqual(body.thinking, {
+ type: 'enabled',
+ budget_tokens: 2048,
+ });
+ });
+
+ it('should include optional config fields in streaming request body', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ cacheSystemPrompt: true,
+ }) as Runner & RunnerProtectedMethods;
+
+ const body = runner['toAnthropicStreamingRequestBody'](
+ 'claude-3-5-haiku',
+ {
+ messages: [
+ {
+ role: 'system',
+ content: [{ text: 'Stay brief.' }],
+ },
+ {
+ role: 'user',
+ content: [{ text: 'Summarize the weather.' }],
+ },
+ ],
+ config: {
+ maxOutputTokens: 64,
+ topK: 2,
+ topP: 0.6,
+ temperature: 0.4,
+ stopSequences: ['STOP'],
+ metadata: { user_id: 'user-abc' },
+ tool_choice: { type: 'any' },
+ thinking: { enabled: true, budgetTokens: 1536 },
+ },
+ tools: [
+ {
+ name: 'summarize_weather',
+ description: 'Summarizes a forecast',
+ inputSchema: { type: 'object' },
+ },
+ ],
+ } as unknown as GenerateRequest,
+ true
+ );
+
+ assert.strictEqual(body.stream, true);
+ assert.ok(Array.isArray(body.system));
+ assert.strictEqual(body.max_tokens, 64);
+ assert.strictEqual(body.top_k, 2);
+ assert.strictEqual(body.top_p, 0.6);
+ assert.strictEqual(body.temperature, 0.4);
+ assert.deepStrictEqual(body.stop_sequences, ['STOP']);
+ assert.deepStrictEqual(body.metadata, { user_id: 'user-abc' });
+ assert.deepStrictEqual(body.tool_choice, { type: 'any' });
+ assert.strictEqual(body.tools?.length, 1);
+ assert.deepStrictEqual(body.thinking, {
+ type: 'enabled',
+ budget_tokens: 1536,
+ });
+ });
+
+ it('should disable thinking when explicitly turned off', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ }) as Runner & RunnerProtectedMethods;
+
+ const body = runner['toAnthropicRequestBody'](
+ 'claude-3-5-haiku',
+ {
+ messages: [],
+ config: {
+ thinking: { enabled: false },
+ },
+ } as unknown as GenerateRequest,
+ false
+ );
+
+ assert.deepStrictEqual(body.thinking, { type: 'disabled' });
+ });
+
+ it('should throw descriptive errors for missing tool refs', () => {
+ const mockClient = createMockAnthropicClient();
+ const runner = new Runner({
+ name: 'claude-3-5-haiku',
+ client: mockClient,
+ cacheSystemPrompt: false,
+ }) as Runner & RunnerProtectedMethods;
+
+ assert.throws(
+ () =>
+ runner['toAnthropicMessageContent']({
+ toolRequest: {
+ name: 'get_weather',
+ input: {},
+ },
+ } as any),
+ /Tool request ref is required/
+ );
+
+ assert.throws(
+ () =>
+ runner['toAnthropicMessageContent']({
+ toolResponse: {
+ ref: undefined,
+ name: 'get_weather',
+ output: 'Sunny',
+ },
+ } as any),
+ /Tool response ref is required/
+ );
+
+ assert.throws(
+ () =>
+ runner['toAnthropicMessageContent']({
+ data: 'unexpected',
+ } as any),
+ /Unsupported genkit part fields/
+ );
+ });
+});
diff --git a/js/plugins/anthropic/tests/streaming_test.ts b/js/plugins/anthropic/tests/streaming_test.ts
new file mode 100644
index 0000000000..84d45e0d9a
--- /dev/null
+++ b/js/plugins/anthropic/tests/streaming_test.ts
@@ -0,0 +1,366 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import * as assert from 'assert';
+import type { ModelAction } from 'genkit/model';
+import { describe, mock, test } from 'node:test';
+import { anthropic } from '../src/index.js';
+import { PluginOptions, __testClient } from '../src/types.js';
+import {
+ createMockAnthropicClient,
+ createMockAnthropicMessage,
+ mockContentBlockStart,
+ mockTextChunk,
+ mockToolUseChunk,
+} from './mocks/anthropic-client.js';
+
+describe('Streaming Integration Tests', () => {
+ test('should use streaming API when onChunk is provided', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockContentBlockStart('Hello'),
+ mockTextChunk(' world'),
+ mockTextChunk('!'),
+ ],
+ messageResponse: createMockAnthropicMessage({
+ text: 'Hello world!',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ // Verify final response
+ assert.ok(response, 'Response should be returned');
+ assert.ok(
+ response.candidates?.[0]?.message.content[0].text,
+ 'Response should have text content'
+ );
+
+ // Since we can't control whether the runner chooses streaming or not from
+ // the plugin level, just verify we got a response
+ // The runner-level tests verify streaming behavior in detail
+ });
+
+ test('should handle streaming with multiple content blocks', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockContentBlockStart('First block'),
+ mockTextChunk(' continues'),
+ {
+ type: 'content_block_start',
+ index: 1,
+ content_block: {
+ type: 'text',
+ text: 'Second block',
+ },
+ } as any,
+ {
+ type: 'content_block_delta',
+ index: 1,
+ delta: {
+ type: 'text_delta',
+ text: ' here',
+ },
+ } as any,
+ ],
+ messageResponse: createMockAnthropicMessage({
+ text: 'First block continues',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ // Verify response is returned even with multiple content blocks
+ assert.ok(response, 'Response should be returned');
+ });
+
+ test('should handle streaming with tool use', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [
+ mockToolUseChunk('toolu_123', 'get_weather', { city: 'NYC' }),
+ ],
+ messageResponse: createMockAnthropicMessage({
+ toolUse: {
+ id: 'toolu_123',
+ name: 'get_weather',
+ input: { city: 'NYC' },
+ },
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Get NYC weather' }] }],
+ tools: [
+ {
+ name: 'get_weather',
+ description: 'Get weather for a city',
+ inputSchema: {
+ type: 'object',
+ properties: {
+ city: { type: 'string' },
+ },
+ required: ['city'],
+ },
+ },
+ ],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ // Verify tool use in response
+ assert.ok(response.candidates?.[0]?.message.content[0].toolRequest);
+ assert.strictEqual(
+ response.candidates[0].message.content[0].toolRequest?.name,
+ 'get_weather'
+ );
+ });
+
+ test('should handle abort signal', async () => {
+ const abortController = new AbortController();
+
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Hello world',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ // Abort before starting
+ abortController.abort();
+
+ // Test that abort signal is passed through
+ // The actual abort behavior is tested in runner tests
+ try {
+ await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: abortController.signal,
+ }
+ );
+ // If we get here, the mock doesn't fully simulate abort behavior,
+ // which is fine since runner tests cover this
+ } catch (error: any) {
+ // Expected abort error
+ assert.ok(
+ error.message.includes('Abort') || error.name === 'AbortError',
+ 'Should throw abort error'
+ );
+ }
+ });
+
+ test('should handle errors during streaming', async () => {
+ const mockClient = createMockAnthropicClient({
+ shouldError: new Error('API error'),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ try {
+ await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+ assert.fail('Should have thrown an error');
+ } catch (error: any) {
+ assert.strictEqual(error.message, 'API error');
+ }
+ });
+
+ test('should handle empty response', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [],
+ messageResponse: createMockAnthropicMessage({
+ text: '',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response, 'Should return response even with empty content');
+ });
+
+ test('should include usage metadata in streaming response', async () => {
+ const mockClient = createMockAnthropicClient({
+ streamChunks: [mockContentBlockStart('Response'), mockTextChunk(' text')],
+ messageResponse: createMockAnthropicMessage({
+ text: 'Response text',
+ usage: {
+ input_tokens: 50,
+ output_tokens: 25,
+ },
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ const response = await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ output: { format: 'text' },
+ },
+ {
+ onChunk: mock.fn() as any,
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ assert.ok(response.usage, 'Should include usage metadata');
+ assert.strictEqual(response.usage?.inputTokens, 50);
+ assert.strictEqual(response.usage?.outputTokens, 25);
+ });
+
+ test('should not stream when onChunk is not provided', async () => {
+ const mockClient = createMockAnthropicClient({
+ messageResponse: createMockAnthropicMessage({
+ text: 'Non-streaming response',
+ }),
+ });
+
+ const plugin = anthropic({
+ apiKey: 'test-key',
+ [__testClient]: mockClient,
+ } as PluginOptions);
+
+ const modelAction = plugin.resolve!(
+ 'model',
+ 'claude-3-5-haiku-20241022'
+ ) as ModelAction;
+
+ await modelAction(
+ {
+ messages: [{ role: 'user', content: [{ text: 'Hello' }] }],
+ },
+ {
+ abortSignal: new AbortController().signal,
+ }
+ );
+
+ // Verify non-streaming API was called
+ const createStub = mockClient.messages.create as any;
+ assert.strictEqual(createStub.mock.calls.length, 1);
+
+ // Verify stream API was NOT called
+ const streamStub = mockClient.messages.stream as any;
+ assert.strictEqual(streamStub.mock.calls.length, 0);
+ });
+});
diff --git a/js/plugins/anthropic/tests/types_test.ts b/js/plugins/anthropic/tests/types_test.ts
new file mode 100644
index 0000000000..64c91e1547
--- /dev/null
+++ b/js/plugins/anthropic/tests/types_test.ts
@@ -0,0 +1,89 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import * as assert from 'assert';
+import { z } from 'genkit';
+import { describe, it } from 'node:test';
+import { AnthropicConfigSchema, resolveBetaEnabled } from '../src/types.js';
+
+describe('resolveBetaEnabled', () => {
+ it('should return true when config.apiVersion is beta', () => {
+ const config: z.infer = {
+ apiVersion: 'beta',
+ };
+ assert.strictEqual(resolveBetaEnabled(config, 'stable'), true);
+ });
+
+ it('should return true when pluginDefaultApiVersion is beta', () => {
+ assert.strictEqual(resolveBetaEnabled(undefined, 'beta'), true);
+ });
+
+ it('should return false when config.apiVersion is stable', () => {
+ const config: z.infer = {
+ apiVersion: 'stable',
+ };
+ assert.strictEqual(resolveBetaEnabled(config, 'stable'), false);
+ });
+
+ it('should return false when both are stable', () => {
+ const config: z.infer = {
+ apiVersion: 'stable',
+ };
+ assert.strictEqual(resolveBetaEnabled(config, 'stable'), false);
+ });
+
+ it('should return false when neither is specified', () => {
+ assert.strictEqual(resolveBetaEnabled(undefined, undefined), false);
+ });
+
+ it('should return false when config is undefined and plugin default is stable', () => {
+ assert.strictEqual(resolveBetaEnabled(undefined, 'stable'), false);
+ });
+
+ it('should prioritize config.apiVersion over pluginDefaultApiVersion (beta over stable)', () => {
+ const config: z.infer = {
+ apiVersion: 'beta',
+ };
+ // Even though plugin default is stable, request config should override
+ assert.strictEqual(resolveBetaEnabled(config, 'stable'), true);
+ });
+
+ it('should prioritize config.apiVersion over pluginDefaultApiVersion (stable over beta)', () => {
+ const config: z.infer = {
+ apiVersion: 'stable',
+ };
+ // Request explicitly wants stable, should override plugin default
+ assert.strictEqual(resolveBetaEnabled(config, 'beta'), false);
+ });
+
+ it('should return false when config is empty object', () => {
+ const config: z.infer = {};
+ assert.strictEqual(resolveBetaEnabled(config, undefined), false);
+ });
+
+ it('should return true when config is empty but plugin default is beta', () => {
+ const config: z.infer = {};
+ assert.strictEqual(resolveBetaEnabled(config, 'beta'), true);
+ });
+
+ it('should handle config with other fields but no apiVersion', () => {
+ const config: z.infer = {
+ metadata: { user_id: 'test-user' },
+ };
+ assert.strictEqual(resolveBetaEnabled(config, 'stable'), false);
+ assert.strictEqual(resolveBetaEnabled(config, 'beta'), true);
+ });
+});
diff --git a/js/plugins/anthropic/tsconfig.json b/js/plugins/anthropic/tsconfig.json
new file mode 100644
index 0000000000..596e2cf729
--- /dev/null
+++ b/js/plugins/anthropic/tsconfig.json
@@ -0,0 +1,4 @@
+{
+ "extends": "../../tsconfig.json",
+ "include": ["src"]
+}
diff --git a/js/plugins/anthropic/tsup.config.ts b/js/plugins/anthropic/tsup.config.ts
new file mode 100644
index 0000000000..d55507161f
--- /dev/null
+++ b/js/plugins/anthropic/tsup.config.ts
@@ -0,0 +1,22 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { defineConfig, type Options } from 'tsup';
+import { defaultOptions } from '../../tsup.common';
+
+export default defineConfig({
+ ...(defaultOptions as Options),
+});
diff --git a/js/pnpm-lock.yaml b/js/pnpm-lock.yaml
index c1918c9103..109b07c772 100644
--- a/js/pnpm-lock.yaml
+++ b/js/pnpm-lock.yaml
@@ -257,6 +257,37 @@ importers:
specifier: ^4.9.0
version: 4.9.5
+ plugins/anthropic:
+ dependencies:
+ '@anthropic-ai/sdk':
+ specifier: ^0.68.0
+ version: 0.68.0(zod@3.25.67)
+ devDependencies:
+ '@types/node':
+ specifier: ^20.11.16
+ version: 20.19.1
+ check-node-version:
+ specifier: ^4.2.1
+ version: 4.2.1
+ genkit:
+ specifier: workspace:*
+ version: link:../../genkit
+ npm-run-all:
+ specifier: ^4.1.5
+ version: 4.1.5
+ rimraf:
+ specifier: ^6.0.1
+ version: 6.0.1
+ tsup:
+ specifier: ^8.3.5
+ version: 8.5.0(postcss@8.4.47)(tsx@4.20.3)(typescript@4.9.5)(yaml@2.8.0)
+ tsx:
+ specifier: ^4.19.2
+ version: 4.20.3
+ typescript:
+ specifier: ^4.9.0
+ version: 4.9.5
+
plugins/checks:
dependencies:
'@genkit-ai/ai':
@@ -994,6 +1025,25 @@ importers:
specifier: '>=12.2'
version: 13.4.0(encoding@0.1.13)
+ testapps/anthropic:
+ dependencies:
+ '@genkit-ai/anthropic':
+ specifier: workspace:*
+ version: link:../../plugins/anthropic
+ genkit:
+ specifier: workspace:*
+ version: link:../../genkit
+ devDependencies:
+ cross-env:
+ specifier: ^10.1.0
+ version: 10.1.0
+ tsx:
+ specifier: ^4.19.2
+ version: 4.20.3
+ typescript:
+ specifier: ^5.6.2
+ version: 5.8.3
+
testapps/basic-gemini:
dependencies:
'@genkit-ai/firebase':
@@ -2042,6 +2092,15 @@ packages:
'@anthropic-ai/sdk@0.24.3':
resolution: {integrity: sha512-916wJXO6T6k8R6BAAcLhLPv/pnLGy7YSEBZXZ1XTFbLcTZE8oTy3oDW9WJf9KKZwMvVcePIfoTSvzXHRcGxkQQ==}
+ '@anthropic-ai/sdk@0.68.0':
+ resolution: {integrity: sha512-SMYAmbbiprG8k1EjEPMTwaTqssDT7Ae+jxcR5kWXiqTlbwMR2AthXtscEVWOHkRfyAV5+y3PFYTJRNa3OJWIEw==}
+ hasBin: true
+ peerDependencies:
+ zod: ^3.25.0 || ^4.0.0
+ peerDependenciesMeta:
+ zod:
+ optional: true
+
'@anthropic-ai/sdk@0.9.1':
resolution: {integrity: sha512-wa1meQ2WSfoY8Uor3EdrJq0jTiZJoKoSii2ZVWRY1oN4Tlr5s59pADg9T79FTbPe1/se5c3pBeZgJL63wmuoBA==}
@@ -2202,6 +2261,10 @@ packages:
peerDependencies:
'@babel/core': ^7.0.0-0
+ '@babel/runtime@7.28.4':
+ resolution: {integrity: sha512-Q/N6JNWvIvPnLDvjlE1OUBLPQHH6l3CltCEsHIujp45zQUSSh8K+gHnaEX45yAT1nyngnINhvWtzN+Nb9D8RAQ==}
+ engines: {node: '>=6.9.0'}
+
'@babel/template@7.25.7':
resolution: {integrity: sha512-wRwtAgI3bAS+JGU2upWNL9lSlDcRCqD05BZ1n3X2ONLH1WilFP6O1otQjeMK/1g0pvYcXC7b/qVUB1keofjtZA==}
engines: {node: '>=6.9.0'}
@@ -4600,6 +4663,10 @@ packages:
resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==}
engines: {node: '>=4'}
+ chalk@3.0.0:
+ resolution: {integrity: sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==}
+ engines: {node: '>=8'}
+
chalk@4.1.2:
resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==}
engines: {node: '>=10'}
@@ -4611,6 +4678,11 @@ packages:
charenc@0.0.2:
resolution: {integrity: sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==}
+ check-node-version@4.2.1:
+ resolution: {integrity: sha512-YYmFYHV/X7kSJhuN/QYHUu998n/TRuDe8UenM3+m5NrkiH670lb9ILqHIvBencvJc4SDh+XcbXMR4b+TtubJiw==}
+ engines: {node: '>=8.3.0'}
+ hasBin: true
+
chokidar@4.0.3:
resolution: {integrity: sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA==}
engines: {node: '>= 14.16.0'}
@@ -5924,6 +5996,10 @@ packages:
json-parse-even-better-errors@2.3.1:
resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==}
+ json-schema-to-ts@3.1.1:
+ resolution: {integrity: sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g==}
+ engines: {node: '>=16'}
+
json-schema-traverse@0.4.1:
resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==}
@@ -6325,6 +6401,9 @@ packages:
makeerror@1.0.12:
resolution: {integrity: sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==}
+ map-values@1.0.1:
+ resolution: {integrity: sha512-BbShUnr5OartXJe1GeccAWtfro11hhgNJg6G9/UtWKjVGvV5U4C09cg5nk8JUevhXODaXY+hQ3xxMUKSs62ONQ==}
+
markdown-it@14.1.0:
resolution: {integrity: sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==}
hasBin: true
@@ -6558,6 +6637,9 @@ packages:
resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==}
engines: {node: '>=0.10.0'}
+ object-filter@1.0.2:
+ resolution: {integrity: sha512-NahvP2vZcy1ZiiYah30CEPw0FpDcSkSePJBMpzl5EQgCmISijiGuJm3SPYp7U+Lf2TljyaIw3E5EgkEx/TNEVA==}
+
object-hash@3.0.0:
resolution: {integrity: sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==}
engines: {node: '>= 6'}
@@ -6923,6 +7005,9 @@ packages:
resolution: {integrity: sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w==}
engines: {node: '>=0.6'}
+ queue-microtask@1.2.3:
+ resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==}
+
range-parser@1.2.1:
resolution: {integrity: sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==}
engines: {node: '>= 0.6'}
@@ -7036,6 +7121,9 @@ packages:
resolution: {integrity: sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==}
engines: {node: '>= 18'}
+ run-parallel@1.2.0:
+ resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==}
+
safe-array-concat@1.1.3:
resolution: {integrity: sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==}
engines: {node: '>=0.4'}
@@ -7074,11 +7162,6 @@ packages:
engines: {node: '>=10'}
hasBin: true
- semver@7.6.3:
- resolution: {integrity: sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==}
- engines: {node: '>=10'}
- hasBin: true
-
semver@7.7.2:
resolution: {integrity: sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==}
engines: {node: '>=10'}
@@ -7408,6 +7491,9 @@ packages:
resolution: {integrity: sha512-aZbgViZrg1QNcG+LULa7nhZpJTZSLm/mXnHXnbAbjmN5aSa0y7V+wvv6+4WaBtpISJzThKy+PIPxc1Nq1EJ9mg==}
engines: {node: '>= 14.0.0'}
+ ts-algebra@2.0.0:
+ resolution: {integrity: sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw==}
+
ts-interface-checker@0.1.13:
resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==}
@@ -7845,6 +7931,12 @@ snapshots:
transitivePeerDependencies:
- encoding
+ '@anthropic-ai/sdk@0.68.0(zod@3.25.67)':
+ dependencies:
+ json-schema-to-ts: 3.1.1
+ optionalDependencies:
+ zod: 3.25.67
+
'@anthropic-ai/sdk@0.9.1(encoding@0.1.13)':
dependencies:
'@types/node': 18.19.112
@@ -8042,6 +8134,8 @@ snapshots:
'@babel/core': 7.25.7
'@babel/helper-plugin-utils': 7.25.7
+ '@babel/runtime@7.28.4': {}
+
'@babel/template@7.25.7':
dependencies:
'@babel/code-frame': 7.25.7
@@ -10100,7 +10194,7 @@ snapshots:
'@opentelemetry/propagator-b3': 1.25.1(@opentelemetry/api@1.9.0)
'@opentelemetry/propagator-jaeger': 1.25.1(@opentelemetry/api@1.9.0)
'@opentelemetry/sdk-trace-base': 1.25.1(@opentelemetry/api@1.9.0)
- semver: 7.6.3
+ semver: 7.7.2
'@opentelemetry/semantic-conventions@1.25.1': {}
@@ -10861,6 +10955,11 @@ snapshots:
escape-string-regexp: 1.0.5
supports-color: 5.5.0
+ chalk@3.0.0:
+ dependencies:
+ ansi-styles: 4.3.0
+ supports-color: 7.2.0
+
chalk@4.1.2:
dependencies:
ansi-styles: 4.3.0
@@ -10870,6 +10969,15 @@ snapshots:
charenc@0.0.2: {}
+ check-node-version@4.2.1:
+ dependencies:
+ chalk: 3.0.0
+ map-values: 1.0.1
+ minimist: 1.2.8
+ object-filter: 1.0.2
+ run-parallel: 1.2.0
+ semver: 6.3.1
+
chokidar@4.0.3:
dependencies:
readdirp: 4.1.2
@@ -12740,6 +12848,11 @@ snapshots:
json-parse-even-better-errors@2.3.1: {}
+ json-schema-to-ts@3.1.1:
+ dependencies:
+ '@babel/runtime': 7.28.4
+ ts-algebra: 2.0.0
+
json-schema-traverse@0.4.1: {}
json-schema-traverse@1.0.0: {}
@@ -13082,6 +13195,8 @@ snapshots:
dependencies:
tmpl: 1.0.5
+ map-values@1.0.1: {}
+
markdown-it@14.1.0:
dependencies:
argparse: 2.0.1
@@ -13293,6 +13408,8 @@ snapshots:
object-assign@4.1.1: {}
+ object-filter@1.0.2: {}
+
object-hash@3.0.0: {}
object-inspect@1.13.1: {}
@@ -13654,6 +13771,8 @@ snapshots:
dependencies:
side-channel: 1.1.0
+ queue-microtask@1.2.3: {}
+
range-parser@1.2.1: {}
raw-body@2.5.2:
@@ -13817,6 +13936,10 @@ snapshots:
transitivePeerDependencies:
- supports-color
+ run-parallel@1.2.0:
+ dependencies:
+ queue-microtask: 1.2.3
+
safe-array-concat@1.1.3:
dependencies:
call-bind: 1.0.8
@@ -13854,8 +13977,6 @@ snapshots:
dependencies:
lru-cache: 6.0.0
- semver@7.6.3: {}
-
semver@7.7.2: {}
send@0.19.0:
@@ -14261,6 +14382,8 @@ snapshots:
triple-beam@1.4.1: {}
+ ts-algebra@2.0.0: {}
+
ts-interface-checker@0.1.13: {}
ts-jest@29.4.0(@babel/core@7.25.7)(@jest/transform@29.7.0)(@jest/types@29.6.3)(babel-jest@29.7.0(@babel/core@7.25.7))(jest-util@29.7.0)(jest@29.7.0(@types/node@20.19.1)(ts-node@10.9.2(@types/node@20.19.1)(typescript@4.9.5)))(typescript@4.9.5):
diff --git a/js/testapps/anthropic/README.md b/js/testapps/anthropic/README.md
new file mode 100644
index 0000000000..3343b90ea7
--- /dev/null
+++ b/js/testapps/anthropic/README.md
@@ -0,0 +1,67 @@
+# Anthropic Plugin Sample
+
+This test app demonstrates usage of the Genkit Anthropic plugin against both the stable and beta runners, organized by feature.
+
+## Directory Structure
+
+```
+src/
+ stable/
+ basic.ts - Basic stable API examples (hello, streaming)
+ text-plain.ts - Text/plain error handling demonstration
+ webp.ts - WEBP image handling demonstration
+ pdf.ts - PDF document processing examples
+ attention-first-page.pdf - Sample PDF file for testing
+ beta/
+ basic.ts - Basic beta API examples
+```
+
+## Setup
+
+1. From the repo root run `pnpm install` followed by `pnpm run setup` to link workspace dependencies.
+2. In this directory, optionally run `pnpm install` if you want a local `node_modules/`.
+3. Export an Anthropic API key (or add it to a `.env` file) before running any samples:
+
+ ```bash
+ export ANTHROPIC_API_KEY=your-key
+ ```
+
+## Available scripts
+
+### Basic Examples
+- `pnpm run build` – Compile the TypeScript sources into `lib/`.
+- `pnpm run start:stable` – Run the compiled stable basic sample.
+- `pnpm run start:beta` – Run the compiled beta basic sample.
+- `pnpm run dev:stable` – Start the Genkit Dev UI over `src/stable/basic.ts` with live reload.
+- `pnpm run dev:beta` – Start the Genkit Dev UI over `src/beta/basic.ts` with live reload.
+
+### Feature-Specific Examples
+- `pnpm run dev:stable:text-plain` – Start Dev UI for text/plain error handling demo.
+- `pnpm run dev:stable:webp` – Start Dev UI for WEBP image handling demo.
+- `pnpm run dev:stable:pdf` – Start Dev UI for PDF document processing demo.
+
+## Flows
+
+Each source file defines flows that can be invoked from the Dev UI or the Genkit CLI:
+
+### Basic Examples
+- `anthropic-stable-hello` – Simple greeting using stable API
+- `anthropic-stable-stream` – Streaming response example
+- `anthropic-beta-hello` – Simple greeting using beta API
+- `anthropic-beta-stream` – Streaming response with beta API
+- `anthropic-beta-opus41` – Test Opus 4.1 model with beta API
+
+### Text/Plain Handling
+- `stable-text-plain-error` – Demonstrates the helpful error when using text/plain as media
+- `stable-text-plain-correct` – Shows the correct way to send text content
+
+### WEBP Image Handling
+- `stable-webp-matching` – WEBP image with matching contentType
+- `stable-webp-mismatched` – WEBP image with mismatched contentType (demonstrates the fix)
+
+### PDF Document Processing
+- `stable-pdf-base64` – Process a PDF from a local file using base64 encoding
+- `stable-pdf-url` – Process a PDF from a publicly accessible URL
+- `stable-pdf-analysis` – Analyze a PDF document for key topics, concepts, and visual elements
+
+Example: `genkit flow:run anthropic-stable-hello`
diff --git a/js/testapps/anthropic/package.json b/js/testapps/anthropic/package.json
new file mode 100644
index 0000000000..08e1a0d2fd
--- /dev/null
+++ b/js/testapps/anthropic/package.json
@@ -0,0 +1,36 @@
+{
+ "name": "anthropic-testapp",
+ "version": "0.0.1",
+ "description": "Sample Genkit app showcasing Anthropic plugin stable and beta usage.",
+ "main": "lib/stable/basic.js",
+ "scripts": {
+ "build": "tsc",
+ "build:watch": "tsc --watch",
+ "start:stable": "node lib/stable/basic.js",
+ "start:beta": "node lib/beta/basic.js",
+ "dev:stable": "genkit start -- npx tsx --watch src/stable/basic.ts",
+ "dev:beta": "genkit start -- npx tsx --watch src/beta/basic.ts",
+ "dev:stable:text-plain": "genkit start -- npx tsx --watch src/stable/text-plain.ts",
+ "dev:stable:webp": "genkit start -- npx tsx --watch src/stable/webp.ts",
+ "dev:stable:pdf": "genkit start -- npx tsx --watch src/stable/pdf.ts",
+ "genkit:dev": "cross-env GENKIT_ENV=dev npm run dev:stable",
+ "genkit:start": "cross-env GENKIT_ENV=dev genkit start -- tsx --watch src/stable/basic.ts",
+ "dev": "export GENKIT_RUNTIME_ID=$(openssl rand -hex 8) && node lib/stable/basic.js 2>&1"
+ },
+ "keywords": [
+ "genkit",
+ "anthropic",
+ "sample"
+ ],
+ "author": "",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "genkit": "workspace:*",
+ "@genkit-ai/anthropic": "workspace:*"
+ },
+ "devDependencies": {
+ "cross-env": "^10.1.0",
+ "tsx": "^4.19.2",
+ "typescript": "^5.6.2"
+ }
+}
diff --git a/js/testapps/anthropic/src/beta/basic.ts b/js/testapps/anthropic/src/beta/basic.ts
new file mode 100644
index 0000000000..d1309b3400
--- /dev/null
+++ b/js/testapps/anthropic/src/beta/basic.ts
@@ -0,0 +1,76 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { anthropic } from '@genkit-ai/anthropic';
+import { genkit } from 'genkit';
+
+const ai = genkit({
+ plugins: [
+ // Default all flows in this sample to the beta surface
+ anthropic({ apiVersion: 'beta', cacheSystemPrompt: true }),
+ ],
+});
+
+const betaHaiku = anthropic.model('claude-3-5-haiku', { apiVersion: 'beta' });
+const betaSonnet = anthropic.model('claude-sonnet-4-5', { apiVersion: 'beta' });
+const betaOpus41 = anthropic.model('claude-opus-4-1', { apiVersion: 'beta' });
+
+ai.defineFlow('anthropic-beta-hello', async () => {
+ const { text } = await ai.generate({
+ model: betaHaiku,
+ prompt:
+ 'You are Claude on the beta API. Provide a concise greeting that mentions that you are using the beta API.',
+ config: { temperature: 0.6 },
+ });
+
+ return text;
+});
+
+ai.defineFlow('anthropic-beta-stream', async (_, { sendChunk }) => {
+ const { stream } = ai.generateStream({
+ model: betaSonnet,
+ prompt: [
+ {
+ text: 'Outline two experimental capabilities unlocked by the Anthropic beta API.',
+ },
+ ],
+ config: {
+ apiVersion: 'beta',
+ temperature: 0.4,
+ },
+ });
+
+ const collected: string[] = [];
+ for await (const chunk of stream) {
+ if (chunk.text) {
+ collected.push(chunk.text);
+ sendChunk(chunk.text);
+ }
+ }
+
+ return collected.join('');
+});
+
+ai.defineFlow('anthropic-beta-opus41', async () => {
+ const { text } = await ai.generate({
+ model: betaOpus41,
+ prompt:
+ 'You are Claude Opus 4.1 on the beta API. Provide a brief greeting that confirms you are using the beta API.',
+ config: { temperature: 0.6 },
+ });
+
+ return text;
+});
diff --git a/js/testapps/anthropic/src/stable/attention-first-page.pdf b/js/testapps/anthropic/src/stable/attention-first-page.pdf
new file mode 100644
index 0000000000..95c6625029
Binary files /dev/null and b/js/testapps/anthropic/src/stable/attention-first-page.pdf differ
diff --git a/js/testapps/anthropic/src/stable/basic.ts b/js/testapps/anthropic/src/stable/basic.ts
new file mode 100644
index 0000000000..246a42539a
--- /dev/null
+++ b/js/testapps/anthropic/src/stable/basic.ts
@@ -0,0 +1,51 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { anthropic } from '@genkit-ai/anthropic';
+import { genkit } from 'genkit';
+
+const ai = genkit({
+ plugins: [
+ // Configure the plugin with environment-driven API key
+ anthropic(),
+ ],
+});
+
+ai.defineFlow('anthropic-stable-hello', async () => {
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ prompt: 'You are a friendly Claude assistant. Greet the user briefly.',
+ });
+
+ return text;
+});
+
+ai.defineFlow('anthropic-stable-stream', async (_, { sendChunk }) => {
+ const { stream } = ai.generateStream({
+ model: anthropic.model('claude-sonnet-4-5'),
+ prompt: 'Compose a short limerick about using Genkit with Anthropic.',
+ });
+
+ let response = '';
+ for await (const chunk of stream) {
+ response += chunk.text ?? '';
+ if (chunk.text) {
+ sendChunk(chunk.text);
+ }
+ }
+
+ return response;
+});
diff --git a/js/testapps/anthropic/src/stable/pdf.ts b/js/testapps/anthropic/src/stable/pdf.ts
new file mode 100644
index 0000000000..8953dff696
--- /dev/null
+++ b/js/testapps/anthropic/src/stable/pdf.ts
@@ -0,0 +1,122 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { anthropic } from '@genkit-ai/anthropic';
+import * as fs from 'fs';
+import { genkit } from 'genkit';
+import * as path from 'path';
+
+const ai = genkit({
+ plugins: [anthropic()],
+});
+
+/**
+ * This flow demonstrates PDF document processing with Claude using base64 encoding.
+ * The PDF is read from the source directory and sent as a base64 data URL.
+ */
+ai.defineFlow('stable-pdf-base64', async () => {
+ // Read PDF file from the same directory as this source file
+ const pdfPath = path.join(__dirname, 'attention-first-page.pdf');
+ const pdfBuffer = fs.readFileSync(pdfPath);
+ const pdfBase64 = pdfBuffer.toString('base64');
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ text: 'What are the key findings or main points in this document?',
+ },
+ {
+ media: {
+ url: `data:application/pdf;base64,${pdfBase64}`,
+ contentType: 'application/pdf',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ return text;
+});
+
+/**
+ * This flow demonstrates PDF document processing with a URL reference.
+ * Note: This requires the PDF to be hosted at a publicly accessible URL.
+ */
+ai.defineFlow('stable-pdf-url', async () => {
+ // Example: Using a publicly hosted PDF URL
+ // In a real application, you would use your own hosted PDF
+ const pdfUrl =
+ 'https://assets.anthropic.com/m/1cd9d098ac3e6467/original/Claude-3-Model-Card-October-Addendum.pdf';
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ text: 'Summarize the key points from this document.',
+ },
+ {
+ media: {
+ url: pdfUrl,
+ contentType: 'application/pdf',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ return text;
+});
+
+/**
+ * This flow demonstrates analyzing specific aspects of a PDF document.
+ * Claude can understand both text and visual elements (charts, tables, images) in PDFs.
+ */
+ai.defineFlow('stable-pdf-analysis', async () => {
+ const pdfPath = path.join(__dirname, 'attention-first-page.pdf');
+ const pdfBuffer = fs.readFileSync(pdfPath);
+ const pdfBase64 = pdfBuffer.toString('base64');
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ text: 'Analyze this document and provide:\n1. The main topic or subject\n2. Any key technical concepts mentioned\n3. Any visual elements (charts, tables, diagrams) if present',
+ },
+ {
+ media: {
+ url: `data:application/pdf;base64,${pdfBase64}`,
+ contentType: 'application/pdf',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ return text;
+});
diff --git a/js/testapps/anthropic/src/stable/text-plain.ts b/js/testapps/anthropic/src/stable/text-plain.ts
new file mode 100644
index 0000000000..0b290d53e6
--- /dev/null
+++ b/js/testapps/anthropic/src/stable/text-plain.ts
@@ -0,0 +1,83 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { anthropic } from '@genkit-ai/anthropic';
+import { genkit } from 'genkit';
+
+const ai = genkit({
+ plugins: [anthropic()],
+});
+
+/**
+ * This flow demonstrates the error that occurs when trying to use text/plain
+ * files as media. The plugin will throw a helpful error message guiding users
+ * to use text content instead.
+ *
+ * Error message: "Unsupported media type: text/plain. Text files should be sent
+ * as text content in the message, not as media. For example, use { text: '...' }
+ * instead of { media: { url: '...', contentType: 'text/plain' } }"
+ */
+ai.defineFlow('stable-text-plain-error', async () => {
+ try {
+ await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ media: {
+ url: 'data:text/plain;base64,SGVsbG8gV29ybGQ=',
+ contentType: 'text/plain',
+ },
+ },
+ ],
+ },
+ ],
+ });
+ return 'Unexpected: Should have thrown an error';
+ } catch (error: any) {
+ return {
+ error: error.message,
+ note: 'This demonstrates the helpful error message for text/plain files',
+ };
+ }
+});
+
+/**
+ * This flow demonstrates the correct way to send text content.
+ * Instead of using media with text/plain, use the text field directly.
+ */
+ai.defineFlow('stable-text-plain-correct', async () => {
+ // Read the text content (in a real app, you'd read from a file)
+ const textContent = 'Hello World\n\nThis is a text file content.';
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ text: `Please summarize this text file content:\n\n${textContent}`,
+ },
+ ],
+ },
+ ],
+ });
+
+ return text;
+});
diff --git a/js/testapps/anthropic/src/stable/webp.ts b/js/testapps/anthropic/src/stable/webp.ts
new file mode 100644
index 0000000000..f8a861024b
--- /dev/null
+++ b/js/testapps/anthropic/src/stable/webp.ts
@@ -0,0 +1,95 @@
+/**
+ * Copyright 2025 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import { anthropic } from '@genkit-ai/anthropic';
+import { genkit } from 'genkit';
+
+const ai = genkit({
+ plugins: [anthropic()],
+});
+
+/**
+ * This flow demonstrates WEBP image handling with matching contentType.
+ * Both the data URL and the contentType field specify image/webp.
+ */
+ai.defineFlow('stable-webp-matching', async () => {
+ // Minimal valid WEBP image (1x1 pixel, transparent)
+ // In a real app, you'd load an actual WEBP image file
+ const webpImageData =
+ '';
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ { text: 'Describe this image:' },
+ {
+ media: {
+ url: webpImageData,
+ contentType: 'image/webp',
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ return text;
+});
+
+/**
+ * This flow demonstrates the fix for WEBP images with mismatched contentType.
+ * Even if contentType says 'image/png', the plugin will use 'image/webp' from
+ * the data URL, preventing API validation errors.
+ *
+ * This fix ensures that the media_type sent to Anthropic matches the actual
+ * image data, which is critical for WEBP images that were previously causing
+ * "Image does not match the provided media type" errors.
+ */
+ai.defineFlow('stable-webp-mismatched', async () => {
+ // Minimal valid WEBP image (1x1 pixel, transparent)
+ const webpImageData =
+ '';
+
+ const { text } = await ai.generate({
+ model: anthropic.model('claude-sonnet-4-5'),
+ messages: [
+ {
+ role: 'user',
+ content: [
+ {
+ text: 'Describe this image (note: contentType is wrong but data URL is correct):',
+ },
+ {
+ media: {
+ // Data URL says WEBP, but contentType says PNG
+ // The plugin will use WEBP from the data URL (the fix)
+ url: webpImageData,
+ contentType: 'image/png', // This mismatch is handled correctly
+ },
+ },
+ ],
+ },
+ ],
+ });
+
+ return {
+ result: text,
+ note: 'The plugin correctly used image/webp from the data URL, not image/png from contentType',
+ };
+});
diff --git a/js/testapps/anthropic/tsconfig.json b/js/testapps/anthropic/tsconfig.json
new file mode 100644
index 0000000000..efbb566bf7
--- /dev/null
+++ b/js/testapps/anthropic/tsconfig.json
@@ -0,0 +1,14 @@
+{
+ "compileOnSave": true,
+ "include": ["src"],
+ "compilerOptions": {
+ "module": "commonjs",
+ "noImplicitReturns": true,
+ "outDir": "lib",
+ "sourceMap": true,
+ "strict": true,
+ "target": "es2017",
+ "skipLibCheck": true,
+ "esModuleInterop": true
+ }
+}