@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.
@@ -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":"AAAA;;;;;;;GAOG;AACH,MAAM,UAAU,WAAW,CAAC,KAAa,EAAE,aAAqB;IAC9D,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,EAAE,CAAC;QAC/D,MAAM,IAAI,KAAK,CAAC,kCAAkC,KAAK,KAAK,aAAa,GAAG,CAAC,CAAC;IAChF,CAAC;IACD,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,KAAK,GAAG,aAAa,GAAG,MAAM,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IAClE,MAAM,EAAE,GAAG,CAAC,OAAO,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IACnE,MAAM,EAAE,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IAC5D,MAAM,EAAE,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IAC7D,MAAM,EAAE,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IAC/D,MAAM,EAAE,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IAC/D,MAAM,IAAI,GAAG,aAAa,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;IAC5C,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;IACpC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,EAAE,CAAC;SAC9B,QAAQ,EAAE;SACV,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IACpB,MAAM,IAAI,GAAG,CAAC,GAAG,GAAG,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;IACpD,OAAO,GAAG,IAAI,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,GAAG,IAAI,GAAG,IAAI,IAAI,IAAI,EAAE,CAAC;AACvE,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,oBAAoB,CAAC,YAAoB;IACvD,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,CAAC;QACnC,MAAM,IAAI,KAAK,CAAC,2CAA2C,YAAY,GAAG,CAAC,CAAC;IAC9E,CAAC;IACD,MAAM,IAAI,GAAG,YAAY,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;IACzC,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC;IAC/C,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC;IACjC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;IACxC,MAAM,CAAC,GAAG,GAAG,GAAG,EAAE,CAAC;IACnB,OAAO,GAAG,IAAI,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,CAAC;SACjF,QAAQ,EAAE;SACV,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;AACxB,CAAC"}
@@ -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.5.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",