mirror of
https://github.com/n8n-io/n8n.git
synced 2026-06-02 09:47:00 +02:00
394 lines
12 KiB
TypeScript
394 lines
12 KiB
TypeScript
import { promises as fs } from 'fs';
|
|
import type { Expectation } from 'mockserver-client';
|
|
import { join } from 'path';
|
|
|
|
import { test as base, expect as baseExpect } from '../../../fixtures/base';
|
|
|
|
const ANTHROPIC_API_KEY = process.env.ANTHROPIC_API_KEY ?? 'mock-anthropic-api-key';
|
|
const HAS_REAL_API_KEY = !!process.env.ANTHROPIC_API_KEY;
|
|
const EXPECTATIONS_DIR = './expectations';
|
|
const INSTANCE_AGENT_SYSTEM_PROMPT_ANCHOR = 'You are the n8n Instance Agent';
|
|
export const SKIP_PROXY_SETUP_ANNOTATION = 'skip-proxy-setup';
|
|
const SYSTEM_PROMPT_ANCHORS = [
|
|
INSTANCE_AGENT_SYSTEM_PROMPT_ANCHOR,
|
|
'You are the n8n Workflow Planner',
|
|
'You are an expert n8n workflow builder',
|
|
'You generate a short descriptive title for a conversation',
|
|
] as const;
|
|
const LEGACY_SYSTEM_ARRAY_PREFIX =
|
|
/\\\[\\\{"type":"text","text":"(?=You are the n8n Instance Agent)/g;
|
|
const LEGACY_SYSTEM_STRING_PREFIX = '[{"type":"text","text":"';
|
|
const BODY_REGEX_WILDCARD = '[\\s\\S]*';
|
|
|
|
function slugify(text: string): string {
|
|
return text
|
|
.toLowerCase()
|
|
.replace(/[^a-z0-9]+/g, '-')
|
|
.replace(/(^-|-$)/g, '');
|
|
}
|
|
|
|
type TestInfoWithSlug = {
|
|
title: string;
|
|
annotations: Array<{ type: string; description?: string }>;
|
|
};
|
|
|
|
export function getInstanceAiTestSlug(testInfo: TestInfoWithSlug): string {
|
|
const expectationSlug = testInfo.annotations.find(
|
|
(annotation) => annotation.type === 'expectation-slug' && annotation.description,
|
|
)?.description;
|
|
|
|
return expectationSlug ?? slugify(testInfo.title);
|
|
}
|
|
|
|
async function loadTraceFile(folder: string): Promise<unknown[]> {
|
|
const filePath = join(EXPECTATIONS_DIR, folder, 'trace.jsonl');
|
|
try {
|
|
const content = await fs.readFile(filePath, 'utf8');
|
|
return content
|
|
.split('\n')
|
|
.filter((line) => line.trim().length > 0)
|
|
.map((line) => JSON.parse(line) as unknown);
|
|
} catch {
|
|
return [];
|
|
}
|
|
}
|
|
|
|
async function writeTraceFile(folder: string, events: unknown[]): Promise<void> {
|
|
const targetDir = join(EXPECTATIONS_DIR, folder);
|
|
await fs.mkdir(targetDir, { recursive: true });
|
|
const filePath = join(targetDir, 'trace.jsonl');
|
|
const jsonl = events.map((e) => JSON.stringify(e)).join('\n') + '\n';
|
|
await fs.writeFile(filePath, jsonl);
|
|
}
|
|
|
|
type AnthropicMessage = {
|
|
role?: unknown;
|
|
content?: unknown;
|
|
};
|
|
|
|
type AnthropicContentBlock = {
|
|
type?: unknown;
|
|
text?: unknown;
|
|
name?: unknown;
|
|
input?: unknown;
|
|
};
|
|
|
|
function escapeRegex(value: string): string {
|
|
return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
}
|
|
|
|
function asContentBlocks(content: unknown): AnthropicContentBlock[] {
|
|
if (typeof content === 'string') return [{ type: 'text', text: content }];
|
|
if (!Array.isArray(content)) return [];
|
|
return content.filter(
|
|
(block): block is AnthropicContentBlock =>
|
|
typeof block === 'object' && block !== null && !Array.isArray(block),
|
|
);
|
|
}
|
|
|
|
function getStableTextAnchor(text: string): string | undefined {
|
|
const trimmed = text.trimStart();
|
|
if (!trimmed) return undefined;
|
|
|
|
const tagMatch = /^<[a-z-]+/.exec(trimmed);
|
|
if (tagMatch) return escapeRegex(tagMatch[0]);
|
|
|
|
return escapeRegex(JSON.stringify(trimmed.slice(0, 120)).slice(1, -1));
|
|
}
|
|
|
|
function getToolUseAnchor(block: AnthropicContentBlock): string | undefined {
|
|
if (block.type !== 'tool_use' || typeof block.name !== 'string') return undefined;
|
|
|
|
const toolNameMatcher = `"type"\\s*:\\s*"tool_use"[\\s\\S]{0,300}"name"\\s*:\\s*"${escapeRegex(block.name)}"`;
|
|
const input = block.input;
|
|
if (typeof input !== 'object' || input === null || Array.isArray(input)) {
|
|
return toolNameMatcher;
|
|
}
|
|
|
|
const action = Reflect.get(input, 'action');
|
|
if (typeof action !== 'string') return toolNameMatcher;
|
|
|
|
return `${toolNameMatcher}[\\s\\S]{0,500}"action"\\s*:\\s*"${escapeRegex(action)}"`;
|
|
}
|
|
|
|
function getLatestMessageAnchor(messages: AnthropicMessage[] | undefined): string | undefined {
|
|
if (!messages) return undefined;
|
|
|
|
for (let messageIndex = messages.length - 1; messageIndex >= 0; messageIndex--) {
|
|
const blocks = asContentBlocks(messages[messageIndex]?.content);
|
|
for (let blockIndex = blocks.length - 1; blockIndex >= 0; blockIndex--) {
|
|
const block = blocks[blockIndex];
|
|
if (block.type === 'text' && typeof block.text === 'string') {
|
|
const textAnchor = getStableTextAnchor(block.text);
|
|
if (textAnchor) return textAnchor;
|
|
}
|
|
|
|
const toolUseAnchor = getToolUseAnchor(block);
|
|
if (toolUseAnchor) return toolUseAnchor;
|
|
}
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
function createAnthropicBodyMatcher(raw: string): { type: 'REGEX'; regex: string } | undefined {
|
|
const parsed = JSON.parse(raw) as {
|
|
system?: string | unknown[];
|
|
messages?: AnthropicMessage[];
|
|
};
|
|
|
|
const system =
|
|
typeof parsed.system === 'string'
|
|
? parsed.system
|
|
: Array.isArray(parsed.system)
|
|
? JSON.stringify(parsed.system)
|
|
: undefined;
|
|
if (!system) return undefined;
|
|
|
|
const anchorIndex = system.indexOf(INSTANCE_AGENT_SYSTEM_PROMPT_ANCHOR);
|
|
const systemSnippetStart = anchorIndex >= 0 ? anchorIndex : 0;
|
|
const systemSnippet = escapeRegex(system.slice(systemSnippetStart, systemSnippetStart + 80));
|
|
const latestMessageAnchor = getLatestMessageAnchor(parsed.messages);
|
|
const matcher = latestMessageAnchor
|
|
? `${systemSnippet}[\\s\\S]*${latestMessageAnchor}`
|
|
: systemSnippet;
|
|
|
|
return {
|
|
type: 'REGEX',
|
|
regex: `[\\s\\S]*${matcher}[\\s\\S]*`,
|
|
};
|
|
}
|
|
|
|
type BodyMatcher = {
|
|
type?: unknown;
|
|
regex?: unknown;
|
|
subString?: unknown;
|
|
[key: string]: unknown;
|
|
};
|
|
|
|
function isBodyMatcher(body: unknown): body is BodyMatcher {
|
|
return typeof body === 'object' && body !== null && !Array.isArray(body);
|
|
}
|
|
|
|
function stripRecordedSystemPromptAnchor(regex: string): string {
|
|
const anchorIndex = SYSTEM_PROMPT_ANCHORS.reduce<number>((nearest, anchor) => {
|
|
const index = regex.indexOf(anchor);
|
|
if (index < 0) return nearest;
|
|
if (nearest < 0) return index;
|
|
return Math.min(nearest, index);
|
|
}, -1);
|
|
|
|
if (anchorIndex < 0) return regex;
|
|
|
|
const latestTurnAnchorIndex = regex.indexOf(BODY_REGEX_WILDCARD, anchorIndex);
|
|
if (latestTurnAnchorIndex < 0) return regex;
|
|
|
|
return `${BODY_REGEX_WILDCARD}${regex.slice(latestTurnAnchorIndex + BODY_REGEX_WILDCARD.length)}`;
|
|
}
|
|
|
|
function loosenRecordedInstanceAiPromptMatcher(expectation: Expectation): Expectation {
|
|
const body = (expectation.httpRequest as { body?: unknown } | undefined)?.body;
|
|
if (!isBodyMatcher(body)) return expectation;
|
|
|
|
if (body.type === 'REGEX' && typeof body.regex === 'string') {
|
|
body.regex = stripRecordedSystemPromptAnchor(
|
|
body.regex.replace(LEGACY_SYSTEM_ARRAY_PREFIX, ''),
|
|
);
|
|
}
|
|
|
|
const stringMatcher = body['string'];
|
|
if (
|
|
body.type === 'STRING' &&
|
|
typeof stringMatcher === 'string' &&
|
|
stringMatcher.startsWith(`${LEGACY_SYSTEM_STRING_PREFIX}${INSTANCE_AGENT_SYSTEM_PROMPT_ANCHOR}`)
|
|
) {
|
|
body['string'] = INSTANCE_AGENT_SYSTEM_PROMPT_ANCHOR;
|
|
body.subString = true;
|
|
}
|
|
|
|
return expectation;
|
|
}
|
|
|
|
type InstanceAiFixtures = {
|
|
anthropicApiKey: string;
|
|
instanceAiProxySetup: undefined;
|
|
};
|
|
|
|
async function safeFetch(input: string, init: RequestInit = {}): Promise<Response | undefined> {
|
|
try {
|
|
return await fetch(input, { ...init, signal: AbortSignal.timeout(10_000) });
|
|
} catch {
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
export const instanceAiTestConfig = {
|
|
timezoneId: 'America/New_York',
|
|
capability: {
|
|
services: ['proxy'],
|
|
env: {
|
|
N8N_ENABLED_MODULES: 'instance-ai',
|
|
N8N_INSTANCE_AI_MODEL: 'anthropic/claude-sonnet-4-6',
|
|
N8N_INSTANCE_AI_MODEL_API_KEY: ANTHROPIC_API_KEY,
|
|
N8N_INSTANCE_AI_LOCAL_GATEWAY_DISABLED: 'true',
|
|
// Prevent community-node-types requests to api-staging.n8n.io
|
|
// from polluting proxy recordings
|
|
N8N_VERIFIED_PACKAGES_ENABLED: 'false',
|
|
},
|
|
},
|
|
} as const;
|
|
|
|
export const test = base.extend<InstanceAiFixtures>({
|
|
anthropicApiKey: async ({}, use) => {
|
|
await use(ANTHROPIC_API_KEY);
|
|
},
|
|
|
|
instanceAiProxySetup: [
|
|
async ({ n8nContainer, backendUrl }, use, testInfo) => {
|
|
// Local-build mode (no Docker container) — skip all proxy setup.
|
|
// LLM calls go straight to Anthropic, no recording or replay.
|
|
if (!n8nContainer) {
|
|
await use(undefined);
|
|
return;
|
|
}
|
|
|
|
const skipsProxySetup = testInfo.annotations.some(
|
|
(annotation) => annotation.type === SKIP_PROXY_SETUP_ANNOTATION,
|
|
);
|
|
if (skipsProxySetup) {
|
|
await use(undefined);
|
|
return;
|
|
}
|
|
const services = n8nContainer.services;
|
|
const testSlug = getInstanceAiTestSlug(testInfo);
|
|
const folder = `instance-ai/${testSlug}`;
|
|
|
|
// Wipe instance-ai threads, per-thread in-memory state, background tasks,
|
|
// and user workflows before clearing the proxy so cleanup traffic from a
|
|
// previous test cannot be captured into this test's recording.
|
|
await safeFetch(`${backendUrl}/rest/instance-ai/test/reset`, { method: 'POST' });
|
|
|
|
await services.proxy.clearAllExpectations();
|
|
|
|
// Install a success response for Slack's `users.profile.get` — the
|
|
// backend's `POST /credentials/test` endpoint calls this when testing
|
|
// a Slack API credential. Seed tokens in these tests are intentionally
|
|
// fake, so hitting the real Slack API returns `invalid_auth` and the
|
|
// frontend's `buildNodeCredentials` then drops the credential from
|
|
// the apply payload (see `getEffectiveCredTestResult` in
|
|
// `useCredentialTesting.ts`). By registering this mock FIRST (before
|
|
// `loadExpectations`), mockserver's FIFO matching serves it ahead of
|
|
// any stale recorded Slack response.
|
|
await services.proxy.createExpectation({
|
|
httpRequest: { method: 'GET', path: '/api/users.profile.get' },
|
|
httpResponse: {
|
|
statusCode: 200,
|
|
headers: { 'Content-Type': ['application/json'] },
|
|
body: JSON.stringify({
|
|
ok: true,
|
|
profile: {
|
|
real_name: 'E2E Test User',
|
|
email: 'e2e@example.test',
|
|
},
|
|
}),
|
|
},
|
|
times: { unlimited: true },
|
|
});
|
|
|
|
// Recording mode: real API key, not CI → proxy forwards to real API,
|
|
// backend records tool I/O. Replay mode: load existing expectations
|
|
// and trace events so the proxy serves recorded responses and the
|
|
// backend remaps tool IDs.
|
|
const isRecording = !process.env.CI && HAS_REAL_API_KEY;
|
|
|
|
if (!isRecording) {
|
|
const traceEvents = await loadTraceFile(folder);
|
|
await services.proxy.loadExpectations(folder, {
|
|
sequential: true,
|
|
repeatLastResponse: false,
|
|
transform: loosenRecordedInstanceAiPromptMatcher,
|
|
});
|
|
|
|
await safeFetch(`${backendUrl}/rest/instance-ai/test/tool-trace`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({
|
|
slug: testSlug,
|
|
...(traceEvents.length > 0 ? { events: traceEvents } : {}),
|
|
}),
|
|
});
|
|
} else {
|
|
await safeFetch(`${backendUrl}/rest/instance-ai/test/tool-trace`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({ slug: testSlug }),
|
|
});
|
|
}
|
|
|
|
await use(undefined);
|
|
|
|
if (!process.env.CI && HAS_REAL_API_KEY) {
|
|
await services.proxy.recordExpectations(folder, {
|
|
clearDir: true,
|
|
transform: (expectation) => {
|
|
const response = expectation.httpResponse as {
|
|
headers?: Record<string, string[]>;
|
|
};
|
|
|
|
if (response?.headers) {
|
|
delete response.headers['anthropic-organization-id'];
|
|
}
|
|
|
|
// Keep a minimal body matcher so the proxy can distinguish
|
|
// between agent type and stable turn context without matching
|
|
// volatile tool output such as workflow and execution IDs.
|
|
const request = expectation.httpRequest as {
|
|
// eslint-disable-next-line id-denylist -- `string` is MockServer's body matcher field name
|
|
body?: { type?: string; string?: string; json?: Record<string, unknown> };
|
|
};
|
|
if (request?.body) {
|
|
const raw =
|
|
request.body['string'] ??
|
|
(request.body.json ? JSON.stringify(request.body.json) : undefined);
|
|
if (raw) {
|
|
try {
|
|
const bodyMatcher = createAnthropicBodyMatcher(raw);
|
|
if (bodyMatcher) {
|
|
request.body = bodyMatcher as unknown as typeof request.body;
|
|
} else {
|
|
delete request.body;
|
|
}
|
|
} catch {
|
|
delete request.body;
|
|
}
|
|
} else {
|
|
delete request.body;
|
|
}
|
|
}
|
|
|
|
return expectation;
|
|
},
|
|
});
|
|
|
|
const traceResponse = await safeFetch(
|
|
`${backendUrl}/rest/instance-ai/test/tool-trace/${testSlug}`,
|
|
);
|
|
if (traceResponse?.ok) {
|
|
const body = (await traceResponse.json()) as { data?: { events?: unknown[] } };
|
|
const events = body?.data?.events ?? [];
|
|
if (events.length > 0) {
|
|
await writeTraceFile(folder, events);
|
|
}
|
|
}
|
|
}
|
|
|
|
await safeFetch(`${backendUrl}/rest/instance-ai/test/tool-trace/${testSlug}`, {
|
|
method: 'DELETE',
|
|
});
|
|
},
|
|
{ auto: true },
|
|
],
|
|
});
|
|
|
|
export const expect = baseExpect;
|