harborai 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (94) hide show
  1. package/README.md +17 -0
  2. package/dist/api.d.ts +62 -0
  3. package/dist/api.d.ts.map +1 -0
  4. package/dist/api.js +943 -0
  5. package/dist/api.js.map +1 -0
  6. package/dist/binary-install.d.ts +34 -0
  7. package/dist/binary-install.d.ts.map +1 -0
  8. package/dist/binary-install.js +181 -0
  9. package/dist/binary-install.js.map +1 -0
  10. package/dist/capabilities.d.ts +5 -0
  11. package/dist/capabilities.d.ts.map +1 -0
  12. package/dist/capabilities.js +315 -0
  13. package/dist/capabilities.js.map +1 -0
  14. package/dist/constants.d.ts +5 -0
  15. package/dist/constants.d.ts.map +1 -0
  16. package/dist/constants.js +10 -0
  17. package/dist/constants.js.map +1 -0
  18. package/dist/dev.d.ts +14 -0
  19. package/dist/dev.d.ts.map +1 -0
  20. package/dist/dev.js +203 -0
  21. package/dist/dev.js.map +1 -0
  22. package/dist/foreground.d.ts +13 -0
  23. package/dist/foreground.d.ts.map +1 -0
  24. package/dist/foreground.js +54 -0
  25. package/dist/foreground.js.map +1 -0
  26. package/dist/import-openapi.d.ts +12 -0
  27. package/dist/import-openapi.d.ts.map +1 -0
  28. package/dist/import-openapi.js +488 -0
  29. package/dist/import-openapi.js.map +1 -0
  30. package/dist/index.d.ts +3 -0
  31. package/dist/index.d.ts.map +1 -0
  32. package/dist/index.js +1174 -0
  33. package/dist/index.js.map +1 -0
  34. package/dist/local-config.d.ts +13 -0
  35. package/dist/local-config.d.ts.map +1 -0
  36. package/dist/local-config.js +147 -0
  37. package/dist/local-config.js.map +1 -0
  38. package/dist/local-daemon-entry.d.ts +2 -0
  39. package/dist/local-daemon-entry.d.ts.map +1 -0
  40. package/dist/local-daemon-entry.js +545 -0
  41. package/dist/local-daemon-entry.js.map +1 -0
  42. package/dist/local-daemon.d.ts +58 -0
  43. package/dist/local-daemon.d.ts.map +1 -0
  44. package/dist/local-daemon.js +386 -0
  45. package/dist/local-daemon.js.map +1 -0
  46. package/dist/mcp/api-client.d.ts +17 -0
  47. package/dist/mcp/api-client.d.ts.map +1 -0
  48. package/dist/mcp/api-client.js +73 -0
  49. package/dist/mcp/api-client.js.map +1 -0
  50. package/dist/mcp/index.d.ts +4 -0
  51. package/dist/mcp/index.d.ts.map +1 -0
  52. package/dist/mcp/index.js +18 -0
  53. package/dist/mcp/index.js.map +1 -0
  54. package/dist/mcp/install.d.ts +30 -0
  55. package/dist/mcp/install.d.ts.map +1 -0
  56. package/dist/mcp/install.js +153 -0
  57. package/dist/mcp/install.js.map +1 -0
  58. package/dist/mcp/schema.d.ts +4 -0
  59. package/dist/mcp/schema.d.ts.map +1 -0
  60. package/dist/mcp/schema.js +104 -0
  61. package/dist/mcp/schema.js.map +1 -0
  62. package/dist/mcp/server.d.ts +4 -0
  63. package/dist/mcp/server.d.ts.map +1 -0
  64. package/dist/mcp/server.js +47 -0
  65. package/dist/mcp/server.js.map +1 -0
  66. package/dist/mcp/tools.d.ts +13 -0
  67. package/dist/mcp/tools.d.ts.map +1 -0
  68. package/dist/mcp/tools.js +147 -0
  69. package/dist/mcp/tools.js.map +1 -0
  70. package/dist/output-download.d.ts +8 -0
  71. package/dist/output-download.d.ts.map +1 -0
  72. package/dist/output-download.js +210 -0
  73. package/dist/output-download.js.map +1 -0
  74. package/dist/output.d.ts +155 -0
  75. package/dist/output.d.ts.map +1 -0
  76. package/dist/output.js +1123 -0
  77. package/dist/output.js.map +1 -0
  78. package/dist/passthrough.d.ts +15 -0
  79. package/dist/passthrough.d.ts.map +1 -0
  80. package/dist/passthrough.js +69 -0
  81. package/dist/passthrough.js.map +1 -0
  82. package/dist/runtime-attribution.d.ts +5 -0
  83. package/dist/runtime-attribution.d.ts.map +1 -0
  84. package/dist/runtime-attribution.js +86 -0
  85. package/dist/runtime-attribution.js.map +1 -0
  86. package/dist/state.d.ts +56 -0
  87. package/dist/state.d.ts.map +1 -0
  88. package/dist/state.js +374 -0
  89. package/dist/state.js.map +1 -0
  90. package/dist/types.d.ts +321 -0
  91. package/dist/types.d.ts.map +1 -0
  92. package/dist/types.js +2 -0
  93. package/dist/types.js.map +1 -0
  94. package/package.json +35 -0
@@ -0,0 +1,153 @@
1
+ import { chmodSync, existsSync, mkdirSync, readFileSync, realpathSync, renameSync, writeFileSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { dirname, join, resolve } from 'node:path';
4
+ const HARBOR_SERVER_NAME = 'harbor';
5
+ const isRecord = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value));
6
+ const resolveConfigPath = (homeDir, client) => {
7
+ if (client === 'claude-desktop') {
8
+ return join(homeDir, 'Library', 'Application Support', 'Claude', 'claude_desktop_config.json');
9
+ }
10
+ if (client === 'cursor') {
11
+ return join(homeDir, '.cursor', 'mcp.json');
12
+ }
13
+ return join(homeDir, '.pi', 'agent', 'mcp.json');
14
+ };
15
+ const clientLabel = (client) => {
16
+ if (client === 'claude-desktop') {
17
+ return 'Claude Desktop';
18
+ }
19
+ if (client === 'cursor') {
20
+ return 'Cursor';
21
+ }
22
+ return 'Pi';
23
+ };
24
+ const detectClient = (homeDir, client) => {
25
+ const path = resolveConfigPath(homeDir, client);
26
+ return {
27
+ client,
28
+ label: clientLabel(client),
29
+ path,
30
+ detected: existsSync(dirname(path)) || existsSync(path),
31
+ };
32
+ };
33
+ const supportedClients = (platform, homeDir) => {
34
+ if (platform !== 'darwin') {
35
+ return [];
36
+ }
37
+ return [
38
+ detectClient(homeDir, 'claude-desktop'),
39
+ detectClient(homeDir, 'cursor'),
40
+ detectClient(homeDir, 'pi'),
41
+ ];
42
+ };
43
+ const resolveCliEntryPath = (cliEntryPath = process.argv[1]) => {
44
+ if (typeof cliEntryPath !== 'string' || cliEntryPath.length === 0) {
45
+ throw new Error('Unable to resolve the Harbor CLI entry path for MCP installation.');
46
+ }
47
+ try {
48
+ return realpathSync(cliEntryPath);
49
+ }
50
+ catch {
51
+ return resolve(cliEntryPath);
52
+ }
53
+ };
54
+ export const buildHarborMcpLaunchConfig = (profileSlug, nodePath = process.execPath, cliEntryPath = process.argv[1]) => ({
55
+ command: realpathSync(nodePath),
56
+ args: [resolveCliEntryPath(cliEntryPath), '--profile', profileSlug, 'mcp'],
57
+ });
58
+ const buildServerConfig = (client, launch) => {
59
+ if (client === 'cursor') {
60
+ return {
61
+ type: 'stdio',
62
+ command: launch.command,
63
+ args: [...launch.args],
64
+ };
65
+ }
66
+ return {
67
+ command: launch.command,
68
+ args: [...launch.args],
69
+ };
70
+ };
71
+ const readConfig = (path) => {
72
+ if (!existsSync(path)) {
73
+ return {};
74
+ }
75
+ const raw = readFileSync(path, 'utf8');
76
+ let parsed;
77
+ try {
78
+ parsed = JSON.parse(raw);
79
+ }
80
+ catch {
81
+ throw new Error(`Invalid JSON in ${path}. Fix the file and retry.`);
82
+ }
83
+ if (!isRecord(parsed)) {
84
+ throw new Error(`Expected ${path} to contain a top-level JSON object.`);
85
+ }
86
+ if (parsed.mcpServers !== undefined && !isRecord(parsed.mcpServers)) {
87
+ throw new Error(`Expected ${path} to contain an object-valued mcpServers property.`);
88
+ }
89
+ return parsed;
90
+ };
91
+ const atomicWriteJson = (path, value) => {
92
+ mkdirSync(dirname(path), { recursive: true });
93
+ const tempPath = `${path}.tmp-${process.pid}`;
94
+ writeFileSync(tempPath, `${JSON.stringify(value, null, 2)}\n`, { mode: 0o600 });
95
+ chmodSync(tempPath, 0o600);
96
+ renameSync(tempPath, path);
97
+ };
98
+ const selectedClients = (client, platform, homeDir) => {
99
+ const detectedClients = supportedClients(platform, homeDir);
100
+ if (platform !== 'darwin') {
101
+ throw new Error('Automatic Harbor MCP client installation is currently supported on macOS only.');
102
+ }
103
+ if (!client) {
104
+ const targets = detectedClients.filter((entry) => entry.detected);
105
+ if (targets.length === 0) {
106
+ throw new Error('No supported MCP client config paths were detected on this Mac.');
107
+ }
108
+ return targets;
109
+ }
110
+ const target = detectedClients.find((entry) => entry.client === client);
111
+ if (!target) {
112
+ throw new Error(`Unsupported MCP client: ${client}. Use claude-desktop, cursor, or pi.`);
113
+ }
114
+ if (!target.detected) {
115
+ throw new Error(`${target.label} was not detected at ${target.path}. Install the client first, then retry.`);
116
+ }
117
+ return [target];
118
+ };
119
+ export const installHarborMcp = ({ client = null, dryRun = false, platform = process.platform, homeDir = homedir(), profileSlug, nodePath = process.execPath, cliEntryPath = process.argv[1], }) => {
120
+ const launch = buildHarborMcpLaunchConfig(profileSlug, nodePath, cliEntryPath);
121
+ const installs = selectedClients(client, platform, homeDir).map((target) => {
122
+ const currentConfig = readConfig(target.path);
123
+ const currentServers = isRecord(currentConfig.mcpServers) ? currentConfig.mcpServers : {};
124
+ const serverConfig = buildServerConfig(target.client, launch);
125
+ const previous = currentServers[HARBOR_SERVER_NAME];
126
+ const action = previous === undefined
127
+ ? (existsSync(target.path) ? 'updated' : 'created')
128
+ : (JSON.stringify(previous) === JSON.stringify(serverConfig) ? 'unchanged' : 'updated');
129
+ if (!dryRun && action !== 'unchanged') {
130
+ atomicWriteJson(target.path, {
131
+ ...currentConfig,
132
+ mcpServers: {
133
+ ...currentServers,
134
+ [HARBOR_SERVER_NAME]: serverConfig,
135
+ },
136
+ });
137
+ }
138
+ return {
139
+ client: target.client,
140
+ label: target.label,
141
+ path: target.path,
142
+ action,
143
+ server_config: serverConfig,
144
+ };
145
+ });
146
+ return {
147
+ profile: profileSlug,
148
+ dry_run: dryRun,
149
+ launch,
150
+ installs,
151
+ };
152
+ };
153
+ //# sourceMappingURL=install.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"install.js","sourceRoot":"","sources":["../../src/mcp/install.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import { z } from 'zod';
2
+ export declare const toMcpInputSchema: (schemaInput: Record<string, unknown> | null | undefined) => Record<string, unknown>;
3
+ export declare const jsonSchemaToZod: (schemaInput: Record<string, unknown>) => z.ZodTypeAny;
4
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../src/mcp/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAA;AAkDvB,eAAO,MAAM,gBAAgB,GAAI,aAAa,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,GAAG,SAAS,4BAGvF,CAAA;AAED,eAAO,MAAM,eAAe,GAAI,aAAa,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAG,CAAC,CAAC,UA+DxE,CAAA"}
@@ -0,0 +1,104 @@
1
+ import { z } from 'zod';
2
+ const EMPTY_INPUT_SCHEMA = {
3
+ type: 'object',
4
+ properties: {},
5
+ additionalProperties: false,
6
+ };
7
+ const asObject = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value)
8
+ ? value
9
+ : {});
10
+ const schemaType = (schema) => {
11
+ const type = schema.type;
12
+ if (typeof type === 'string') {
13
+ return type;
14
+ }
15
+ if (Array.isArray(type)) {
16
+ return type.find((entry) => typeof entry === 'string' && entry !== 'null') ?? null;
17
+ }
18
+ return null;
19
+ };
20
+ const isNullable = (schema) => Array.isArray(schema.type) && schema.type.includes('null');
21
+ const enumSchema = (values) => {
22
+ const filtered = values.filter((value) => (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean' || value === null));
23
+ if (filtered.length === 0) {
24
+ return null;
25
+ }
26
+ if (filtered.every((value) => typeof value === 'string')) {
27
+ const entries = [...new Set(filtered)];
28
+ return entries.length === 1 ? z.literal(entries[0]) : z.enum(entries);
29
+ }
30
+ const literals = filtered.map((value) => z.literal(value));
31
+ if (literals.length === 1) {
32
+ return literals[0];
33
+ }
34
+ return z.union(literals);
35
+ };
36
+ export const toMcpInputSchema = (schemaInput) => {
37
+ const schema = asObject(schemaInput);
38
+ return Object.keys(schema).length > 0 ? schema : EMPTY_INPUT_SCHEMA;
39
+ };
40
+ export const jsonSchemaToZod = (schemaInput) => {
41
+ const schema = toMcpInputSchema(schemaInput);
42
+ const type = schemaType(schema);
43
+ let base;
44
+ if (Array.isArray(schema.enum)) {
45
+ base = enumSchema(schema.enum) ?? z.any();
46
+ }
47
+ else if (type === 'object') {
48
+ const properties = asObject(schema.properties);
49
+ const required = new Set((Array.isArray(schema.required) ? schema.required : []).filter((entry) => typeof entry === 'string'));
50
+ const shape = Object.fromEntries(Object.entries(properties).map(([key, value]) => {
51
+ const propertySchema = jsonSchemaToZod(asObject(value));
52
+ return [key, required.has(key) ? propertySchema : propertySchema.optional()];
53
+ }));
54
+ const objectSchema = z.object(shape);
55
+ base = schema.additionalProperties === false ? objectSchema.strict() : objectSchema.passthrough();
56
+ }
57
+ else if (type === 'array') {
58
+ base = z.array(jsonSchemaToZod(asObject(schema.items)));
59
+ }
60
+ else if (type === 'boolean') {
61
+ base = z.boolean();
62
+ }
63
+ else if (type === 'integer') {
64
+ let integerSchema = z.number().int();
65
+ if (typeof schema.minimum === 'number') {
66
+ integerSchema = integerSchema.min(schema.minimum);
67
+ }
68
+ if (typeof schema.maximum === 'number') {
69
+ integerSchema = integerSchema.max(schema.maximum);
70
+ }
71
+ base = integerSchema;
72
+ }
73
+ else if (type === 'number') {
74
+ let numberSchema = z.number();
75
+ if (typeof schema.minimum === 'number') {
76
+ numberSchema = numberSchema.min(schema.minimum);
77
+ }
78
+ if (typeof schema.maximum === 'number') {
79
+ numberSchema = numberSchema.max(schema.maximum);
80
+ }
81
+ base = numberSchema;
82
+ }
83
+ else if (type === 'string') {
84
+ let stringSchema = z.string();
85
+ if (typeof schema.minLength === 'number') {
86
+ stringSchema = stringSchema.min(schema.minLength);
87
+ }
88
+ if (typeof schema.maxLength === 'number') {
89
+ stringSchema = stringSchema.max(schema.maxLength);
90
+ }
91
+ base = stringSchema;
92
+ }
93
+ else {
94
+ base = z.any();
95
+ }
96
+ if (typeof schema.description === 'string' && schema.description.length > 0) {
97
+ base = base.describe(schema.description);
98
+ }
99
+ if (isNullable(schema)) {
100
+ base = base.nullable();
101
+ }
102
+ return base;
103
+ };
104
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/mcp/schema.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ export declare const createHarborMcpServer: () => Promise<McpServer>;
3
+ export declare const runHarborMcpStdioServer: () => Promise<void>;
4
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/mcp/server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AAwBnE,eAAO,MAAM,qBAAqB,0BAmBjC,CAAA;AAED,eAAO,MAAM,uBAAuB,qBAWnC,CAAA"}
@@ -0,0 +1,47 @@
1
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
3
+ import { HarborApiClient } from './api-client.js';
4
+ import { CLI_PACKAGE_VERSION } from '../constants.js';
5
+ import { jsonSchemaToZod } from './schema.js';
6
+ import { buildMcpTools, executeMcpTool } from './tools.js';
7
+ const waitForShutdownSignal = () => new Promise((resolve) => {
8
+ const finish = () => {
9
+ process.off('SIGINT', finish);
10
+ process.off('SIGTERM', finish);
11
+ process.off('disconnect', finish);
12
+ process.stdin.off('close', finish);
13
+ resolve();
14
+ };
15
+ process.once('SIGINT', finish);
16
+ process.once('SIGTERM', finish);
17
+ process.once('disconnect', finish);
18
+ process.stdin.once('close', finish);
19
+ process.stdin.resume();
20
+ });
21
+ export const createHarborMcpServer = async () => {
22
+ const client = new HarborApiClient();
23
+ const capabilities = await client.fetchCapabilities();
24
+ const tools = buildMcpTools(capabilities);
25
+ const server = new McpServer({ name: 'harbor', version: CLI_PACKAGE_VERSION });
26
+ for (const tool of tools) {
27
+ server.registerTool(tool.name, {
28
+ title: tool.title,
29
+ description: tool.description,
30
+ inputSchema: jsonSchemaToZod(tool.inputSchema),
31
+ }, async (input) => executeMcpTool(client, tool, input));
32
+ }
33
+ return server;
34
+ };
35
+ export const runHarborMcpStdioServer = async () => {
36
+ const server = await createHarborMcpServer();
37
+ const transport = new StdioServerTransport();
38
+ try {
39
+ await server.connect(transport);
40
+ await waitForShutdownSignal();
41
+ }
42
+ finally {
43
+ await transport.close().catch(() => undefined);
44
+ await server.close().catch(() => undefined);
45
+ }
46
+ };
47
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/mcp/server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AACnE,OAAO,EAAE,oBAAoB,EAAE,MAAM,2CAA2C,CAAA;AAEhF,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAA;AACjD,OAAO,EAAE,mBAAmB,EAAE,MAAM,iBAAiB,CAAA;AACrD,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAA;AAC7C,OAAO,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,YAAY,CAAA;AAE1D,MAAM,qBAAqB,GAAG,GAAG,EAAE,CAAC,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;IAChE,MAAM,MAAM,GAAG,GAAG,EAAE;QAClB,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAA;QAC7B,OAAO,CAAC,GAAG,CAAC,SAAS,EAAE,MAAM,CAAC,CAAA;QAC9B,OAAO,CAAC,GAAG,CAAC,YAAY,EAAE,MAAM,CAAC,CAAA;QACjC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;QAClC,OAAO,EAAE,CAAA;IACX,CAAC,CAAA;IAED,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAA;IAC9B,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,MAAM,CAAC,CAAA;IAC/B,OAAO,CAAC,IAAI,CAAC,YAAY,EAAE,MAAM,CAAC,CAAA;IAClC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IACnC,OAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAA;AACxB,CAAC,CAAC,CAAA;AAEF,MAAM,CAAC,MAAM,qBAAqB,GAAG,KAAK,IAAI,EAAE;IAC9C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAA;IACpC,MAAM,YAAY,GAAG,MAAM,MAAM,CAAC,iBAAiB,EAAE,CAAA;IACrD,MAAM,KAAK,GAAG,aAAa,CAAC,YAAY,CAAC,CAAA;IACzC,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,mBAAmB,EAAE,CAAC,CAAA;IAE9E,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,CAAC,YAAY,CACjB,IAAI,CAAC,IAAI,EACT;YACE,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,WAAW,EAAE,eAAe,CAAC,IAAI,CAAC,WAAW,CAAC;SAC/C,EACD,KAAK,EAAE,KAA8B,EAAE,EAAE,CAAC,cAAc,CAAC,MAAM,EAAE,IAAI,EAAE,KAAK,CAAC,CAC9E,CAAA;IACH,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC,CAAA;AAED,MAAM,CAAC,MAAM,uBAAuB,GAAG,KAAK,IAAI,EAAE;IAChD,MAAM,MAAM,GAAG,MAAM,qBAAqB,EAAE,CAAA;IAC5C,MAAM,SAAS,GAAG,IAAI,oBAAoB,EAAE,CAAA;IAE5C,IAAI,CAAC;QACH,MAAM,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAA;QAC/B,MAAM,qBAAqB,EAAE,CAAA;IAC/B,CAAC;YAAS,CAAC;QACT,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,CAAA;QAC9C,MAAM,MAAM,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,SAAS,CAAC,CAAA;IAC7C,CAAC;AACH,CAAC,CAAA"}
@@ -0,0 +1,13 @@
1
+ import type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';
2
+ import type { PublishedSnapshot } from '../types.js';
3
+ import { type HarborApiClient, type HarborCatalogCapability } from './api-client.js';
4
+ export interface HarborMcpTool {
5
+ name: string;
6
+ title: string;
7
+ description: string;
8
+ inputSchema: Record<string, unknown>;
9
+ snapshot: PublishedSnapshot;
10
+ }
11
+ export declare const buildMcpTools: (capabilities: HarborCatalogCapability[]) => HarborMcpTool[];
12
+ export declare const executeMcpTool: (client: HarborApiClient, tool: HarborMcpTool, input: Record<string, unknown>) => Promise<CallToolResult>;
13
+ //# sourceMappingURL=tools.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.d.ts","sourceRoot":"","sources":["../../src/mcp/tools.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,oCAAoC,CAAA;AAExE,OAAO,KAAK,EAA8C,iBAAiB,EAAE,MAAM,aAAa,CAAA;AAChG,OAAO,EAAkB,KAAK,eAAe,EAAE,KAAK,uBAAuB,EAAE,MAAM,iBAAiB,CAAA;AAOpG,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAA;IACZ,KAAK,EAAE,MAAM,CAAA;IACb,WAAW,EAAE,MAAM,CAAA;IACnB,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;IACpC,QAAQ,EAAE,iBAAiB,CAAA;CAC5B;AA+HD,eAAO,MAAM,aAAa,GAAI,cAAc,uBAAuB,EAAE,KAAG,aAAa,EAQxB,CAAA;AAE7D,eAAO,MAAM,cAAc,GACzB,QAAQ,eAAe,EACvB,MAAM,aAAa,EACnB,OAAO,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAC7B,OAAO,CAAC,cAAc,CAgCxB,CAAA"}
@@ -0,0 +1,147 @@
1
+ import { HarborApiError } from './api-client.js';
2
+ import { toMcpInputSchema } from './schema.js';
3
+ const FAILURE_STATES = new Set(['failed', 'cancelled']);
4
+ const POLL_INTERVAL_MS = 1_000;
5
+ const POLL_TIMEOUT_MS = 10 * 60_000;
6
+ const asObject = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value)
7
+ ? value
8
+ : {});
9
+ const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
10
+ const result = (payload, isError = false, text) => ({
11
+ isError,
12
+ content: [{
13
+ type: 'text',
14
+ text: text ?? JSON.stringify(payload, null, 2),
15
+ }],
16
+ structuredContent: payload,
17
+ });
18
+ const approvalRequiredResult = (tool, approvalId, approvalUrl, reason) => result({
19
+ error: 'approval_required',
20
+ message: reason ?? `Approval is required to execute ${tool.snapshot.capability.slug}.`,
21
+ capability: tool.snapshot.capability.slug,
22
+ approval_id: approvalId ?? null,
23
+ approval_url: approvalUrl ?? null,
24
+ }, true);
25
+ const apiErrorResult = (error) => result({
26
+ error: error.code,
27
+ message: error.message,
28
+ status: error.status,
29
+ payload: asObject(error.payload),
30
+ }, true);
31
+ const executionText = (logs) => logs.events
32
+ .filter((event) => event.kind !== 'status' && event.kind !== 'system')
33
+ .map((event) => {
34
+ if (event.kind === 'stderr') {
35
+ return `[stderr] ${event.message}`;
36
+ }
37
+ if (event.kind === 'stdout' || !event.kind) {
38
+ return event.message;
39
+ }
40
+ return `[${event.kind}] ${event.message}`;
41
+ })
42
+ .join('\n');
43
+ const formatExecutionLogs = (tool, execution, logs) => {
44
+ const payload = {
45
+ execution_id: execution.execution_id,
46
+ capability: tool.snapshot.capability.slug,
47
+ state: execution.state,
48
+ exit_code: execution.exit_code ?? logs.exit_code ?? null,
49
+ error_code: execution.error_code ?? logs.error_code ?? null,
50
+ error_message: execution.error_message ?? logs.error_message ?? null,
51
+ logs: logs.events.map((event) => ({
52
+ sequence: event.sequence ?? null,
53
+ kind: event.kind ?? null,
54
+ message: event.message,
55
+ payload_json: event.payload_json ?? null,
56
+ occurred_at: event.occurred_at ?? null,
57
+ })),
58
+ };
59
+ const text = executionText(logs) || JSON.stringify(payload, null, 2);
60
+ const isError = FAILURE_STATES.has(execution.state ?? '') || (typeof payload.exit_code === 'number' && payload.exit_code !== 0);
61
+ return result(payload, isError, text);
62
+ };
63
+ const formatHttpResponse = (tool, execution, logs) => {
64
+ if (FAILURE_STATES.has(execution.state ?? '')) {
65
+ const payload = {
66
+ error: execution.error_code ?? 'execution_failed',
67
+ message: execution.error_message ?? logs.error_message ?? `Execution ${execution.state ?? 'failed'}.`,
68
+ execution_id: execution.execution_id,
69
+ capability: tool.snapshot.capability.slug,
70
+ state: execution.state,
71
+ };
72
+ const text = [
73
+ payload.message,
74
+ executionText(logs),
75
+ ].filter(Boolean).join('\n\n');
76
+ return result(payload, true, text);
77
+ }
78
+ const stdoutEvent = [...logs.events].reverse().find((event) => event.kind === 'stdout');
79
+ const payload = asObject(stdoutEvent?.payload_json);
80
+ if (Object.keys(payload).length > 0) {
81
+ return result(payload);
82
+ }
83
+ const text = stdoutEvent?.message ?? JSON.stringify({
84
+ execution_id: execution.execution_id,
85
+ capability: tool.snapshot.capability.slug,
86
+ state: execution.state,
87
+ }, null, 2);
88
+ return result({
89
+ execution_id: execution.execution_id,
90
+ capability: tool.snapshot.capability.slug,
91
+ state: execution.state,
92
+ output: stdoutEvent?.payload_json ?? stdoutEvent?.message ?? null,
93
+ }, false, text);
94
+ };
95
+ const waitForExecution = async (client, executionId) => {
96
+ const deadline = Date.now() + POLL_TIMEOUT_MS;
97
+ while (true) {
98
+ const execution = await client.getExecution(executionId);
99
+ if (client.isTerminalState(execution.state)) {
100
+ return execution;
101
+ }
102
+ if (Date.now() >= deadline) {
103
+ throw new Error(`Timed out waiting for execution ${executionId} to complete.`);
104
+ }
105
+ await sleep(POLL_INTERVAL_MS);
106
+ }
107
+ };
108
+ export const buildMcpTools = (capabilities) => capabilities
109
+ .map((capability) => ({
110
+ name: capability.snapshot.capability.slug,
111
+ title: capability.snapshot.capability.title ?? capability.snapshot.capability.slug,
112
+ description: capability.snapshot.capability.description ?? capability.snapshot.capability.slug,
113
+ inputSchema: toMcpInputSchema(asObject(capability.snapshot.contract?.input_schema)),
114
+ snapshot: capability.snapshot,
115
+ }))
116
+ .sort((left, right) => left.name.localeCompare(right.name));
117
+ export const executeMcpTool = async (client, tool, input) => {
118
+ try {
119
+ const start = await client.startExecution(tool.snapshot.capability.slug, input);
120
+ if (start.status === 'approval_required') {
121
+ return approvalRequiredResult(tool, start.approval_id, start.approval_url, start.reason);
122
+ }
123
+ if (!start.execution_id) {
124
+ return result({
125
+ error: 'execution_missing',
126
+ message: `Execution ${tool.snapshot.capability.slug} was accepted without an execution id.`,
127
+ }, true);
128
+ }
129
+ const execution = client.isTerminalState(start.state)
130
+ ? await client.getExecution(start.execution_id)
131
+ : await waitForExecution(client, start.execution_id);
132
+ const logs = await client.getExecutionLogs(start.execution_id);
133
+ return tool.snapshot.binding?.adapter_type === 'cli_stdio'
134
+ ? formatExecutionLogs(tool, execution, logs)
135
+ : formatHttpResponse(tool, execution, logs);
136
+ }
137
+ catch (error) {
138
+ if (error instanceof HarborApiError) {
139
+ return apiErrorResult(error);
140
+ }
141
+ return result({
142
+ error: 'execution_error',
143
+ message: error instanceof Error ? error.message : String(error),
144
+ }, true);
145
+ }
146
+ };
147
+ //# sourceMappingURL=tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.js","sourceRoot":"","sources":["../../src/mcp/tools.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface DownloadableOutput {
2
+ kind: 'url' | 'data_url';
3
+ value: string;
4
+ suggestedName?: string;
5
+ }
6
+ export declare const extractDownloadableOutputs: (payload: unknown) => DownloadableOutput[];
7
+ export declare const saveDownloadableOutputs: (payload: unknown, outputPath: string, fetchImpl?: typeof fetch) => Promise<string[]>;
8
+ //# sourceMappingURL=output-download.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output-download.d.ts","sourceRoot":"","sources":["../src/output-download.ts"],"names":[],"mappings":"AAGA,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,KAAK,GAAG,UAAU,CAAA;IACxB,KAAK,EAAE,MAAM,CAAA;IACb,aAAa,CAAC,EAAE,MAAM,CAAA;CACvB;AAkID,eAAO,MAAM,0BAA0B,GAAI,SAAS,OAAO,KAAG,kBAAkB,EAyD/E,CAAA;AAED,eAAO,MAAM,uBAAuB,GAClC,SAAS,OAAO,EAChB,YAAY,MAAM,EAClB,YAAW,OAAO,KAAa,KAC9B,OAAO,CAAC,MAAM,EAAE,CAsDlB,CAAA"}