open-design/apps/daemon/tests/connectors-routes.test.ts
marco 5dd70b5016
Some checks failed
ci / Validate workspace (push) Successful in 12m32s
landing-page-ci / Validate landing page (push) Successful in 9m41s
landing-page-deploy / Deploy landing page (push) Failing after 5m23s
github-metrics / Generate repository metrics SVG (push) Failing after 2m3s
refresh-contributors-wall / Refresh contributors wall cache bust (push) Failing after 11s
Initial import: open-design source for helix-mind.ai distribution
This repository contains the open-design daemon CLI source code, built
and packaged at https://helix-mind.ai/cli/open-design/latest.tgz for use
by the HelixMind /design slash command.

Licenses: Apache-2.0 (root) + MIT (skills/*)
2026-05-06 20:50:24 +02:00

557 lines
26 KiB
TypeScript

// @ts-nocheck
import { request as httpRequest } from 'node:http';
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
import { startServer } from '../src/server.js';
import { ComposioConnectorProvider, composioConnectorProvider, getStaticComposioCatalogDefinitions } from '../src/connectors/composio.js';
import { readComposioConfig, writeComposioConfig } from '../src/connectors/composio-config.js';
import { deleteConnectorCredentialsByProvider } from '../src/connectors/service.js';
import { CHAT_TOOL_ENDPOINTS, CHAT_TOOL_OPERATIONS, toolTokenRegistry } from '../src/tool-tokens.js';
let server;
let baseUrl;
let originalComposioConfig;
const originalFetch = globalThis.fetch;
let lastComposioLinkRequest;
let lastComposioAuthConfigRequest;
let composioDiscoveryRequestCounts;
function composioJson(body, status = 200) {
return new Response(JSON.stringify(body), {
status,
headers: { 'content-type': 'application/json' },
});
}
function createDeferred() {
let resolve;
const promise = new Promise((innerResolve) => {
resolve = innerResolve;
});
return { promise, resolve };
}
function mockComposioFetch(options = {}) {
const {
authConfigs = [{ id: 'ac_github', status: 'ENABLED', toolkit: { slug: 'github' } }],
createAuthConfigResponse,
delayFirstAuthConfigs,
delayFirstToolkits,
linkResponse = { connected_account_id: 'ca_github', status: 'ACTIVE', account_label: 'octocat@example.com' },
} = options;
composioDiscoveryRequestCounts = { authConfigs: 0, createdAuthConfigs: 0, toolkits: 0, tools: 0 };
vi.stubGlobal('fetch', async (input, init) => {
const url = typeof input === 'string' ? input : input instanceof URL ? input.toString() : input.url;
if (url.startsWith('http://127.0.0.1:') || url.startsWith('http://localhost:')) {
return originalFetch(input, init);
}
const parsed = new URL(url);
if (parsed.pathname === '/api/v3/auth_configs') {
composioDiscoveryRequestCounts.authConfigs += 1;
if (delayFirstAuthConfigs && composioDiscoveryRequestCounts.authConfigs === 1) {
delayFirstAuthConfigs.started.resolve();
await delayFirstAuthConfigs.release.promise;
}
return composioJson({ items: authConfigs });
}
if (parsed.pathname === '/api/v3.1/auth_configs' && init?.method === 'POST') {
composioDiscoveryRequestCounts.createdAuthConfigs += 1;
lastComposioAuthConfigRequest = typeof init?.body === 'string' ? JSON.parse(init.body) : undefined;
const toolkitSlug = lastComposioAuthConfigRequest?.toolkit?.slug ?? 'GITHUB';
return composioJson(createAuthConfigResponse ?? { id: `ac_${String(toolkitSlug).toLowerCase()}`, status: 'ENABLED', toolkit: { slug: toolkitSlug } });
}
if (parsed.pathname === '/api/v3.1/toolkits') {
composioDiscoveryRequestCounts.toolkits += 1;
if (delayFirstToolkits && composioDiscoveryRequestCounts.toolkits === 1) {
delayFirstToolkits.started.resolve();
await delayFirstToolkits.release.promise;
}
return composioJson({ items: [{ slug: 'github', name: 'GitHub', description: 'GitHub toolkit', categories: [{ name: 'Developer' }] }] });
}
if (parsed.pathname === '/api/v3.1/tools' && parsed.searchParams.get('toolkit_slug') === 'github') {
composioDiscoveryRequestCounts.tools += 1;
return composioJson({ items: [{ slug: 'GITHUB_SEARCH_REPOSITORIES', name: 'Search repositories', description: 'Search public and private repositories', toolkit: { slug: 'github' }, input_parameters: { type: 'object', properties: { query: { type: 'string' } }, required: ['query'], additionalProperties: false }, tags: ['read'] }] });
}
if (parsed.pathname === '/api/v3.1/tools' && parsed.searchParams.get('toolkit_slug') === 'slack') {
composioDiscoveryRequestCounts.tools += 1;
return composioJson({ items: [
{ slug: 'SLACK_LIST_CHANNELS', name: 'List channels', description: 'List Slack channels', toolkit: { slug: 'slack' }, input_parameters: { type: 'object', additionalProperties: false }, tags: ['read'] },
{ slug: 'SLACK_SEND_MESSAGE', name: 'Send message', description: 'Send a Slack message', toolkit: { slug: 'slack' }, input_parameters: { type: 'object', additionalProperties: true }, tags: ['write'] },
] });
}
if (parsed.pathname === '/api/v3.1/connected_accounts/link') {
lastComposioLinkRequest = typeof init?.body === 'string' ? JSON.parse(init.body) : undefined;
return composioJson(linkResponse);
}
if (parsed.pathname === '/api/v3/connected_accounts/ca_github') {
return composioJson({ connected_account_id: 'ca_github', status: 'ACTIVE', account_label: 'octocat@example.com', toolkit: { slug: 'github' }, auth_config: { id: 'ac_github' } });
}
if (parsed.pathname === '/api/v3/connected_accounts/ca_slack') {
return composioJson({ connected_account_id: 'ca_slack', status: 'ACTIVE', account_label: 'slack@example.com', toolkit: { slug: 'slack' }, auth_config: { id: 'ac_slack' } });
}
if (parsed.pathname === '/api/v3.1/tools/execute/GITHUB_SEARCH_REPOSITORIES') {
return composioJson({ successful: true, data: { results: [] }, log_id: 'log_1' });
}
if (parsed.pathname === '/api/v3/connected_accounts/ca_github' && init?.method === 'DELETE') {
return composioJson({ ok: true });
}
return composioJson({ message: `Unhandled Composio mock: ${url}` }, 404);
});
}
beforeEach(async () => {
originalComposioConfig = readComposioConfig();
lastComposioLinkRequest = undefined;
lastComposioAuthConfigRequest = undefined;
mockComposioFetch();
const started = await startServer({ port: 0, returnServer: true });
server = started.server;
baseUrl = started.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
});
afterEach(async () => {
deleteConnectorCredentialsByProvider('composio');
writeComposioConfig(originalComposioConfig ?? { apiKey: '' });
composioConnectorProvider.clearDiscoveryCache();
await new Promise((resolve, reject) => {
if (!server) return resolve(undefined);
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
server = undefined;
toolTokenRegistry.clear();
vi.unstubAllGlobals();
vi.useRealTimers();
});
async function jsonFetch(url, init) {
const response = await fetch(url, init);
return { status: response.status, body: await response.json() };
}
async function requestWithHostHeader(method, url, host, body) {
const target = new URL(url);
return await new Promise((resolve, reject) => {
const req = httpRequest(
{
protocol: target.protocol,
hostname: target.hostname,
port: target.port,
path: target.pathname + target.search,
method,
headers: {
host,
...(body === undefined ? {} : { 'content-type': 'application/json' }),
},
},
(res) => {
const chunks = [];
res.on('data', (chunk) => chunks.push(chunk));
res.on('end', () => {
resolve({
status: res.statusCode,
body: Buffer.concat(chunks).toString('utf8'),
});
});
},
);
req.on('error', reject);
req.end(body === undefined ? undefined : JSON.stringify(body));
});
}
async function postWithHostHeader(url, host) {
return requestWithHostHeader('POST', url, host);
}
async function putWithHostHeader(url, host, body) {
return requestWithHostHeader('PUT', url, host, body);
}
function mintConnectorToolToken(projectId = 'connector-route-project', runId = 'connector-route-run', overrides = {}) {
return toolTokenRegistry.mint({
projectId,
runId,
allowedEndpoints: CHAT_TOOL_ENDPOINTS,
allowedOperations: CHAT_TOOL_OPERATIONS,
...overrides,
}).token;
}
describe('connector routes', () => {
it('lists catalog connectors without hitting Composio discovery endpoints', async () => {
const response = await jsonFetch(`${baseUrl}/api/connectors`);
expect(response.status).toBe(200);
expect(response.body.connectors.map((connector) => connector.id)).toEqual(expect.arrayContaining(['github', 'notion', 'google_drive', 'slack', 'zoom']));
expect(response.body.connectors.length).toBeGreaterThan(100);
const github = response.body.connectors.find((connector) => connector.id === 'github');
expect(github).toMatchObject({
id: 'github',
name: 'GitHub',
provider: 'composio',
auth: { provider: 'composio', configured: false },
});
expect(github.tools).toEqual(expect.arrayContaining([expect.objectContaining({ name: 'github.github_search_repositories' })]));
expect(response.body.connectors.find((connector) => connector.id === 'google_drive')).toMatchObject({
id: 'google_drive',
auth: { provider: 'composio', configured: false },
});
expect(response.body.connectors.find((connector) => connector.id === 'notion')).toMatchObject({
id: 'notion',
auth: { provider: 'composio', configured: false },
});
expect(composioDiscoveryRequestCounts).toEqual({ authConfigs: 0, createdAuthConfigs: 0, toolkits: 0, tools: 0 });
});
it('reuses Composio discovery results across consecutive discovery requests', async () => {
const first = await jsonFetch(`${baseUrl}/api/connectors/discovery`);
const second = await jsonFetch(`${baseUrl}/api/connectors/discovery`);
expect(first.status).toBe(200);
expect(second.status).toBe(200);
expect(first.body.connectors.map((connector) => connector.id)).toEqual(expect.arrayContaining(['github', 'notion', 'google_drive', 'slack', 'zoom']));
expect(second.body.connectors.map((connector) => connector.id)).toEqual(expect.arrayContaining(['github', 'notion', 'google_drive', 'slack', 'zoom']));
expect(first.body.connectors.find((connector) => connector.id === 'slack')?.tools).toEqual(expect.arrayContaining([expect.objectContaining({ name: 'slack.slack_list_channels' })]));
expect(first.body.meta).toMatchObject({ provider: 'composio' });
expect(composioDiscoveryRequestCounts).toEqual({ authConfigs: 1, createdAuthConfigs: 0, toolkits: 1, tools: 2 });
});
it('returns connector statuses by connectorId', async () => {
await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
const response = await jsonFetch(`${baseUrl}/api/connectors/status`);
expect(response.status).toBe(200);
expect(response.body.statuses.github).toMatchObject({ status: 'connected', accountLabel: 'octocat@example.com' });
expect(response.body.statuses.notion).toMatchObject({ status: 'available' });
expect(response.body.statuses.google_drive).toMatchObject({ status: 'available' });
});
it('returns static catalog connectors even when Composio auth configs are empty', async () => {
await new Promise((resolve, reject) => {
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
mockComposioFetch({
authConfigs: [],
linkResponse: { connected_account_id: 'ca_slack', status: 'ACTIVE', account_label: 'slack@example.com' },
});
composioConnectorProvider.clearDiscoveryCache();
const started = await startServer({ port: 0, returnServer: true });
server = started.server;
baseUrl = started.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
const response = await jsonFetch(`${baseUrl}/api/connectors`);
expect(response.status).toBe(200);
expect(response.body.connectors.map((connector) => connector.id)).toEqual(expect.arrayContaining(['github', 'notion', 'google_drive', 'slack', 'zoom']));
expect(response.body.connectors.every((connector) => connector.auth?.configured === false)).toBe(true);
});
it('returns static catalog connectors before Composio is configured', async () => {
writeComposioConfig({ apiKey: '' });
composioConnectorProvider.clearDiscoveryCache();
const response = await jsonFetch(`${baseUrl}/api/connectors`);
expect(response.status).toBe(200);
expect(response.body.connectors.map((connector) => connector.id)).toEqual(expect.arrayContaining(['github', 'notion', 'google_drive', 'slack', 'zoom']));
expect(response.body.connectors.every((connector) => connector.auth?.configured === false)).toBe(true);
});
it('returns connector detail and 404 for unknown connectors', async () => {
const detail = await jsonFetch(`${baseUrl}/api/connectors/github`);
expect(detail.status).toBe(200);
expect(detail.body.connector).toMatchObject({ id: 'github', name: 'GitHub' });
const missing = await jsonFetch(`${baseUrl}/api/connectors/missing`);
expect(missing.status).toBe(404);
expect(missing.body.error.code).toBe('CONNECTOR_NOT_FOUND');
});
it('connects and disconnects a Composio connector', async () => {
const connect = await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
expect(connect.status).toBe(200);
expect(connect.body.connector).toMatchObject({ id: 'github', status: 'connected', accountLabel: 'octocat@example.com' });
const disconnect = await jsonFetch(`${baseUrl}/api/connectors/github/connection`, { method: 'DELETE' });
expect(disconnect.status).toBe(200);
expect(disconnect.body.connector).toMatchObject({ id: 'github', status: 'available' });
});
it('rejects cross-origin connector connect requests before starting provider auth', async () => {
const connect = await jsonFetch(`${baseUrl}/api/connectors/github/connect`, {
method: 'POST',
headers: { Origin: 'https://attacker.example' },
});
expect(connect.status).toBe(403);
expect(JSON.stringify(connect.body.error)).toContain('Cross-origin');
expect(lastComposioLinkRequest).toBeUndefined();
});
it('rejects Composio config updates from non-loopback daemon hosts', async () => {
const response = await putWithHostHeader(`${baseUrl}/api/connectors/composio/config`, 'example.com', { apiKey: 'cmp_remote' });
expect(response.status).toBe(403);
expect(response.body).toContain('request host must be a loopback daemon address');
expect(readComposioConfig().apiKey).toBe('cmp_test');
});
it('clears Composio connector credentials when rotating to a key with the same tail', async () => {
const connect = await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
expect(connect.status).toBe(200);
expect(connect.body.connector).toMatchObject({ id: 'github', status: 'connected' });
const rotate = await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_rotated_test' }),
});
const statuses = await jsonFetch(`${baseUrl}/api/connectors/status`);
expect(rotate.status).toBe(200);
expect(rotate.body).toMatchObject({ configured: true, apiKeyTail: 'test' });
expect(statuses.body.statuses.github).toMatchObject({ status: 'available' });
});
it('creates a managed Composio auth config when connecting an unconfigured connector', async () => {
await new Promise((resolve, reject) => {
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
mockComposioFetch({
authConfigs: [],
linkResponse: { connected_account_id: 'ca_slack', status: 'ACTIVE', account_label: 'slack@example.com' },
});
composioConnectorProvider.clearDiscoveryCache();
const started = await startServer({ port: 0, returnServer: true });
server = started.server;
baseUrl = started.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
const connect = await jsonFetch(`${baseUrl}/api/connectors/slack/connect`, { method: 'POST' });
const token = mintConnectorToolToken('connector-auto-auth-project', 'connector-auto-auth-run');
const tools = await jsonFetch(`${baseUrl}/api/tools/connectors/list`, {
headers: { Authorization: `Bearer ${token}` },
});
expect(connect.status).toBe(200);
expect(connect.body.connector).toMatchObject({ id: 'slack', status: 'connected', auth: { configured: true } });
expect(connect.body.connector.tools).toEqual(expect.arrayContaining([
expect.objectContaining({ name: 'slack.slack_list_channels' }),
expect.objectContaining({ name: 'slack.slack_send_message' }),
]));
expect(lastComposioAuthConfigRequest).toEqual({
toolkit: { slug: 'SLACK' },
auth_config: { type: 'use_composio_managed_auth' },
});
expect(lastComposioLinkRequest).toMatchObject({ auth_config_id: 'ac_slack' });
expect(tools.status).toBe(200);
expect(tools.body.connectors.find((connector) => connector.id === 'slack')?.tools).toEqual([
expect.objectContaining({ name: 'slack.slack_list_channels' }),
]);
expect(composioDiscoveryRequestCounts).toMatchObject({ authConfigs: 2, createdAuthConfigs: 1 });
});
it('rejects immediate Composio connections when account validation does not match the connector', async () => {
await new Promise((resolve, reject) => {
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
mockComposioFetch({
authConfigs: [],
linkResponse: { connected_account_id: 'ca_github', status: 'ACTIVE', account_label: 'octocat@example.com' },
});
composioConnectorProvider.clearDiscoveryCache();
const started = await startServer({ port: 0, returnServer: true });
server = started.server;
baseUrl = started.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
const connect = await jsonFetch(`${baseUrl}/api/connectors/slack/connect`, { method: 'POST' });
expect(connect.status).toBe(403);
expect(connect.body.error.code).toBe('CONNECTOR_EXECUTION_FAILED');
});
it('does not let stale in-flight discovery overwrite a newly created auth config', async () => {
const started = createDeferred();
const release = createDeferred();
await new Promise((resolve, reject) => {
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
mockComposioFetch({
authConfigs: [],
delayFirstToolkits: { started, release },
linkResponse: { connected_account_id: 'ca_slack', status: 'ACTIVE', account_label: 'slack@example.com' },
});
composioConnectorProvider.clearDiscoveryCache();
const restarted = await startServer({ port: 0, returnServer: true });
server = restarted.server;
baseUrl = restarted.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
const staleDiscovery = composioConnectorProvider.listDefinitions();
await started.promise;
const slack = getStaticComposioCatalogDefinitions().find((connector) => connector.id === 'slack');
await composioConnectorProvider.connect(slack, `${baseUrl}/api/connectors/oauth/callback/slack`);
release.resolve();
await staleDiscovery;
const hydrated = await composioConnectorProvider.getDefinition('slack');
expect(hydrated?.tools.map((tool) => tool.name)).toEqual(expect.arrayContaining(['slack.slack_list_channels', 'slack.slack_send_message']));
expect(hydrated?.allowedToolNames).toEqual(['slack.slack_list_channels']);
});
it('TTL-prunes pending Composio OAuth states even if callbacks never arrive', async () => {
mockComposioFetch({
linkResponse: {
connected_account_id: 'ca_github',
status: 'INITIATED',
redirect_url: 'https://example.com/oauth',
},
});
vi.useFakeTimers();
vi.setSystemTime(new Date('2026-04-30T00:00:00.000Z'));
const provider = new ComposioConnectorProvider();
const github = getStaticComposioCatalogDefinitions().find((connector) => connector.id === 'github');
await provider.connect(github, `${baseUrl}/api/connectors/oauth/callback/github`);
expect(provider.pendingConnections.size).toBe(1);
vi.advanceTimersByTime(10 * 60 * 1000 + 1);
await provider.connect(github, `${baseUrl}/api/connectors/oauth/callback/github`);
expect(provider.pendingConnections.size).toBe(1);
});
it('returns branded callback HTML that notifies the opener', async () => {
await new Promise((resolve, reject) => {
if (!server) return resolve(undefined);
server.close((error) => (error ? reject(error) : resolve(undefined)));
});
mockComposioFetch({
linkResponse: {
connected_account_id: 'ca_github',
status: 'INITIATED',
redirect_url: 'https://example.com/oauth',
},
});
const started = await startServer({ port: 0, returnServer: true });
server = started.server;
baseUrl = started.url;
await jsonFetch(`${baseUrl}/api/connectors/composio/config`, {
method: 'PUT',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ apiKey: 'cmp_test' }),
});
const connect = await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
expect(connect.status).toBe(200);
expect(connect.body.auth).toMatchObject({ kind: 'redirect_required' });
const callbackUrl = new URL(lastComposioLinkRequest.callback_url);
const response = await fetch(
`${baseUrl}/api/connectors/oauth/callback/github?state=${encodeURIComponent(callbackUrl.searchParams.get('state'))}&status=success&connected_account_id=ca_github`,
);
const html = await response.text();
expect(response.status).toBe(200);
expect(html).toContain('<main aria-labelledby="callback-title">');
expect(html).toContain('GitHub connected');
expect(html).toContain('Open Design');
expect(html).toContain('open-design:connector-connected');
expect(html).not.toContain('<p>Connector connected. You can close this window.</p>');
});
it('accepts bracketed IPv6 loopback host headers for connector callback URLs', async () => {
const url = new URL(baseUrl);
const response = await postWithHostHeader(`${baseUrl}/api/connectors/github/connect`, `[::1]:${url.port}`);
expect(response.status).toBe(200);
expect(JSON.parse(response.body).auth).toMatchObject({ kind: 'connected' });
expect(lastComposioLinkRequest.callback_url).toContain(`[::1]:${url.port}/api/connectors/oauth/callback`);
});
it('accepts IPv4 loopback alias host headers for connector callback URLs', async () => {
const url = new URL(baseUrl);
const response = await postWithHostHeader(`${baseUrl}/api/connectors/github/connect`, `127.0.0.2:${url.port}`);
expect(response.status).toBe(200);
expect(JSON.parse(response.body).auth).toMatchObject({ kind: 'connected' });
expect(lastComposioLinkRequest.callback_url).toContain(`127.0.0.2:${url.port}/api/connectors/oauth/callback`);
});
it('lists connected Composio tools through run-scoped tool auth', async () => {
await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
const token = mintConnectorToolToken();
const response = await jsonFetch(`${baseUrl}/api/tools/connectors/list`, {
headers: { Authorization: `Bearer ${token}` },
});
expect(response.status).toBe(200);
expect(response.body.connectors.map((connector) => connector.id)).toEqual(['github']);
expect(response.body.connectors[0].tools).toEqual(expect.arrayContaining([
expect.objectContaining({ name: 'github.github_search_repositories', safety: expect.objectContaining({ sideEffect: 'read', approval: 'auto' }) }),
]));
});
it('executes connected Composio tools through run-scoped tool auth', async () => {
await jsonFetch(`${baseUrl}/api/connectors/github/connect`, { method: 'POST' });
const token = mintConnectorToolToken('connector-execute-project', 'connector-execute-run');
const response = await jsonFetch(`${baseUrl}/api/tools/connectors/execute`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
body: JSON.stringify({ connectorId: 'github', toolName: 'github.github_search_repositories', input: { query: 'open-design' } }),
});
expect(response.status).toBe(200);
expect(response.body).toMatchObject({ ok: true, connectorId: 'github', accountLabel: 'octocat@example.com', toolName: 'github.github_search_repositories' });
expect(response.body.output).toMatchObject({ toolName: 'github.github_search_repositories', providerToolId: 'GITHUB_SEARCH_REPOSITORIES', data: { results: [] } });
});
it('rejects connector tool requests outside token scope', async () => {
const listOnlyToken = mintConnectorToolToken('connector-scope-project', 'connector-scope-run', {
allowedEndpoints: ['/api/tools/connectors/list'],
allowedOperations: ['connectors:list'],
});
const execute = await jsonFetch(`${baseUrl}/api/tools/connectors/execute`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${listOnlyToken}` },
body: JSON.stringify({ connectorId: 'github', toolName: 'github.github_search_repositories', input: { query: 'open-design' } }),
});
expect(execute.status).toBe(403);
expect(execute.body.error.code).toBe('TOOL_ENDPOINT_DENIED');
});
});