@zykeco/sync-server 0.5.0 → 0.7.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 +70 -0
- package/dist/app.d.ts +4 -0
- package/dist/app.js +58 -0
- package/dist/app.js.map +1 -0
- package/dist/env.d.ts +6 -0
- package/dist/env.js +9 -1
- package/dist/env.js.map +1 -1
- package/dist/index.js +2 -48
- package/dist/index.js.map +1 -1
- package/dist/mcp/auth-store.d.ts +25 -0
- package/dist/mcp/auth-store.js +51 -0
- package/dist/mcp/auth-store.js.map +1 -0
- package/dist/mcp/consent.d.ts +16 -0
- package/dist/mcp/consent.js +90 -0
- package/dist/mcp/consent.js.map +1 -0
- package/dist/mcp/metric-types.d.ts +2 -0
- package/dist/mcp/metric-types.js +42 -0
- package/dist/mcp/metric-types.js.map +1 -0
- package/dist/mcp/oauth.d.ts +61 -0
- package/dist/mcp/oauth.js +149 -0
- package/dist/mcp/oauth.js.map +1 -0
- package/dist/mcp/server.d.ts +35 -0
- package/dist/mcp/server.js +94 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/timezones.d.ts +17 -0
- package/dist/mcp/timezones.js +32 -0
- package/dist/mcp/timezones.js.map +1 -0
- package/dist/mcp/tools.d.ts +4 -0
- package/dist/mcp/tools.js +236 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/mcp/transforms.d.ts +14 -0
- package/dist/mcp/transforms.js +45 -0
- package/dist/mcp/transforms.js.map +1 -0
- package/dist/routes/mcp.d.ts +20 -0
- package/dist/routes/mcp.js +315 -0
- package/dist/routes/mcp.js.map +1 -0
- package/package.json +2 -2
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Format a UTC instant as an ISO 8601 string with a timezone offset attached.
|
|
3
|
+
*
|
|
4
|
+
* formatTzIso(1716115262000, 120) === '2024-05-19T12:01:02+02:00'
|
|
5
|
+
*
|
|
6
|
+
* Seconds precision only — sub-second is intentionally dropped because none of
|
|
7
|
+
* the source timestamps carry it meaningfully.
|
|
8
|
+
*/
|
|
9
|
+
export function formatTzIso(utcMs, offsetMinutes) {
|
|
10
|
+
if (!Number.isFinite(utcMs) || !Number.isFinite(offsetMinutes)) {
|
|
11
|
+
throw new Error(`formatTzIso: non-finite input (${utcMs}, ${offsetMinutes})`);
|
|
12
|
+
}
|
|
13
|
+
const shifted = new Date(utcMs + offsetMinutes * 60_000);
|
|
14
|
+
const yyyy = shifted.getUTCFullYear().toString().padStart(4, '0');
|
|
15
|
+
const mm = (shifted.getUTCMonth() + 1).toString().padStart(2, '0');
|
|
16
|
+
const dd = shifted.getUTCDate().toString().padStart(2, '0');
|
|
17
|
+
const HH = shifted.getUTCHours().toString().padStart(2, '0');
|
|
18
|
+
const MM = shifted.getUTCMinutes().toString().padStart(2, '0');
|
|
19
|
+
const SS = shifted.getUTCSeconds().toString().padStart(2, '0');
|
|
20
|
+
const sign = offsetMinutes >= 0 ? '+' : '-';
|
|
21
|
+
const abs = Math.abs(offsetMinutes);
|
|
22
|
+
const offH = Math.floor(abs / 60)
|
|
23
|
+
.toString()
|
|
24
|
+
.padStart(2, '0');
|
|
25
|
+
const offM = (abs % 60).toString().padStart(2, '0');
|
|
26
|
+
return `${yyyy}-${mm}-${dd}T${HH}:${MM}:${SS}${sign}${offH}:${offM}`;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Format a signed duration in seconds as `hh:mm:ss` (or `-hh:mm:ss`). Supports
|
|
30
|
+
* durations larger than 24h — hours are not modulo'd.
|
|
31
|
+
*/
|
|
32
|
+
export function formatHmsFromSeconds(totalSeconds) {
|
|
33
|
+
if (!Number.isFinite(totalSeconds)) {
|
|
34
|
+
throw new Error(`formatHmsFromSeconds: non-finite input (${totalSeconds})`);
|
|
35
|
+
}
|
|
36
|
+
const sign = totalSeconds < 0 ? '-' : '';
|
|
37
|
+
const abs = Math.floor(Math.abs(totalSeconds));
|
|
38
|
+
const h = Math.floor(abs / 3600);
|
|
39
|
+
const m = Math.floor((abs % 3600) / 60);
|
|
40
|
+
const s = abs % 60;
|
|
41
|
+
return `${sign}${h.toString().padStart(2, '0')}:${m.toString().padStart(2, '0')}:${s
|
|
42
|
+
.toString()
|
|
43
|
+
.padStart(2, '0')}`;
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=transforms.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transforms.js","sourceRoot":"","sources":["../../src/mcp/transforms.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { Hono } from 'hono';
|
|
2
|
+
import type { Env, McpEnv } from '../env.js';
|
|
3
|
+
import { type AuthStore } from '../mcp/auth-store.js';
|
|
4
|
+
import type { Db } from '../db/client.js';
|
|
5
|
+
export interface McpDeps {
|
|
6
|
+
db: Db;
|
|
7
|
+
env: Env;
|
|
8
|
+
mcp: McpEnv;
|
|
9
|
+
authStore?: AuthStore;
|
|
10
|
+
}
|
|
11
|
+
export interface McpAppRoutes {
|
|
12
|
+
mcp: Hono;
|
|
13
|
+
discovery: Hono;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Build the MCP sub-apps. `mcp` is mounted under `/mcp`; `discovery` is
|
|
17
|
+
* mounted at `/` so RFC 8414 / RFC 9728 `.well-known` URLs sit at the root
|
|
18
|
+
* (which is what most MCP clients — including claude.ai — probe).
|
|
19
|
+
*/
|
|
20
|
+
export declare function mcpRoutes(deps: McpDeps): McpAppRoutes;
|
|
@@ -0,0 +1,315 @@
|
|
|
1
|
+
import { timingSafeEqual } from 'node:crypto';
|
|
2
|
+
import { Hono } from 'hono';
|
|
3
|
+
import { createAuthStore } from '../mcp/auth-store.js';
|
|
4
|
+
import { renderConsent } from '../mcp/consent.js';
|
|
5
|
+
import { grantAuthorizationCode, grantClientCredentials, verifyToken } from '../mcp/oauth.js';
|
|
6
|
+
import { createMcpServer } from '../mcp/server.js';
|
|
7
|
+
import { buildTools } from '../mcp/tools.js';
|
|
8
|
+
import { createTimezoneResolver } from '../mcp/timezones.js';
|
|
9
|
+
function issuerFrom(reqUrl) {
|
|
10
|
+
const u = new URL(reqUrl);
|
|
11
|
+
return `${u.protocol}//${u.host}`;
|
|
12
|
+
}
|
|
13
|
+
function authMetadata(issuer) {
|
|
14
|
+
return {
|
|
15
|
+
issuer,
|
|
16
|
+
authorization_endpoint: `${issuer}/mcp/authorize`,
|
|
17
|
+
token_endpoint: `${issuer}/mcp/oauth/token`,
|
|
18
|
+
registration_endpoint: `${issuer}/mcp/register`,
|
|
19
|
+
grant_types_supported: ['authorization_code', 'client_credentials'],
|
|
20
|
+
response_types_supported: ['code'],
|
|
21
|
+
code_challenge_methods_supported: ['S256'],
|
|
22
|
+
token_endpoint_auth_methods_supported: ['none', 'client_secret_post', 'client_secret_basic'],
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
function resourceMetadata(issuer) {
|
|
26
|
+
return {
|
|
27
|
+
resource: `${issuer}/mcp`,
|
|
28
|
+
authorization_servers: [issuer],
|
|
29
|
+
bearer_methods_supported: ['header'],
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Build the MCP sub-apps. `mcp` is mounted under `/mcp`; `discovery` is
|
|
34
|
+
* mounted at `/` so RFC 8414 / RFC 9728 `.well-known` URLs sit at the root
|
|
35
|
+
* (which is what most MCP clients — including claude.ai — probe).
|
|
36
|
+
*/
|
|
37
|
+
export function mcpRoutes(deps) {
|
|
38
|
+
const { db, env, mcp } = deps;
|
|
39
|
+
const tz = createTimezoneResolver(db);
|
|
40
|
+
const server = createMcpServer(buildTools(db, tz));
|
|
41
|
+
const store = deps.authStore ?? createAuthStore();
|
|
42
|
+
store.bootstrap(mcp.clientId);
|
|
43
|
+
const app = new Hono();
|
|
44
|
+
// Kept under `/mcp/oauth/...` for backward compat with older clients.
|
|
45
|
+
app.get('/oauth/.well-known/oauth-authorization-server', (c) => c.json(authMetadata(issuerFrom(c.req.url))));
|
|
46
|
+
// --- Dynamic Client Registration (RFC 7591) ------------------------------
|
|
47
|
+
app.post('/register', async (c) => {
|
|
48
|
+
let body;
|
|
49
|
+
try {
|
|
50
|
+
body = (await c.req.json());
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return c.json({ error: 'invalid_client_metadata', error_description: 'body must be JSON' }, 400);
|
|
54
|
+
}
|
|
55
|
+
const redirectUris = body.redirect_uris;
|
|
56
|
+
if (!Array.isArray(redirectUris) || redirectUris.length === 0) {
|
|
57
|
+
return c.json({ error: 'invalid_redirect_uri', error_description: 'redirect_uris is required' }, 400);
|
|
58
|
+
}
|
|
59
|
+
for (const u of redirectUris) {
|
|
60
|
+
if (typeof u !== 'string' || !isAcceptableRedirectUri(u)) {
|
|
61
|
+
return c.json({ error: 'invalid_redirect_uri' }, 400);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
const clientName = typeof body.client_name === 'string' ? body.client_name : null;
|
|
65
|
+
const client = store.registerClient({
|
|
66
|
+
redirectUris: redirectUris,
|
|
67
|
+
clientName,
|
|
68
|
+
});
|
|
69
|
+
return c.json({
|
|
70
|
+
client_id: client.clientId,
|
|
71
|
+
client_id_issued_at: Math.floor(client.createdAt / 1000),
|
|
72
|
+
redirect_uris: client.redirectUris,
|
|
73
|
+
...(client.clientName ? { client_name: client.clientName } : {}),
|
|
74
|
+
grant_types: ['authorization_code'],
|
|
75
|
+
response_types: ['code'],
|
|
76
|
+
token_endpoint_auth_method: 'none',
|
|
77
|
+
}, 201);
|
|
78
|
+
});
|
|
79
|
+
// --- Authorization endpoint (auth code + PKCE) ---------------------------
|
|
80
|
+
app.get('/authorize', (c) => {
|
|
81
|
+
const q = c.req.query();
|
|
82
|
+
const v = validateAuthorizeParams(q, store);
|
|
83
|
+
if (v.kind === 'fatal')
|
|
84
|
+
return c.text(`authorization error: ${v.message}`, 400);
|
|
85
|
+
if (v.kind === 'redirect') {
|
|
86
|
+
return c.redirect(appendParams(v.redirectUri, {
|
|
87
|
+
error: v.error,
|
|
88
|
+
error_description: v.message,
|
|
89
|
+
state: q.state ?? '',
|
|
90
|
+
}), 302);
|
|
91
|
+
}
|
|
92
|
+
return c.html(renderConsent({
|
|
93
|
+
clientId: v.client.clientId,
|
|
94
|
+
clientName: v.client.clientName,
|
|
95
|
+
redirectUri: q.redirect_uri,
|
|
96
|
+
state: q.state ?? '',
|
|
97
|
+
codeChallenge: q.code_challenge,
|
|
98
|
+
codeChallengeMethod: 'S256',
|
|
99
|
+
scope: q.scope ?? '',
|
|
100
|
+
error: null,
|
|
101
|
+
}));
|
|
102
|
+
});
|
|
103
|
+
app.post('/authorize', async (c) => {
|
|
104
|
+
const form = await c.req.parseBody();
|
|
105
|
+
const params = { response_type: 'code' };
|
|
106
|
+
for (const k of [
|
|
107
|
+
'client_id',
|
|
108
|
+
'redirect_uri',
|
|
109
|
+
'state',
|
|
110
|
+
'code_challenge',
|
|
111
|
+
'code_challenge_method',
|
|
112
|
+
'scope',
|
|
113
|
+
'password',
|
|
114
|
+
]) {
|
|
115
|
+
const val = form[k];
|
|
116
|
+
if (typeof val === 'string')
|
|
117
|
+
params[k] = val;
|
|
118
|
+
}
|
|
119
|
+
const v = validateAuthorizeParams(params, store);
|
|
120
|
+
if (v.kind === 'fatal')
|
|
121
|
+
return c.text(`authorization error: ${v.message}`, 400);
|
|
122
|
+
if (v.kind === 'redirect') {
|
|
123
|
+
return c.redirect(appendParams(v.redirectUri, {
|
|
124
|
+
error: v.error,
|
|
125
|
+
error_description: v.message,
|
|
126
|
+
state: params.state ?? '',
|
|
127
|
+
}), 302);
|
|
128
|
+
}
|
|
129
|
+
const password = params.password ?? '';
|
|
130
|
+
const a = Buffer.from(password);
|
|
131
|
+
const b = Buffer.from(env.readSecret);
|
|
132
|
+
const ok = a.length === b.length && timingSafeEqual(a, b);
|
|
133
|
+
if (!ok) {
|
|
134
|
+
return c.html(renderConsent({
|
|
135
|
+
clientId: v.client.clientId,
|
|
136
|
+
clientName: v.client.clientName,
|
|
137
|
+
redirectUri: params.redirect_uri,
|
|
138
|
+
state: params.state ?? '',
|
|
139
|
+
codeChallenge: params.code_challenge,
|
|
140
|
+
codeChallengeMethod: 'S256',
|
|
141
|
+
scope: params.scope ?? '',
|
|
142
|
+
error: 'Invalid read secret.',
|
|
143
|
+
}), 401);
|
|
144
|
+
}
|
|
145
|
+
const code = store.issueCode({
|
|
146
|
+
clientId: v.client.clientId,
|
|
147
|
+
redirectUri: params.redirect_uri,
|
|
148
|
+
codeChallenge: params.code_challenge,
|
|
149
|
+
codeChallengeMethod: 'S256',
|
|
150
|
+
});
|
|
151
|
+
return c.redirect(appendParams(params.redirect_uri, { code, state: params.state ?? '' }), 302);
|
|
152
|
+
});
|
|
153
|
+
// --- Token endpoint (both grants) ----------------------------------------
|
|
154
|
+
app.post('/oauth/token', async (c) => {
|
|
155
|
+
let params;
|
|
156
|
+
const ct = c.req.header('content-type') ?? '';
|
|
157
|
+
try {
|
|
158
|
+
if (ct.includes('application/x-www-form-urlencoded')) {
|
|
159
|
+
params = new URLSearchParams(await c.req.text());
|
|
160
|
+
}
|
|
161
|
+
else if (ct.includes('application/json')) {
|
|
162
|
+
const body = (await c.req.json());
|
|
163
|
+
params = new URLSearchParams();
|
|
164
|
+
for (const [k, val] of Object.entries(body)) {
|
|
165
|
+
if (typeof val === 'string')
|
|
166
|
+
params.set(k, val);
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
else {
|
|
170
|
+
return c.json({ error: 'invalid_request', error_description: 'unsupported content-type' }, 400);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
catch {
|
|
174
|
+
return c.json({ error: 'invalid_request', error_description: 'malformed body' }, 400);
|
|
175
|
+
}
|
|
176
|
+
const grantType = params.get('grant_type') ?? '';
|
|
177
|
+
if (grantType === 'authorization_code') {
|
|
178
|
+
const result = grantAuthorizationCode({
|
|
179
|
+
grantType,
|
|
180
|
+
code: params.get('code') ?? '',
|
|
181
|
+
redirectUri: params.get('redirect_uri') ?? '',
|
|
182
|
+
clientId: params.get('client_id') ?? '',
|
|
183
|
+
codeVerifier: params.get('code_verifier') ?? '',
|
|
184
|
+
}, mcp, store);
|
|
185
|
+
if (!result.ok) {
|
|
186
|
+
return c.json({
|
|
187
|
+
error: result.error,
|
|
188
|
+
...(result.description ? { error_description: result.description } : {}),
|
|
189
|
+
}, result.status);
|
|
190
|
+
}
|
|
191
|
+
return c.json({
|
|
192
|
+
access_token: result.token.accessToken,
|
|
193
|
+
token_type: 'Bearer',
|
|
194
|
+
expires_in: result.token.expiresIn,
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
// RFC 6749 §2.3.1: client credentials may also be supplied via Basic auth.
|
|
198
|
+
let clientId = params.get('client_id') ?? '';
|
|
199
|
+
let clientSecret = params.get('client_secret') ?? '';
|
|
200
|
+
const basic = c.req.header('authorization');
|
|
201
|
+
if (basic?.toLowerCase().startsWith('basic ')) {
|
|
202
|
+
const decoded = Buffer.from(basic.slice(6), 'base64').toString('utf8');
|
|
203
|
+
const sep = decoded.indexOf(':');
|
|
204
|
+
if (sep > 0) {
|
|
205
|
+
if (!clientId)
|
|
206
|
+
clientId = decoded.slice(0, sep);
|
|
207
|
+
if (!clientSecret)
|
|
208
|
+
clientSecret = decoded.slice(sep + 1);
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
const result = grantClientCredentials({ clientId, clientSecret, grantType }, mcp, env.readSecret);
|
|
212
|
+
if (!result.ok) {
|
|
213
|
+
return c.json({
|
|
214
|
+
error: result.error,
|
|
215
|
+
...(result.description ? { error_description: result.description } : {}),
|
|
216
|
+
}, result.status);
|
|
217
|
+
}
|
|
218
|
+
return c.json({
|
|
219
|
+
access_token: result.token.accessToken,
|
|
220
|
+
token_type: 'Bearer',
|
|
221
|
+
expires_in: result.token.expiresIn,
|
|
222
|
+
});
|
|
223
|
+
});
|
|
224
|
+
// --- MCP endpoint (JSON-RPC over POST). Requires Bearer token. -----------
|
|
225
|
+
app.post('/', async (c) => {
|
|
226
|
+
const auth = c.req.header('authorization') ?? '';
|
|
227
|
+
const issuer = issuerFrom(c.req.url);
|
|
228
|
+
if (!auth.toLowerCase().startsWith('bearer ')) {
|
|
229
|
+
c.header('WWW-Authenticate', `Bearer realm="MCP", resource_metadata="${issuer}/.well-known/oauth-protected-resource"`);
|
|
230
|
+
return c.json({ error: 'unauthorized' }, 401);
|
|
231
|
+
}
|
|
232
|
+
const token = auth.slice(7).trim();
|
|
233
|
+
const v = verifyToken(token, mcp);
|
|
234
|
+
if (!v.ok) {
|
|
235
|
+
c.header('WWW-Authenticate', `Bearer realm="MCP", error="invalid_token", resource_metadata="${issuer}/.well-known/oauth-protected-resource"`);
|
|
236
|
+
return c.json({ error: 'unauthorized', reason: v.reason }, 401);
|
|
237
|
+
}
|
|
238
|
+
let body;
|
|
239
|
+
try {
|
|
240
|
+
body = await c.req.json();
|
|
241
|
+
}
|
|
242
|
+
catch {
|
|
243
|
+
return c.json({ jsonrpc: '2.0', id: null, error: { code: -32700, message: 'parse error' } }, 400);
|
|
244
|
+
}
|
|
245
|
+
const response = await server.handle(body);
|
|
246
|
+
return c.json(response);
|
|
247
|
+
});
|
|
248
|
+
// --- Root-level discovery aliases ----------------------------------------
|
|
249
|
+
const discovery = new Hono();
|
|
250
|
+
discovery.get('/.well-known/oauth-authorization-server', (c) => c.json(authMetadata(issuerFrom(c.req.url))));
|
|
251
|
+
discovery.get('/.well-known/oauth-protected-resource', (c) => c.json(resourceMetadata(issuerFrom(c.req.url))));
|
|
252
|
+
// Per draft RFC 9728: the resource server may publish PRM at a path
|
|
253
|
+
// matching the resource URL, e.g. /.well-known/oauth-protected-resource/mcp.
|
|
254
|
+
discovery.get('/.well-known/oauth-protected-resource/mcp', (c) => c.json(resourceMetadata(issuerFrom(c.req.url))));
|
|
255
|
+
return { mcp: app, discovery };
|
|
256
|
+
}
|
|
257
|
+
function validateAuthorizeParams(q, store) {
|
|
258
|
+
const clientId = q.client_id;
|
|
259
|
+
const redirectUri = q.redirect_uri;
|
|
260
|
+
if (!clientId)
|
|
261
|
+
return { kind: 'fatal', message: 'missing client_id' };
|
|
262
|
+
if (!redirectUri)
|
|
263
|
+
return { kind: 'fatal', message: 'missing redirect_uri' };
|
|
264
|
+
const client = store.getClient(clientId);
|
|
265
|
+
if (!client)
|
|
266
|
+
return { kind: 'fatal', message: 'unknown client_id' };
|
|
267
|
+
if (!client.redirectUris.includes(redirectUri)) {
|
|
268
|
+
return { kind: 'fatal', message: 'redirect_uri not registered for this client_id' };
|
|
269
|
+
}
|
|
270
|
+
if (q.response_type !== 'code') {
|
|
271
|
+
return {
|
|
272
|
+
kind: 'redirect',
|
|
273
|
+
redirectUri,
|
|
274
|
+
error: 'unsupported_response_type',
|
|
275
|
+
message: 'only response_type=code is supported',
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
if (!q.code_challenge) {
|
|
279
|
+
return {
|
|
280
|
+
kind: 'redirect',
|
|
281
|
+
redirectUri,
|
|
282
|
+
error: 'invalid_request',
|
|
283
|
+
message: 'code_challenge is required (PKCE)',
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
if (q.code_challenge_method !== 'S256') {
|
|
287
|
+
return {
|
|
288
|
+
kind: 'redirect',
|
|
289
|
+
redirectUri,
|
|
290
|
+
error: 'invalid_request',
|
|
291
|
+
message: 'only code_challenge_method=S256 is supported',
|
|
292
|
+
};
|
|
293
|
+
}
|
|
294
|
+
return { kind: 'ok', client };
|
|
295
|
+
}
|
|
296
|
+
function appendParams(base, extra) {
|
|
297
|
+
const u = new URL(base);
|
|
298
|
+
for (const [k, v] of Object.entries(extra)) {
|
|
299
|
+
if (v !== '')
|
|
300
|
+
u.searchParams.set(k, v);
|
|
301
|
+
}
|
|
302
|
+
return u.toString();
|
|
303
|
+
}
|
|
304
|
+
function isAcceptableRedirectUri(uri) {
|
|
305
|
+
try {
|
|
306
|
+
const parsed = new URL(uri);
|
|
307
|
+
const isLoopback = parsed.protocol === 'http:' &&
|
|
308
|
+
(parsed.hostname === '127.0.0.1' || parsed.hostname === 'localhost');
|
|
309
|
+
return parsed.protocol === 'https:' || isLoopback;
|
|
310
|
+
}
|
|
311
|
+
catch {
|
|
312
|
+
return false;
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
//# sourceMappingURL=mcp.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mcp.js","sourceRoot":"","sources":["../../src/routes/mcp.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@zykeco/sync-server",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.7.0",
|
|
4
4
|
"description": "Self-hosted Zyke sync server (Hono + Drizzle + libsql). Imports to auto-start; ships a migrate CLI.",
|
|
5
5
|
"license": "AGPL-3.0-or-later",
|
|
6
6
|
"author": "P2LS9K GmbH <office@p2ls9k.com>",
|
|
@@ -43,7 +43,7 @@
|
|
|
43
43
|
"scripts": {
|
|
44
44
|
"build": "tsc -p tsconfig.json && chmod +x dist/migrate-cli.js dist/startup-cli.js dist/upgrade-cli.js",
|
|
45
45
|
"typecheck": "tsc -p tsconfig.json --noEmit",
|
|
46
|
-
"test": "node --test",
|
|
46
|
+
"test": "node --import tsx --test 'test/**/*.test.ts'",
|
|
47
47
|
"dev": "tsx --watch src/index.ts",
|
|
48
48
|
"start": "node dist/index.js",
|
|
49
49
|
"clean": "rm -rf dist .tsbuildinfo",
|