Replace the confused Profile abstraction with a dedicated Secret resource
following Kubernetes conventions. Servers now have env entries with inline
values or secretRef references. Env vars are resolved and passed to
containers at startup (fixes existing gap).
- Add Secret CRUD (model, repo, service, routes, CLI commands)
- Server env: {name, value} or {name, valueFrom: {secretRef: {name, key}}}
- Add env-resolver utility shared by instance startup and config generation
- Remove all profile-related code (models, services, routes, CLI, tests)
- Update backup/restore for secrets instead of profiles
- describe secret masks values by default, --show-values to reveal
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
71 lines
2.3 KiB
TypeScript
71 lines
2.3 KiB
TypeScript
import { describe, it, expect } from 'vitest';
|
|
import { generateMcpConfig } from '../src/services/mcp-config-generator.js';
|
|
import type { McpServer } from '@prisma/client';
|
|
|
|
function makeServer(overrides: Partial<McpServer> = {}): McpServer {
|
|
return {
|
|
id: 's1',
|
|
name: 'slack',
|
|
description: 'Slack MCP',
|
|
packageName: '@anthropic/slack-mcp',
|
|
dockerImage: null,
|
|
transport: 'STDIO',
|
|
repositoryUrl: null,
|
|
env: [],
|
|
version: 1,
|
|
createdAt: new Date(),
|
|
updatedAt: new Date(),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
describe('generateMcpConfig', () => {
|
|
it('returns empty mcpServers for empty input', () => {
|
|
const result = generateMcpConfig([]);
|
|
expect(result).toEqual({ mcpServers: {} });
|
|
});
|
|
|
|
it('generates config for a single server', () => {
|
|
const result = generateMcpConfig([
|
|
{ server: makeServer(), resolvedEnv: {} },
|
|
]);
|
|
expect(result.mcpServers['slack']).toBeDefined();
|
|
expect(result.mcpServers['slack']?.command).toBe('npx');
|
|
expect(result.mcpServers['slack']?.args).toEqual(['-y', '@anthropic/slack-mcp']);
|
|
});
|
|
|
|
it('includes resolved env when present', () => {
|
|
const result = generateMcpConfig([
|
|
{ server: makeServer(), resolvedEnv: { SLACK_TEAM_ID: 'T123' } },
|
|
]);
|
|
const config = result.mcpServers['slack'];
|
|
expect(config?.env).toBeDefined();
|
|
expect(config?.env?.['SLACK_TEAM_ID']).toBe('T123');
|
|
});
|
|
|
|
it('omits env when resolvedEnv is empty', () => {
|
|
const result = generateMcpConfig([
|
|
{ server: makeServer(), resolvedEnv: {} },
|
|
]);
|
|
expect(result.mcpServers['slack']?.env).toBeUndefined();
|
|
});
|
|
|
|
it('generates multiple server configs', () => {
|
|
const result = generateMcpConfig([
|
|
{ server: makeServer({ name: 'slack' }), resolvedEnv: {} },
|
|
{ server: makeServer({ name: 'github', id: 's2', packageName: '@anthropic/github-mcp' }), resolvedEnv: {} },
|
|
]);
|
|
expect(Object.keys(result.mcpServers)).toHaveLength(2);
|
|
expect(result.mcpServers['slack']).toBeDefined();
|
|
expect(result.mcpServers['github']).toBeDefined();
|
|
});
|
|
|
|
it('uses server name as fallback when packageName is null', () => {
|
|
const server = makeServer({ packageName: null });
|
|
const result = generateMcpConfig([
|
|
{ server, resolvedEnv: {} },
|
|
]);
|
|
expect(result.mcpServers['slack']?.args).toEqual(['-y', 'slack']);
|
|
});
|
|
});
|