@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.
- package/README.md +15 -0
- package/dist/api.d.ts +53 -0
- package/dist/api.d.ts.map +1 -0
- package/dist/api.js +842 -0
- package/dist/api.js.map +1 -0
- package/dist/capabilities.d.ts +5 -0
- package/dist/capabilities.d.ts.map +1 -0
- package/dist/capabilities.js +308 -0
- package/dist/capabilities.js.map +1 -0
- package/dist/constants.d.ts +4 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +4 -0
- package/dist/constants.js.map +1 -0
- package/dist/dev.d.ts +14 -0
- package/dist/dev.d.ts.map +1 -0
- package/dist/dev.js +203 -0
- package/dist/dev.js.map +1 -0
- package/dist/foreground.d.ts +13 -0
- package/dist/foreground.d.ts.map +1 -0
- package/dist/foreground.js +54 -0
- package/dist/foreground.js.map +1 -0
- package/dist/import-openapi.d.ts +12 -0
- package/dist/import-openapi.d.ts.map +1 -0
- package/dist/import-openapi.js +488 -0
- package/dist/import-openapi.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +941 -0
- package/dist/index.js.map +1 -0
- package/dist/local-config.d.ts +12 -0
- package/dist/local-config.d.ts.map +1 -0
- package/dist/local-config.js +93 -0
- package/dist/local-config.js.map +1 -0
- package/dist/local-daemon-entry.d.ts +2 -0
- package/dist/local-daemon-entry.d.ts.map +1 -0
- package/dist/local-daemon-entry.js +545 -0
- package/dist/local-daemon-entry.js.map +1 -0
- package/dist/local-daemon.d.ts +58 -0
- package/dist/local-daemon.d.ts.map +1 -0
- package/dist/local-daemon.js +385 -0
- package/dist/local-daemon.js.map +1 -0
- package/dist/mcp/api-client.d.ts +72 -0
- package/dist/mcp/api-client.d.ts.map +1 -0
- package/dist/mcp/api-client.js +210 -0
- package/dist/mcp/api-client.js.map +1 -0
- package/dist/mcp/index.d.ts +4 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +18 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/mcp/schema.d.ts +3 -0
- package/dist/mcp/schema.d.ts.map +1 -0
- package/dist/mcp/schema.js +95 -0
- package/dist/mcp/schema.js.map +1 -0
- package/dist/mcp/server.d.ts +5 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/mcp/server.js +57 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/tools.d.ts +12 -0
- package/dist/mcp/tools.d.ts.map +1 -0
- package/dist/mcp/tools.js +54 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/output-download.d.ts +8 -0
- package/dist/output-download.d.ts.map +1 -0
- package/dist/output-download.js +210 -0
- package/dist/output-download.js.map +1 -0
- package/dist/output.d.ts +124 -0
- package/dist/output.d.ts.map +1 -0
- package/dist/output.js +752 -0
- package/dist/output.js.map +1 -0
- package/dist/passthrough.d.ts +15 -0
- package/dist/passthrough.d.ts.map +1 -0
- package/dist/passthrough.js +68 -0
- package/dist/passthrough.js.map +1 -0
- package/dist/runtime-attribution.d.ts +5 -0
- package/dist/runtime-attribution.d.ts.map +1 -0
- package/dist/runtime-attribution.js +86 -0
- package/dist/runtime-attribution.js.map +1 -0
- package/dist/state.d.ts +56 -0
- package/dist/state.d.ts.map +1 -0
- package/dist/state.js +374 -0
- package/dist/state.js.map +1 -0
- package/dist/types.d.ts +284 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- 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 @@
|
|
|
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 @@
|
|
|
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"}
|