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
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/*)
557 lines
26 KiB
TypeScript
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');
|
|
});
|
|
});
|