@zonko-ai/harbor 0.1.0

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 (86) hide show
  1. package/README.md +15 -0
  2. package/dist/api.d.ts +53 -0
  3. package/dist/api.d.ts.map +1 -0
  4. package/dist/api.js +842 -0
  5. package/dist/api.js.map +1 -0
  6. package/dist/capabilities.d.ts +5 -0
  7. package/dist/capabilities.d.ts.map +1 -0
  8. package/dist/capabilities.js +308 -0
  9. package/dist/capabilities.js.map +1 -0
  10. package/dist/constants.d.ts +4 -0
  11. package/dist/constants.d.ts.map +1 -0
  12. package/dist/constants.js +4 -0
  13. package/dist/constants.js.map +1 -0
  14. package/dist/dev.d.ts +14 -0
  15. package/dist/dev.d.ts.map +1 -0
  16. package/dist/dev.js +203 -0
  17. package/dist/dev.js.map +1 -0
  18. package/dist/foreground.d.ts +13 -0
  19. package/dist/foreground.d.ts.map +1 -0
  20. package/dist/foreground.js +54 -0
  21. package/dist/foreground.js.map +1 -0
  22. package/dist/import-openapi.d.ts +12 -0
  23. package/dist/import-openapi.d.ts.map +1 -0
  24. package/dist/import-openapi.js +488 -0
  25. package/dist/import-openapi.js.map +1 -0
  26. package/dist/index.d.ts +3 -0
  27. package/dist/index.d.ts.map +1 -0
  28. package/dist/index.js +941 -0
  29. package/dist/index.js.map +1 -0
  30. package/dist/local-config.d.ts +12 -0
  31. package/dist/local-config.d.ts.map +1 -0
  32. package/dist/local-config.js +93 -0
  33. package/dist/local-config.js.map +1 -0
  34. package/dist/local-daemon-entry.d.ts +2 -0
  35. package/dist/local-daemon-entry.d.ts.map +1 -0
  36. package/dist/local-daemon-entry.js +545 -0
  37. package/dist/local-daemon-entry.js.map +1 -0
  38. package/dist/local-daemon.d.ts +58 -0
  39. package/dist/local-daemon.d.ts.map +1 -0
  40. package/dist/local-daemon.js +385 -0
  41. package/dist/local-daemon.js.map +1 -0
  42. package/dist/mcp/api-client.d.ts +72 -0
  43. package/dist/mcp/api-client.d.ts.map +1 -0
  44. package/dist/mcp/api-client.js +210 -0
  45. package/dist/mcp/api-client.js.map +1 -0
  46. package/dist/mcp/index.d.ts +4 -0
  47. package/dist/mcp/index.d.ts.map +1 -0
  48. package/dist/mcp/index.js +18 -0
  49. package/dist/mcp/index.js.map +1 -0
  50. package/dist/mcp/schema.d.ts +3 -0
  51. package/dist/mcp/schema.d.ts.map +1 -0
  52. package/dist/mcp/schema.js +95 -0
  53. package/dist/mcp/schema.js.map +1 -0
  54. package/dist/mcp/server.d.ts +5 -0
  55. package/dist/mcp/server.d.ts.map +1 -0
  56. package/dist/mcp/server.js +57 -0
  57. package/dist/mcp/server.js.map +1 -0
  58. package/dist/mcp/tools.d.ts +12 -0
  59. package/dist/mcp/tools.d.ts.map +1 -0
  60. package/dist/mcp/tools.js +54 -0
  61. package/dist/mcp/tools.js.map +1 -0
  62. package/dist/output-download.d.ts +8 -0
  63. package/dist/output-download.d.ts.map +1 -0
  64. package/dist/output-download.js +210 -0
  65. package/dist/output-download.js.map +1 -0
  66. package/dist/output.d.ts +124 -0
  67. package/dist/output.d.ts.map +1 -0
  68. package/dist/output.js +752 -0
  69. package/dist/output.js.map +1 -0
  70. package/dist/passthrough.d.ts +15 -0
  71. package/dist/passthrough.d.ts.map +1 -0
  72. package/dist/passthrough.js +68 -0
  73. package/dist/passthrough.js.map +1 -0
  74. package/dist/runtime-attribution.d.ts +5 -0
  75. package/dist/runtime-attribution.d.ts.map +1 -0
  76. package/dist/runtime-attribution.js +86 -0
  77. package/dist/runtime-attribution.js.map +1 -0
  78. package/dist/state.d.ts +56 -0
  79. package/dist/state.d.ts.map +1 -0
  80. package/dist/state.js +374 -0
  81. package/dist/state.js.map +1 -0
  82. package/dist/types.d.ts +284 -0
  83. package/dist/types.d.ts.map +1 -0
  84. package/dist/types.js +2 -0
  85. package/dist/types.js.map +1 -0
  86. package/package.json +35 -0
@@ -0,0 +1,210 @@
1
+ import { existsSync, readFileSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { join, resolve } from 'node:path';
4
+ export class HarborApiError extends Error {
5
+ status;
6
+ code;
7
+ payload;
8
+ constructor(status, code, message, payload) {
9
+ super(message);
10
+ this.status = status;
11
+ this.code = code;
12
+ this.payload = payload;
13
+ }
14
+ }
15
+ const DEFAULT_PROFILE_SLUG = 'local';
16
+ const PROFILE_SLUG_PATTERN = /^[A-Za-z0-9](?:[A-Za-z0-9._-]{0,62}[A-Za-z0-9])?$/;
17
+ const TERMINAL_STATES = new Set(['succeeded', 'failed', 'cancelled']);
18
+ const normalizeProfileSlug = (value) => {
19
+ if (typeof value !== 'string') {
20
+ return null;
21
+ }
22
+ const slug = value.trim();
23
+ if (!slug || !PROFILE_SLUG_PATTERN.test(slug)) {
24
+ return null;
25
+ }
26
+ return slug;
27
+ };
28
+ const resolveHarborHome = () => resolve(process.env.HARBOR_HOME ?? join(homedir(), '.harbor'));
29
+ const profileOverrideFromArgv = (argv = process.argv.slice(2)) => {
30
+ for (let index = 0; index < argv.length; index += 1) {
31
+ const arg = argv[index];
32
+ if (arg === '--profile') {
33
+ const value = argv[index + 1];
34
+ return typeof value === 'string' && !value.startsWith('-') ? value : null;
35
+ }
36
+ if (arg.startsWith('--profile=')) {
37
+ return arg.slice('--profile='.length);
38
+ }
39
+ }
40
+ return null;
41
+ };
42
+ const readJson = (path) => {
43
+ if (!existsSync(path)) {
44
+ return null;
45
+ }
46
+ try {
47
+ return JSON.parse(readFileSync(path, 'utf8'));
48
+ }
49
+ catch {
50
+ return null;
51
+ }
52
+ };
53
+ const readProfile = () => {
54
+ const harborHome = resolveHarborHome();
55
+ const configPath = join(harborHome, 'config.json');
56
+ const legacyProfilePath = join(harborHome, 'profile.json');
57
+ const selectedProfileSlug = normalizeProfileSlug(profileOverrideFromArgv())
58
+ ?? normalizeProfileSlug(process.env.HARBOR_PROFILE)
59
+ ?? normalizeProfileSlug(readJson(configPath)?.active_profile_slug)
60
+ ?? normalizeProfileSlug(readJson(legacyProfilePath)?.profile_slug)
61
+ ?? DEFAULT_PROFILE_SLUG;
62
+ const profilePath = join(harborHome, 'profiles', `${selectedProfileSlug}.json`);
63
+ const profile = readJson(profilePath);
64
+ if (profile) {
65
+ return profile;
66
+ }
67
+ const legacyProfile = readJson(legacyProfilePath);
68
+ if (legacyProfile && (normalizeProfileSlug(legacyProfile.profile_slug) ?? DEFAULT_PROFILE_SLUG) === selectedProfileSlug) {
69
+ return legacyProfile;
70
+ }
71
+ return {};
72
+ };
73
+ const isExpired = (value, skewMs = 30_000) => {
74
+ if (!value) {
75
+ return true;
76
+ }
77
+ const expiresAt = Date.parse(value);
78
+ if (Number.isNaN(expiresAt)) {
79
+ return true;
80
+ }
81
+ return expiresAt <= Date.now() + skewMs;
82
+ };
83
+ const readJsonBody = async (response) => {
84
+ const text = await response.text();
85
+ if (!text) {
86
+ return null;
87
+ }
88
+ try {
89
+ return JSON.parse(text);
90
+ }
91
+ catch {
92
+ return text;
93
+ }
94
+ };
95
+ const asObject = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value)
96
+ ? value
97
+ : {});
98
+ const asString = (...values) => {
99
+ for (const value of values) {
100
+ if (typeof value === 'string' && value.length > 0) {
101
+ return value;
102
+ }
103
+ }
104
+ return null;
105
+ };
106
+ const withTimeout = (timeoutMs) => AbortSignal.timeout(timeoutMs);
107
+ export class HarborApiClient {
108
+ baseUrl;
109
+ agentToken;
110
+ constructor() {
111
+ const profile = readProfile();
112
+ const apiUrl = process.env.HARBOR_API_URL ?? profile.api_url ?? null;
113
+ if (!apiUrl) {
114
+ throw new Error('No Harbor API is configured. Start the Harbor API manually and run `harbor auth claim` first.');
115
+ }
116
+ const agentToken = process.env.HARBOR_AGENT_TOKEN ?? (!isExpired(profile.token_expires_at) ? profile.agent_token ?? null : null);
117
+ if (!agentToken) {
118
+ throw new Error('No Harbor agent token is available. Run `harbor auth claim` first.');
119
+ }
120
+ this.baseUrl = apiUrl;
121
+ this.agentToken = agentToken;
122
+ }
123
+ async request(method, path, body, timeoutMs = 30_000) {
124
+ const response = await fetch(`${this.baseUrl}${path}`, {
125
+ method,
126
+ headers: {
127
+ Accept: 'application/json',
128
+ Authorization: `Bearer ${this.agentToken}`,
129
+ ...(body ? { 'Content-Type': 'application/json' } : {}),
130
+ },
131
+ body: body ? JSON.stringify(body) : undefined,
132
+ signal: withTimeout(timeoutMs),
133
+ });
134
+ const payload = await readJsonBody(response);
135
+ if (!response.ok) {
136
+ const record = asObject(payload);
137
+ const errorRecord = asObject(record.error);
138
+ throw new HarborApiError(response.status, asString(record.code, errorRecord.code) ?? `http_${response.status}`, asString(record.message, errorRecord.message, response.statusText) ?? 'Harbor API request failed.', payload);
139
+ }
140
+ return payload;
141
+ }
142
+ async fetchCatalog() {
143
+ const payload = await this.request('GET', '/v1/capabilities', undefined, 15_000);
144
+ const record = asObject(payload);
145
+ const capabilities = Array.isArray(record.capabilities) ? record.capabilities : [];
146
+ return capabilities.flatMap((entry) => {
147
+ const capability = asObject(entry);
148
+ const snapshot = asObject(capability.snapshot);
149
+ return snapshot?.capability?.slug ? [{
150
+ slug: snapshot.capability.slug,
151
+ snapshot,
152
+ }] : [];
153
+ });
154
+ }
155
+ async startExecution(capabilitySlug, input, timeoutMs) {
156
+ const payload = await this.request('POST', '/v1/executions', {
157
+ capability_slug: capabilitySlug,
158
+ input,
159
+ }, timeoutMs);
160
+ const record = asObject(payload);
161
+ return {
162
+ status: asString(record.status, record.result) ?? 'accepted',
163
+ execution_id: asString(record.execution_id),
164
+ capability: asString(record.capability, record.capability_slug, capabilitySlug),
165
+ state: asString(record.state),
166
+ };
167
+ }
168
+ async fetchExecutionLogs(executionId, options = {}) {
169
+ const search = new URLSearchParams();
170
+ if (options.follow) {
171
+ search.set('follow', 'true');
172
+ }
173
+ if (typeof options.after === 'number') {
174
+ search.set('after', String(options.after));
175
+ }
176
+ if (typeof options.waitMs === 'number') {
177
+ search.set('wait_ms', String(options.waitMs));
178
+ }
179
+ const suffix = search.size > 0 ? `?${search.toString()}` : '';
180
+ const payload = await this.request('GET', `/v1/executions/${encodeURIComponent(executionId)}/logs${suffix}`, undefined, 35_000);
181
+ const record = asObject(payload);
182
+ return {
183
+ execution_id: asString(record.execution_id, executionId) ?? executionId,
184
+ state: asString(record.state, record.status),
185
+ exit_code: typeof record.exit_code === 'number' ? record.exit_code : null,
186
+ error_code: asString(record.error_code),
187
+ error_message: asString(record.error_message),
188
+ started_at: asString(record.started_at),
189
+ completed_at: asString(record.completed_at),
190
+ events: Array.isArray(record.events)
191
+ ? record.events.flatMap((event) => {
192
+ const normalized = asObject(event);
193
+ return typeof normalized.message === 'string'
194
+ ? [{
195
+ sequence: typeof normalized.sequence === 'number' ? normalized.sequence : undefined,
196
+ kind: asString(normalized.kind) ?? undefined,
197
+ message: normalized.message,
198
+ payload_json: normalized.payload_json,
199
+ occurred_at: asString(normalized.occurred_at) ?? undefined,
200
+ }]
201
+ : [];
202
+ })
203
+ : [],
204
+ };
205
+ }
206
+ isTerminalState(state) {
207
+ return Boolean(state && TERMINAL_STATES.has(state));
208
+ }
209
+ }
210
+ //# sourceMappingURL=api-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"api-client.js","sourceRoot":"","sources":["../../src/mcp/api-client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ #!/usr/bin/env node
2
+ export { ALLOWED_MCP_SLUGS, createHarborMcpServer, runHarborMcpStdioServer } from './server.js';
3
+ export declare const runHarborMcpCli: () => Promise<void>;
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":";AAKA,OAAO,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAA;AAE/F,eAAO,MAAM,eAAe,qBAE3B,CAAA"}
@@ -0,0 +1,18 @@
1
+ #!/usr/bin/env node
2
+ import { pathToFileURL } from 'node:url';
3
+ import { runHarborMcpStdioServer } from './server.js';
4
+ export { ALLOWED_MCP_SLUGS, createHarborMcpServer, runHarborMcpStdioServer } from './server.js';
5
+ export const runHarborMcpCli = async () => {
6
+ await runHarborMcpStdioServer();
7
+ };
8
+ const isMainModule = () => {
9
+ const entry = process.argv[1];
10
+ return entry ? import.meta.url === pathToFileURL(entry).href : false;
11
+ };
12
+ if (isMainModule()) {
13
+ runHarborMcpCli().catch((error) => {
14
+ process.stderr.write(`${error instanceof Error ? error.message : String(error)}\n`);
15
+ process.exitCode = 1;
16
+ });
17
+ }
18
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAA;AAExC,OAAO,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAA;AAErD,OAAO,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAA;AAE/F,MAAM,CAAC,MAAM,eAAe,GAAG,KAAK,IAAI,EAAE;IACxC,MAAM,uBAAuB,EAAE,CAAA;AACjC,CAAC,CAAA;AAED,MAAM,YAAY,GAAG,GAAG,EAAE;IACxB,MAAM,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IAC7B,OAAO,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,KAAK,aAAa,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAA;AACtE,CAAC,CAAA;AAED,IAAI,YAAY,EAAE,EAAE,CAAC;IACnB,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;QAChC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;QACnF,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAA;IACtB,CAAC,CAAC,CAAA;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { z } from 'zod';
2
+ export declare const jsonSchemaToZod: (schemaInput: Record<string, unknown>) => z.ZodTypeAny;
3
+ //# 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;AA4CvB,eAAO,MAAM,eAAe,GAAI,aAAa,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAG,CAAC,CAAC,UA+DxE,CAAA"}
@@ -0,0 +1,95 @@
1
+ import { z } from 'zod';
2
+ const asObject = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value)
3
+ ? value
4
+ : {});
5
+ const schemaType = (schema) => {
6
+ const type = schema.type;
7
+ if (typeof type === 'string') {
8
+ return type;
9
+ }
10
+ if (Array.isArray(type)) {
11
+ return type.find((entry) => typeof entry === 'string' && entry !== 'null') ?? null;
12
+ }
13
+ return null;
14
+ };
15
+ const isNullable = (schema) => Array.isArray(schema.type) && schema.type.includes('null');
16
+ const enumSchema = (values) => {
17
+ const filtered = values.filter((value) => (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean' || value === null));
18
+ if (filtered.length === 0) {
19
+ return null;
20
+ }
21
+ if (filtered.every((value) => typeof value === 'string')) {
22
+ const entries = [...new Set(filtered)];
23
+ return entries.length === 1 ? z.literal(entries[0]) : z.enum(entries);
24
+ }
25
+ const literals = filtered.map((value) => z.literal(value));
26
+ if (literals.length === 1) {
27
+ return literals[0];
28
+ }
29
+ return z.union(literals);
30
+ };
31
+ export const jsonSchemaToZod = (schemaInput) => {
32
+ const schema = asObject(schemaInput);
33
+ const type = schemaType(schema);
34
+ let base;
35
+ if (Array.isArray(schema.enum)) {
36
+ base = enumSchema(schema.enum) ?? z.any();
37
+ }
38
+ else if (type === 'object') {
39
+ const properties = asObject(schema.properties);
40
+ const required = new Set((Array.isArray(schema.required) ? schema.required : []).filter((entry) => typeof entry === 'string'));
41
+ const shape = Object.fromEntries(Object.entries(properties).map(([key, value]) => {
42
+ const propertySchema = jsonSchemaToZod(asObject(value));
43
+ return [key, required.has(key) ? propertySchema : propertySchema.optional()];
44
+ }));
45
+ const objectSchema = z.object(shape);
46
+ base = schema.additionalProperties === false ? objectSchema.strict() : objectSchema.passthrough();
47
+ }
48
+ else if (type === 'array') {
49
+ base = z.array(jsonSchemaToZod(asObject(schema.items)));
50
+ }
51
+ else if (type === 'boolean') {
52
+ base = z.boolean();
53
+ }
54
+ else if (type === 'integer') {
55
+ let integerSchema = z.number().int();
56
+ if (typeof schema.minimum === 'number') {
57
+ integerSchema = integerSchema.min(schema.minimum);
58
+ }
59
+ if (typeof schema.maximum === 'number') {
60
+ integerSchema = integerSchema.max(schema.maximum);
61
+ }
62
+ base = integerSchema;
63
+ }
64
+ else if (type === 'number') {
65
+ let numberSchema = z.number();
66
+ if (typeof schema.minimum === 'number') {
67
+ numberSchema = numberSchema.min(schema.minimum);
68
+ }
69
+ if (typeof schema.maximum === 'number') {
70
+ numberSchema = numberSchema.max(schema.maximum);
71
+ }
72
+ base = numberSchema;
73
+ }
74
+ else if (type === 'string') {
75
+ let stringSchema = z.string();
76
+ if (typeof schema.minLength === 'number') {
77
+ stringSchema = stringSchema.min(schema.minLength);
78
+ }
79
+ if (typeof schema.maxLength === 'number') {
80
+ stringSchema = stringSchema.max(schema.maxLength);
81
+ }
82
+ base = stringSchema;
83
+ }
84
+ else {
85
+ base = z.any();
86
+ }
87
+ if (typeof schema.description === 'string' && schema.description.length > 0) {
88
+ base = base.describe(schema.description);
89
+ }
90
+ if (isNullable(schema)) {
91
+ base = base.nullable();
92
+ }
93
+ return base;
94
+ };
95
+ //# 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,5 @@
1
+ import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
+ export declare const ALLOWED_MCP_SLUGS: ReadonlySet<string>;
3
+ export declare const createHarborMcpServer: () => Promise<McpServer>;
4
+ export declare const runHarborMcpStdioServer: () => Promise<void>;
5
+ //# 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;AAOnE,eAAO,MAAM,iBAAiB,EAAE,WAAW,CAAC,MAAM,CAQhD,CAAA;AAkBF,eAAO,MAAM,qBAAqB,0BA0BjC,CAAA;AAED,eAAO,MAAM,uBAAuB,qBAWnC,CAAA"}
@@ -0,0 +1,57 @@
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 { jsonSchemaToZod } from './schema.js';
5
+ import { buildMcpTools, executeMcpTool } from './tools.js';
6
+ export const ALLOWED_MCP_SLUGS = new Set([
7
+ 'search.brave',
8
+ 'search.exa',
9
+ 'search.serper',
10
+ 'search.tavily',
11
+ 'search.grok',
12
+ 'media.openrouter',
13
+ 'media.replicate',
14
+ ]);
15
+ const waitForShutdownSignal = () => new Promise((resolve) => {
16
+ const finish = () => {
17
+ process.off('SIGINT', finish);
18
+ process.off('SIGTERM', finish);
19
+ process.off('disconnect', finish);
20
+ process.stdin.off('close', finish);
21
+ resolve();
22
+ };
23
+ process.once('SIGINT', finish);
24
+ process.once('SIGTERM', finish);
25
+ process.once('disconnect', finish);
26
+ process.stdin.once('close', finish);
27
+ process.stdin.resume();
28
+ });
29
+ export const createHarborMcpServer = async () => {
30
+ const client = new HarborApiClient();
31
+ const capabilities = await client.fetchCatalog();
32
+ const allowed = capabilities.filter((c) => ALLOWED_MCP_SLUGS.has(c.slug));
33
+ const tools = buildMcpTools(allowed);
34
+ const server = new McpServer({ name: 'harbor', version: '0.1.0' });
35
+ for (const tool of tools) {
36
+ ;
37
+ server.registerTool(tool.name, {
38
+ title: tool.title,
39
+ description: tool.description,
40
+ inputSchema: jsonSchemaToZod(tool.inputSchema),
41
+ }, async (input) => executeMcpTool(client, tool, input));
42
+ }
43
+ return server;
44
+ };
45
+ export const runHarborMcpStdioServer = async () => {
46
+ const server = await createHarborMcpServer();
47
+ const transport = new StdioServerTransport();
48
+ try {
49
+ await server.connect(transport);
50
+ await waitForShutdownSignal();
51
+ }
52
+ finally {
53
+ await transport.close().catch(() => undefined);
54
+ await server.close().catch(() => undefined);
55
+ }
56
+ };
57
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/mcp/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ import type { CallToolResult } from '@modelcontextprotocol/sdk/types.js';
2
+ import type { HarborApiClient, HarborCatalogCapability, HarborSnapshot } from './api-client.js';
3
+ export interface HarborMcpTool {
4
+ name: string;
5
+ title: string;
6
+ description: string;
7
+ inputSchema: Record<string, unknown>;
8
+ snapshot: HarborSnapshot;
9
+ }
10
+ export declare const buildMcpTools: (capabilities: HarborCatalogCapability[]) => HarborMcpTool[];
11
+ export declare const executeMcpTool: (client: HarborApiClient, tool: HarborMcpTool, input: Record<string, unknown>) => Promise<CallToolResult>;
12
+ //# 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,EAAE,eAAe,EAAE,uBAAuB,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAA;AAE/F,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,cAAc,CAAA;CACzB;AAiBD,eAAO,MAAM,aAAa,GAAI,cAAc,uBAAuB,EAAE,KAAG,aAAa,EAqBxB,CAAA;AAE7D,eAAO,MAAM,cAAc,GACzB,QAAQ,eAAe,EACvB,MAAM,aAAa,EACnB,OAAO,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAC7B,OAAO,CAAC,cAAc,CAuBxB,CAAA"}
@@ -0,0 +1,54 @@
1
+ const asObject = (value) => (value !== null && typeof value === 'object' && !Array.isArray(value)
2
+ ? value
3
+ : {});
4
+ const executionRequestTimeoutMs = (snapshot) => {
5
+ const config = snapshot.binding?.config ?? {};
6
+ const requestTimeoutMs = typeof config.timeout_ms === 'number' && Number.isFinite(config.timeout_ms) && config.timeout_ms > 0
7
+ ? Math.floor(config.timeout_ms)
8
+ : 30_000;
9
+ return requestTimeoutMs + 5_000;
10
+ };
11
+ export const buildMcpTools = (capabilities) => capabilities
12
+ .map((capability) => {
13
+ const mcpSurface = asObject(capability.snapshot.surface?.mcp);
14
+ const toolName = typeof mcpSurface.tool_name === 'string' && mcpSurface.tool_name.length > 0
15
+ ? mcpSurface.tool_name
16
+ : capability.snapshot.capability.slug.replace(/\./g, '_');
17
+ const title = typeof mcpSurface.title === 'string' && mcpSurface.title.length > 0
18
+ ? mcpSurface.title
19
+ : capability.snapshot.capability.title;
20
+ const description = typeof mcpSurface.description === 'string' && mcpSurface.description.length > 0
21
+ ? mcpSurface.description
22
+ : capability.snapshot.capability.description;
23
+ return {
24
+ name: toolName,
25
+ title,
26
+ description,
27
+ inputSchema: capability.snapshot.contract?.input_schema ?? { type: 'object', additionalProperties: false, properties: {} },
28
+ snapshot: capability.snapshot,
29
+ };
30
+ })
31
+ .sort((left, right) => left.name.localeCompare(right.name));
32
+ export const executeMcpTool = async (client, tool, input) => {
33
+ const start = await client.startExecution(tool.snapshot.capability.slug, input, executionRequestTimeoutMs(tool.snapshot));
34
+ const isError = start.state === 'failed' || start.state === 'cancelled';
35
+ return {
36
+ isError,
37
+ content: [{
38
+ type: 'text',
39
+ text: JSON.stringify({
40
+ execution_id: start.execution_id,
41
+ capability: start.capability ?? tool.snapshot.capability.slug,
42
+ state: start.state ?? null,
43
+ status: start.status,
44
+ }, null, 2),
45
+ }],
46
+ structuredContent: {
47
+ execution_id: start.execution_id,
48
+ capability: start.capability ?? tool.snapshot.capability.slug,
49
+ state: start.state ?? null,
50
+ status: start.status,
51
+ },
52
+ };
53
+ };
54
+ //# 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"}