@maravilla-labs/vite-plugin 0.3.8 → 0.3.10

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,286 @@
1
+ // src/event-dispatcher.ts
2
+ import { existsSync, statSync } from "fs";
3
+ import { resolve, join } from "path";
4
+ import { pathToFileURL } from "url";
5
+
6
+ // src/event-matcher.ts
7
+ function matches(trigger, event) {
8
+ switch (trigger.kind) {
9
+ case "kv":
10
+ return matchKv(trigger, event);
11
+ case "db":
12
+ return matchDb(trigger, event);
13
+ case "auth":
14
+ return matchAuth(trigger, event);
15
+ case "channel":
16
+ return matchChannel(trigger, event);
17
+ case "storage":
18
+ return matchStorage(trigger, event);
19
+ case "deploy":
20
+ return matchDeploy(trigger, event);
21
+ case "ren":
22
+ return matchRen(trigger, event);
23
+ case "schedule":
24
+ return false;
25
+ case "queue":
26
+ return false;
27
+ default:
28
+ return false;
29
+ }
30
+ }
31
+ function matchKv(t, e) {
32
+ if (e.r !== "kv") return false;
33
+ if (t.op !== void 0) {
34
+ const eventOp = e.t === "kv.put" ? "put" : e.t === "kv.delete" ? "delete" : e.t === "kv.expired" ? "expired" : null;
35
+ if (eventOp !== t.op) return false;
36
+ }
37
+ if (t.namespace !== void 0 && e.ns !== t.namespace) return false;
38
+ if (t.keyPattern !== void 0) {
39
+ if (!e.k) return false;
40
+ if (!globMatch(t.keyPattern, e.k)) return false;
41
+ }
42
+ return true;
43
+ }
44
+ function matchDb(t, e) {
45
+ if (e.r !== "db") return false;
46
+ if (e.ns !== t.collection) return false;
47
+ if (t.op !== void 0) {
48
+ const expected = t.op === "insert" ? "db.document.created" : t.op === "update" ? "db.document.updated" : t.op === "delete" ? "db.document.deleted" : null;
49
+ if (e.t !== expected) return false;
50
+ }
51
+ return true;
52
+ }
53
+ function matchAuth(t, e) {
54
+ if (e.r !== "auth") return false;
55
+ if (t.op !== void 0) {
56
+ const expected = {
57
+ registered: "auth.user.registered",
58
+ loggedIn: "auth.user.logged_in",
59
+ loggedOut: "auth.user.logged_out",
60
+ loggedOutAll: "auth.user.logged_out_all",
61
+ deleted: "auth.user.deleted",
62
+ updated: "auth.user.updated"
63
+ }[t.op];
64
+ return e.t === expected;
65
+ }
66
+ return e.t.startsWith("auth.user.");
67
+ }
68
+ function matchChannel(t, e) {
69
+ if (e.r !== "realtime" && e.r !== "presence") return false;
70
+ if (!e.ch) return false;
71
+ if (!globMatch(t.channel, e.ch)) return false;
72
+ if (t.type !== void 0 && e.t !== t.type) return false;
73
+ return true;
74
+ }
75
+ function matchStorage(t, e) {
76
+ if (e.r !== "storage") return false;
77
+ if (t.op !== void 0) {
78
+ const eventOp = e.t === "storage.put" ? "put" : e.t === "storage.delete" ? "delete" : null;
79
+ if (eventOp !== t.op) return false;
80
+ }
81
+ if (t.keyPattern !== void 0) {
82
+ if (!e.k) return false;
83
+ if (!globMatch(t.keyPattern, e.k)) return false;
84
+ }
85
+ return true;
86
+ }
87
+ function matchDeploy(t, e) {
88
+ if (e.r !== "deploy") return false;
89
+ const expected = `deploy.${t.phase}`;
90
+ return e.t === expected;
91
+ }
92
+ function matchRen(t, e) {
93
+ const m = t.match;
94
+ if (m.r !== void 0 && e.r !== m.r) return false;
95
+ if (m.t !== void 0 && e.t !== m.t) return false;
96
+ if (m.ns !== void 0 && e.ns !== m.ns) return false;
97
+ return true;
98
+ }
99
+ function globMatch(pattern, input) {
100
+ if (pattern === "*") return true;
101
+ if (!pattern.includes("*")) return pattern === input;
102
+ const segments = pattern.split("*");
103
+ let cursor = 0;
104
+ const first = segments[0];
105
+ if (first && first.length > 0) {
106
+ if (!input.startsWith(first)) return false;
107
+ cursor = first.length;
108
+ }
109
+ for (let i = 1; i < segments.length - 1; i += 1) {
110
+ const seg = segments[i];
111
+ if (!seg || seg.length === 0) continue;
112
+ const idx = input.indexOf(seg, cursor);
113
+ if (idx === -1) return false;
114
+ cursor = idx + seg.length;
115
+ }
116
+ if (segments.length > 1) {
117
+ const last = segments[segments.length - 1];
118
+ if (last && last.length > 0) {
119
+ const tail = input.slice(cursor);
120
+ return tail.endsWith(last) && tail.length >= last.length;
121
+ }
122
+ }
123
+ return true;
124
+ }
125
+
126
+ // src/event-dispatcher.ts
127
+ async function startEventDispatcher(opts) {
128
+ const bundlePath = resolve(opts.projectDir, ".maravilla/events.js");
129
+ if (!existsSync(bundlePath)) {
130
+ console.warn(
131
+ "[maravilla][events] no .maravilla/events.js found \u2014 event auto-fire disabled. Run `pnpm build` (or `npm run build`) once to materialise the events bundle."
132
+ );
133
+ return () => {
134
+ };
135
+ }
136
+ let registry = {};
137
+ let bundleMtime = 0;
138
+ async function loadBundle() {
139
+ const stat = statSync(bundlePath);
140
+ if (stat.mtimeMs === bundleMtime) return;
141
+ try {
142
+ await import(pathToFileURL(bundlePath).href + `?t=${stat.mtimeMs}`);
143
+ const reg = globalThis.__maravilla_event_registry;
144
+ if (reg && typeof reg === "object") {
145
+ registry = reg;
146
+ bundleMtime = stat.mtimeMs;
147
+ const handlerCount = Object.keys(registry).length;
148
+ console.log(
149
+ `[maravilla][events] loaded ${handlerCount} handler${handlerCount === 1 ? "" : "s"} from .maravilla/events.js`
150
+ );
151
+ } else {
152
+ console.warn("[maravilla][events] events bundle did not expose __maravilla_event_registry");
153
+ }
154
+ } catch (err) {
155
+ console.error(
156
+ "[maravilla][events] failed to load events bundle:",
157
+ err?.message ?? err
158
+ );
159
+ }
160
+ }
161
+ await loadBundle();
162
+ const fsWatcher = (await import("fs")).watch(
163
+ join(opts.projectDir, ".maravilla"),
164
+ { persistent: false },
165
+ (event, filename) => {
166
+ if (filename === "events.js") {
167
+ setTimeout(() => {
168
+ void loadBundle();
169
+ }, 100);
170
+ }
171
+ }
172
+ );
173
+ let aborter = new AbortController();
174
+ let reconnectTimer = null;
175
+ let stopped = false;
176
+ async function connect() {
177
+ if (stopped) return;
178
+ aborter = new AbortController();
179
+ const url = `${opts.devServerUrl}/api/maravilla/ren`;
180
+ try {
181
+ const res = await fetch(url, {
182
+ headers: {
183
+ Accept: "text/event-stream",
184
+ "X-Tenant-Id": opts.tenant
185
+ },
186
+ signal: aborter.signal
187
+ });
188
+ if (!res.ok || !res.body) {
189
+ throw new Error(`SSE connect ${res.status}`);
190
+ }
191
+ const reader = res.body.getReader();
192
+ const decoder = new TextDecoder();
193
+ let buf = "";
194
+ while (!stopped) {
195
+ const { done, value } = await reader.read();
196
+ if (done) break;
197
+ buf += decoder.decode(value, { stream: true });
198
+ let idx;
199
+ while ((idx = buf.indexOf("\n\n")) !== -1) {
200
+ const frame = buf.slice(0, idx);
201
+ buf = buf.slice(idx + 2);
202
+ handleFrame(frame);
203
+ }
204
+ }
205
+ } catch (err) {
206
+ if (stopped) return;
207
+ const msg = err?.message ?? String(err);
208
+ if (!msg.includes("aborted")) {
209
+ console.warn(`[maravilla][events] REN SSE disconnected: ${msg}; reconnecting in 2s`);
210
+ reconnectTimer = setTimeout(() => {
211
+ void connect();
212
+ }, 2e3);
213
+ }
214
+ }
215
+ }
216
+ function handleFrame(frame) {
217
+ for (const line of frame.split("\n")) {
218
+ if (!line.startsWith("data:")) continue;
219
+ const payload = line.slice(5).trim();
220
+ if (!payload) continue;
221
+ let event;
222
+ try {
223
+ event = JSON.parse(payload);
224
+ } catch {
225
+ continue;
226
+ }
227
+ void dispatch(event);
228
+ }
229
+ }
230
+ async function dispatch(event) {
231
+ const platform = (await import("@maravilla-labs/platform")).getPlatform();
232
+ globalThis.platform = platform;
233
+ for (const [id, entry] of Object.entries(registry)) {
234
+ if (!matches(entry.trigger, event)) continue;
235
+ try {
236
+ await entry.handler(eventPayloadFor(event), {
237
+ platform,
238
+ handlerId: id,
239
+ tenant: opts.tenant,
240
+ traceId: globalThis.crypto?.randomUUID?.() ?? String(Date.now())
241
+ });
242
+ } catch (err) {
243
+ console.error(
244
+ `[maravilla][events] handler ${id} failed:`,
245
+ err?.message ?? err
246
+ );
247
+ }
248
+ }
249
+ }
250
+ function eventPayloadFor(event) {
251
+ switch (event.r) {
252
+ case "storage": {
253
+ const op = event.t.startsWith("storage.") ? event.t.slice("storage.".length) : event.t;
254
+ return { op, key: event.k, data: event.data, ts: event.ts };
255
+ }
256
+ case "kv": {
257
+ const op = event.t.startsWith("kv.") ? event.t.slice("kv.".length) : event.t;
258
+ return { op, namespace: event.ns, key: event.k, data: event.data, ts: event.ts };
259
+ }
260
+ case "db": {
261
+ const op = event.t.startsWith("db.document.") ? event.t.slice("db.document.".length) : event.t;
262
+ return { op, collection: event.ns, document: event.data, ts: event.ts };
263
+ }
264
+ case "auth": {
265
+ return { type: event.t, user_id: event.k, data: event.data, ts: event.ts };
266
+ }
267
+ case "realtime":
268
+ case "presence": {
269
+ return { type: event.t, channel: event.ch, data: event.data, ts: event.ts };
270
+ }
271
+ default:
272
+ return event;
273
+ }
274
+ }
275
+ void connect();
276
+ return () => {
277
+ stopped = true;
278
+ aborter?.abort();
279
+ if (reconnectTimer) clearTimeout(reconnectTimer);
280
+ fsWatcher.close();
281
+ };
282
+ }
283
+ export {
284
+ startEventDispatcher
285
+ };
286
+ //# sourceMappingURL=event-dispatcher-R5FS6WGE.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/event-dispatcher.ts","../src/event-matcher.ts"],"sourcesContent":["/**\n * Dev-mode event dispatcher. The runtime's\n * `crates/platform/src/events/dispatcher.rs` does this server-side in\n * Rust + Deno isolate. In dev there's no isolate, so we run the same\n * loop here in tenant Node.js (via the Vite dev server's process):\n *\n * 1. Load the events bundle the build pipeline produced\n * (`.maravilla/events.js` — sets `globalThis.__maravilla_event_registry`).\n * 2. Subscribe to the dev-server's REN SSE stream.\n * 3. For each `RenEvent`, run the matcher against every registered\n * trigger. Invoke matched handlers with `{ event, ctx: { platform } }`.\n *\n * Result: a tenant calling `platform.env.STORAGE.put('invites/.../videos/x.webm', ...)`\n * sees the synthesized transcode + thumbnail handlers fire in dev exactly\n * like they do in prod — same semantics, same timing model (after the put\n * returns, async via SSE), same content-addressed output keys.\n */\n\nimport { existsSync, statSync } from 'node:fs';\nimport { resolve, join } from 'node:path';\nimport { pathToFileURL } from 'node:url';\nimport { matches, type RenEvent, type Trigger } from './event-matcher.js';\n\ninterface RegistryEntry {\n trigger: Trigger;\n handler: (event: unknown, ctx: unknown) => unknown | Promise<unknown>;\n}\n\ntype Registry = Record<string, RegistryEntry>;\n\ninterface DispatcherOptions {\n /** Project root (where `.maravilla/` lives). */\n projectDir: string;\n /** URL of the Maravilla dev-server. */\n devServerUrl: string;\n /** Tenant id sent on outgoing requests + used to key the SSE stream. */\n tenant: string;\n}\n\n/**\n * Stand up a dev-mode event dispatcher. Returns a teardown function.\n * Safe to call when no events bundle exists — emits one warning and\n * no-ops until a build materialises `.maravilla/events.js`.\n */\nexport async function startEventDispatcher(opts: DispatcherOptions): Promise<() => void> {\n const bundlePath = resolve(opts.projectDir, '.maravilla/events.js');\n\n if (!existsSync(bundlePath)) {\n // Most demos / tenants don't have a build until the first\n // production-style build runs. Keep the dev server up; document\n // the expectation in a single-line warning.\n console.warn(\n '[maravilla][events] no .maravilla/events.js found — event auto-fire disabled. ' +\n 'Run `pnpm build` (or `npm run build`) once to materialise the events bundle.',\n );\n return () => {};\n }\n\n let registry: Registry = {};\n let bundleMtime = 0;\n\n async function loadBundle() {\n const stat = statSync(bundlePath);\n if (stat.mtimeMs === bundleMtime) return;\n try {\n // The bundle is IIFE — when imported it executes top-level and\n // sets `globalThis.__maravilla_event_registry` as a side effect.\n // Cache-bust via querystring so HMR rebuilds get picked up.\n await import(pathToFileURL(bundlePath).href + `?t=${stat.mtimeMs}`);\n const reg = (globalThis as { __maravilla_event_registry?: Registry })\n .__maravilla_event_registry;\n if (reg && typeof reg === 'object') {\n registry = reg;\n bundleMtime = stat.mtimeMs;\n const handlerCount = Object.keys(registry).length;\n console.log(\n `[maravilla][events] loaded ${handlerCount} handler${handlerCount === 1 ? '' : 's'} from .maravilla/events.js`,\n );\n } else {\n console.warn('[maravilla][events] events bundle did not expose __maravilla_event_registry');\n }\n } catch (err) {\n console.error(\n '[maravilla][events] failed to load events bundle:',\n (err as Error)?.message ?? err,\n );\n }\n }\n\n await loadBundle();\n\n // Watch the bundle for HMR-style refresh.\n const fsWatcher = (await import('node:fs')).watch(\n join(opts.projectDir, '.maravilla'),\n { persistent: false },\n (event, filename) => {\n if (filename === 'events.js') {\n // Slight debounce: filesystem may emit multiple `change` events\n // while esbuild rewrites the file.\n setTimeout(() => { void loadBundle(); }, 100);\n }\n },\n );\n\n // SSE consumer. Use the `eventsource` polyfill in Node — `EventSource`\n // is only available in browsers natively.\n let aborter: AbortController | null = new AbortController();\n let reconnectTimer: NodeJS.Timeout | null = null;\n let stopped = false;\n\n async function connect(): Promise<void> {\n if (stopped) return;\n aborter = new AbortController();\n const url = `${opts.devServerUrl}/api/maravilla/ren`;\n try {\n const res = await fetch(url, {\n headers: {\n Accept: 'text/event-stream',\n 'X-Tenant-Id': opts.tenant,\n },\n signal: aborter.signal,\n });\n if (!res.ok || !res.body) {\n throw new Error(`SSE connect ${res.status}`);\n }\n const reader = res.body.getReader();\n const decoder = new TextDecoder();\n let buf = '';\n // Read SSE frames. `data: <json>` lines are events; blank line ends\n // the frame.\n while (!stopped) {\n const { done, value } = await reader.read();\n if (done) break;\n buf += decoder.decode(value, { stream: true });\n let idx: number;\n while ((idx = buf.indexOf('\\n\\n')) !== -1) {\n const frame = buf.slice(0, idx);\n buf = buf.slice(idx + 2);\n handleFrame(frame);\n }\n }\n } catch (err) {\n if (stopped) return;\n const msg = (err as Error)?.message ?? String(err);\n if (!msg.includes('aborted')) {\n // Reconnect on transport errors. Constant 2s backoff is fine\n // for a dev tool — operator can ctrl-c if the loop is wrong.\n console.warn(`[maravilla][events] REN SSE disconnected: ${msg}; reconnecting in 2s`);\n reconnectTimer = setTimeout(() => { void connect(); }, 2000);\n }\n }\n }\n\n function handleFrame(frame: string): void {\n for (const line of frame.split('\\n')) {\n if (!line.startsWith('data:')) continue;\n const payload = line.slice(5).trim();\n if (!payload) continue;\n let event: RenEvent;\n try {\n event = JSON.parse(payload) as RenEvent;\n } catch {\n continue;\n }\n void dispatch(event);\n }\n }\n\n async function dispatch(event: RenEvent): Promise<void> {\n const platform = (await import('@maravilla-labs/platform')).getPlatform();\n // Stamp platform onto globalThis so the bundle's buildCtx() picks it\n // up — same shape the runtime uses inside the Deno isolate.\n (globalThis as { platform?: unknown }).platform = platform;\n\n for (const [id, entry] of Object.entries(registry)) {\n if (!matches(entry.trigger, event)) continue;\n try {\n await entry.handler(eventPayloadFor(event), {\n platform,\n handlerId: id,\n tenant: opts.tenant,\n traceId: globalThis.crypto?.randomUUID?.() ?? String(Date.now()),\n });\n } catch (err) {\n console.error(\n `[maravilla][events] handler ${id} failed:`,\n (err as Error)?.message ?? err,\n );\n }\n }\n }\n\n /**\n * Mirror `crates/platform/src/events/dispatcher.rs:163-247`'s payload\n * transform — handlers receive `{ op, key, data, ts }` for storage,\n * `{ namespace, key, value, ts, op }` for kv, etc., not the raw\n * RenEvent.\n */\n function eventPayloadFor(event: RenEvent): unknown {\n switch (event.r) {\n case 'storage': {\n const op = event.t.startsWith('storage.') ? event.t.slice('storage.'.length) : event.t;\n return { op, key: event.k, data: event.data, ts: event.ts };\n }\n case 'kv': {\n const op = event.t.startsWith('kv.') ? event.t.slice('kv.'.length) : event.t;\n return { op, namespace: event.ns, key: event.k, data: event.data, ts: event.ts };\n }\n case 'db': {\n const op = event.t.startsWith('db.document.') ? event.t.slice('db.document.'.length) : event.t;\n return { op, collection: event.ns, document: event.data, ts: event.ts };\n }\n case 'auth': {\n return { type: event.t, user_id: event.k, data: event.data, ts: event.ts };\n }\n case 'realtime':\n case 'presence': {\n return { type: event.t, channel: event.ch, data: event.data, ts: event.ts };\n }\n default:\n return event;\n }\n }\n\n void connect();\n\n return () => {\n stopped = true;\n aborter?.abort();\n if (reconnectTimer) clearTimeout(reconnectTimer);\n fsWatcher.close();\n };\n}\n","/**\n * TS port of `crates/platform/src/events/matcher.rs`. Used by the\n * dev-mode event dispatcher in this package to match incoming\n * `RenEvent`s against the trigger descriptors emitted in\n * `.maravilla/events.json`.\n *\n * Keep in lockstep with the Rust matcher — the golden-vector tests\n * across both implementations should produce the same yes/no for the\n * same (trigger, event) pair.\n */\n\nexport interface RenEvent {\n /** Event type, e.g. `\"storage.put\"`, `\"kv.put\"`, `\"presence.join\"` */\n t: string;\n /** Resource domain, e.g. `\"storage\"`, `\"kv\"`, `\"realtime\"`, `\"auth\"` */\n r: string;\n /** Resource key (e.g. storage path, kv key) */\n k?: string | null;\n /** Namespace / bucket / collection */\n ns?: string | null;\n /** Realtime channel name (for pub/sub) */\n ch?: string | null;\n /** Deployment id tagging the event origin */\n dep?: string | null;\n data?: unknown;\n ts?: number;\n}\n\n/** Trigger shapes mirror `crates/platform/src/events/types.rs` (camelCase). */\nexport type Trigger =\n | { kind: 'kv'; namespace?: string; keyPattern?: string; op?: 'put' | 'delete' | 'expired' }\n | { kind: 'db'; collection: string; op?: 'insert' | 'update' | 'delete' }\n | { kind: 'auth'; op?: string }\n | { kind: 'channel'; channel: string; type?: string }\n | { kind: 'storage'; keyPattern?: string; op?: 'put' | 'delete' }\n | { kind: 'deploy'; phase: 'ready' | 'draining' | 'stopped' }\n | { kind: 'ren'; match: { r?: string; t?: string; ns?: string } }\n | { kind: 'schedule'; cron: string } // never matches RenEvents\n | { kind: 'queue'; name: string }; // never matches RenEvents\n\n/**\n * Returns `true` if the event matches the trigger. Mirrors the Rust\n * `matches()` in `events/matcher.rs:12-34`.\n */\nexport function matches(trigger: Trigger, event: RenEvent): boolean {\n switch (trigger.kind) {\n case 'kv': return matchKv(trigger, event);\n case 'db': return matchDb(trigger, event);\n case 'auth': return matchAuth(trigger, event);\n case 'channel': return matchChannel(trigger, event);\n case 'storage': return matchStorage(trigger, event);\n case 'deploy': return matchDeploy(trigger, event);\n case 'ren': return matchRen(trigger, event);\n case 'schedule': return false;\n case 'queue': return false;\n default: return false;\n }\n}\n\nfunction matchKv(t: Extract<Trigger, { kind: 'kv' }>, e: RenEvent): boolean {\n if (e.r !== 'kv') return false;\n if (t.op !== undefined) {\n const eventOp = e.t === 'kv.put' ? 'put' : e.t === 'kv.delete' ? 'delete' : e.t === 'kv.expired' ? 'expired' : null;\n if (eventOp !== t.op) return false;\n }\n if (t.namespace !== undefined && e.ns !== t.namespace) return false;\n if (t.keyPattern !== undefined) {\n if (!e.k) return false;\n if (!globMatch(t.keyPattern, e.k)) return false;\n }\n return true;\n}\n\nfunction matchDb(t: Extract<Trigger, { kind: 'db' }>, e: RenEvent): boolean {\n if (e.r !== 'db') return false;\n if (e.ns !== t.collection) return false;\n if (t.op !== undefined) {\n const expected = t.op === 'insert' ? 'db.document.created'\n : t.op === 'update' ? 'db.document.updated'\n : t.op === 'delete' ? 'db.document.deleted' : null;\n if (e.t !== expected) return false;\n }\n return true;\n}\n\nfunction matchAuth(t: Extract<Trigger, { kind: 'auth' }>, e: RenEvent): boolean {\n if (e.r !== 'auth') return false;\n if (t.op !== undefined) {\n const expected = ({\n registered: 'auth.user.registered',\n loggedIn: 'auth.user.logged_in',\n loggedOut: 'auth.user.logged_out',\n loggedOutAll: 'auth.user.logged_out_all',\n deleted: 'auth.user.deleted',\n updated: 'auth.user.updated',\n } as Record<string, string>)[t.op];\n return e.t === expected;\n }\n return e.t.startsWith('auth.user.');\n}\n\nfunction matchChannel(t: Extract<Trigger, { kind: 'channel' }>, e: RenEvent): boolean {\n if (e.r !== 'realtime' && e.r !== 'presence') return false;\n if (!e.ch) return false;\n if (!globMatch(t.channel, e.ch)) return false;\n if (t.type !== undefined && e.t !== t.type) return false;\n return true;\n}\n\nfunction matchStorage(t: Extract<Trigger, { kind: 'storage' }>, e: RenEvent): boolean {\n if (e.r !== 'storage') return false;\n if (t.op !== undefined) {\n const eventOp = e.t === 'storage.put' ? 'put' : e.t === 'storage.delete' ? 'delete' : null;\n if (eventOp !== t.op) return false;\n }\n if (t.keyPattern !== undefined) {\n if (!e.k) return false;\n if (!globMatch(t.keyPattern, e.k)) return false;\n }\n return true;\n}\n\nfunction matchDeploy(t: Extract<Trigger, { kind: 'deploy' }>, e: RenEvent): boolean {\n if (e.r !== 'deploy') return false;\n const expected = `deploy.${t.phase}`;\n return e.t === expected;\n}\n\nfunction matchRen(t: Extract<Trigger, { kind: 'ren' }>, e: RenEvent): boolean {\n const m = t.match;\n if (m.r !== undefined && e.r !== m.r) return false;\n if (m.t !== undefined && e.t !== m.t) return false;\n if (m.ns !== undefined && e.ns !== m.ns) return false;\n return true;\n}\n\n/**\n * Minimal glob matcher supporting `*` (zero or more chars) and literal\n * segments. Mirrors `events/matcher.rs::glob_match`. Not full POSIX —\n * intentionally tiny. Handles `*`, `prefix:*`, `*:suffix`, `a:*:b`.\n */\nexport function globMatch(pattern: string, input: string): boolean {\n if (pattern === '*') return true;\n if (!pattern.includes('*')) return pattern === input;\n\n const segments = pattern.split('*');\n let cursor = 0;\n\n // First segment must match as prefix (unless empty, i.e. leading `*`).\n const first = segments[0];\n if (first && first.length > 0) {\n if (!input.startsWith(first)) return false;\n cursor = first.length;\n }\n\n // Middle segments: find each in order.\n for (let i = 1; i < segments.length - 1; i += 1) {\n const seg = segments[i];\n if (!seg || seg.length === 0) continue;\n const idx = input.indexOf(seg, cursor);\n if (idx === -1) return false;\n cursor = idx + seg.length;\n }\n\n // Last segment must match as suffix (unless empty, i.e. trailing `*`).\n if (segments.length > 1) {\n const last = segments[segments.length - 1];\n if (last && last.length > 0) {\n const tail = input.slice(cursor);\n return tail.endsWith(last) && tail.length >= last.length;\n }\n }\n return true;\n}\n"],"mappings":"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","names":[]}
package/dist/index.js CHANGED
@@ -320,6 +320,20 @@ function maravilla(options = {}) {
320
320
  return result;
321
321
  }
322
322
  });
323
+ try {
324
+ const { startEventDispatcher } = await import("./event-dispatcher-R5FS6WGE.js");
325
+ const teardown = await startEventDispatcher({
326
+ projectDir: process.cwd(),
327
+ devServerUrl,
328
+ tenant
329
+ });
330
+ server.httpServer?.once("close", () => teardown());
331
+ } catch (err) {
332
+ console.error(
333
+ "[maravilla][events] failed to start dispatcher:",
334
+ err?.message ?? err
335
+ );
336
+ }
323
337
  }
324
338
  };
325
339
  }
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/functions.ts","../src/client.ts","../src/index.ts"],"sourcesContent":["import type { Plugin, ViteDevServer } from 'vite';\nimport { buildFunctions, developmentServer } from '@maravilla-labs/functions';\nimport { join } from 'node:path';\n\nexport interface FunctionsPluginOptions {\n functionsDir?: string;\n functionsPort?: number;\n watch?: boolean;\n}\n\n/**\n * Vite plugin for Maravilla functions development\n * Handles building and serving functions with hot reload\n */\nexport function maravillaFunctions(options: FunctionsPluginOptions = {}): Plugin {\n const {\n functionsDir = 'functions',\n functionsPort = 3003,\n watch = true,\n } = options;\n\n let functionsBundle: any = null;\n let cleanupWatcher: (() => void) | null = null;\n\n return {\n name: 'vite-plugin-maravilla-functions',\n \n async configureServer(server: ViteDevServer) {\n const resolvedFunctionsDir = join(server.config.root, functionsDir);\n \n // Check if functions directory exists\n try {\n await import('node:fs/promises').then(fs => fs.access(resolvedFunctionsDir));\n } catch {\n console.log(`[maravilla-functions] No functions directory found at ${resolvedFunctionsDir}`);\n return;\n }\n\n console.log(`[maravilla-functions] Found functions at ${resolvedFunctionsDir}`);\n\n // Build functions initially\n try {\n functionsBundle = await buildFunctions({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n production: false,\n minify: false,\n });\n\n console.log(`[maravilla-functions] Built ${functionsBundle.functions} functions:`);\n functionsBundle.routes.forEach((route: any) => {\n console.log(` ${route.path} → ${route.name} (${route.methods.join(', ')})`);\n });\n } catch (error) {\n console.error('[maravilla-functions] Build failed:', error);\n return;\n }\n\n // Set up dev server with watcher\n if (watch) {\n const cleanup = await developmentServer({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n watch: true,\n onRebuild: async () => {\n // Reload functions bundle\n functionsBundle = await buildFunctions({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n production: false,\n minify: false,\n });\n \n // Notify Vite of the change\n server.ws.send({\n type: 'custom',\n event: 'maravilla:functions-reload',\n data: { routes: functionsBundle.routes }\n });\n }\n });\n cleanupWatcher = cleanup || null;\n }\n\n // Add middleware to handle function requests\n server.middlewares.use(async (req, res, next) => {\n // Only handle /api routes\n if (!req.url?.startsWith('/api')) {\n return next();\n }\n\n // Check if this matches a function route\n const matchedRoute = functionsBundle?.routes.find((route: any) => {\n const routePath = route.path;\n const requestPath = req.url?.split('?')[0];\n return routePath === requestPath || \n (routePath === '/api' && requestPath === '/api') ||\n (routePath === '/api' && requestPath === '/api/');\n });\n\n if (!matchedRoute) {\n return next();\n }\n\n // Check if method is allowed\n const method = req.method?.toUpperCase() || 'GET';\n if (!matchedRoute.methods.includes(method)) {\n res.statusCode = 405;\n res.setHeader('Allow', matchedRoute.methods.join(', '));\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({\n error: 'Method not allowed',\n allowed: matchedRoute.methods\n }));\n return;\n }\n\n console.log(`[maravilla-functions] ${method} ${req.url} → ${matchedRoute.name}`);\n\n // For now, execute the function in a simple way\n // In production, this would use the V8 isolate runtime\n try {\n const functionModule = await import(\n join(server.config.root, '.maravilla/functions-dev/functions.js')\n );\n\n // Create a mock Request object\n const url = new URL(req.url!, `http://localhost:${server.config.server?.port || 5173}`);\n const headers = new Headers();\n Object.entries(req.headers).forEach(([key, value]) => {\n if (typeof value === 'string') {\n headers.append(key, value);\n } else if (Array.isArray(value)) {\n value.forEach(v => headers.append(key, v));\n }\n });\n\n // Get body if present\n let body = null;\n if (req.method && ['POST', 'PUT', 'PATCH'].includes(req.method)) {\n const MAX_BODY_SIZE = 10 * 1024 * 1024; // 10MB\n body = await new Promise<string>((resolve, reject) => {\n let data = '';\n let size = 0;\n req.on('data', (chunk: Buffer | string) => {\n size += typeof chunk === 'string' ? Buffer.byteLength(chunk) : chunk.length;\n if (size > MAX_BODY_SIZE) {\n req.destroy(new Error('Request body too large'));\n reject(new Error('Request body too large'));\n return;\n }\n data += chunk;\n });\n req.on('end', () => resolve(data));\n req.on('error', reject);\n }).catch((err) => {\n res.statusCode = 413;\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({ error: 'Payload too large', message: err.message }));\n return null;\n });\n if (body === null) return;\n }\n\n const request = {\n url: url.toString(),\n method: req.method || 'GET',\n headers,\n json: body ? () => Promise.resolve(JSON.parse(body)) : undefined,\n text: () => Promise.resolve(body || ''),\n };\n\n // Call the function handler\n const response = await functionModule.handleFunctionRequest(request);\n\n // Send response\n res.statusCode = response.status || 200;\n Object.entries(response.headers || {}).forEach(([key, value]) => {\n res.setHeader(key, value as string);\n });\n \n if (response instanceof Response) {\n const responseBody = await response.text();\n res.end(responseBody);\n } else {\n res.end(JSON.stringify(response));\n }\n } catch (error: any) {\n console.error('[maravilla-functions] Execution error:', error);\n res.statusCode = 500;\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({\n error: 'Function execution failed',\n message: error.message,\n stack: process.env.NODE_ENV === 'development' ? error.stack : undefined\n }));\n }\n });\n },\n\n async closeBundle() {\n if (cleanupWatcher) {\n cleanupWatcher();\n }\n }\n };\n}","/**\n * Platform client for communicating with the dev server\n */\n\nexport interface PlatformClient {\n kv: KvClient;\n db: DbClient;\n}\n\nexport interface KvClient {\n get(namespace: string, key: string): Promise<any>;\n put(namespace: string, key: string, value: any, ttl?: number): Promise<void>;\n delete(namespace: string, key: string): Promise<void>;\n list(namespace: string, options?: KvListOptions): Promise<KvListResponse>;\n}\n\nexport interface DbClient {\n find(collection: string, filter?: any, options?: DbFindOptions): Promise<any[]>;\n findOne(collection: string, filter: any): Promise<any | null>;\n insertOne(collection: string, document: any): Promise<string>;\n updateOne(collection: string, filter: any, update: any): Promise<{ modified: number }>;\n deleteOne(collection: string, filter: any): Promise<{ deleted: number }>;\n}\n\nexport interface KvListOptions {\n prefix?: string;\n limit?: number;\n cursor?: string;\n}\n\nexport interface KvListResponse {\n success: boolean;\n result: Array<{ name: string; expiration?: number; metadata?: any }>;\n result_info: {\n count: number;\n cursor?: string;\n };\n}\n\nexport interface DbFindOptions {\n limit?: number;\n skip?: number;\n sort?: any;\n}\n\nexport function createPlatformClient(baseUrl: string, tenant?: string): PlatformClient {\n const tenantId = tenant || 'dev-tenant';\n const headers = {\n 'Content-Type': 'application/json',\n 'X-Tenant-Id': tenantId,\n };\n\n const fetchWithError = async (url: string, options: RequestInit = {}) => {\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n ...headers,\n ...options.headers,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new Error(`Platform API error: ${response.status} - ${error}`);\n }\n\n return response;\n } catch (error) {\n if (error instanceof TypeError && error.message.includes('fetch')) {\n throw new Error(\n `Failed to connect to Maravilla dev server at ${baseUrl}. ` +\n 'Please ensure the dev server is running (cargo run -p dev-server)'\n );\n }\n throw error;\n }\n };\n\n const kv: KvClient = {\n async get(namespace: string, key: string) {\n const response = await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`);\n if (response.status === 404) return null;\n return response.json();\n },\n\n async put(namespace: string, key: string, value: any, ttl?: number) {\n const requestHeaders: Record<string, string> = { ...headers };\n if (ttl) {\n requestHeaders['X-TTL'] = ttl.toString();\n }\n\n await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`, {\n method: 'PUT',\n headers: requestHeaders,\n body: JSON.stringify(value),\n });\n },\n\n async delete(namespace: string, key: string) {\n await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`, {\n method: 'DELETE',\n });\n },\n\n async list(namespace: string, options: KvListOptions = {}) {\n const response = await fetchWithError(`${baseUrl}/api/kv/${namespace}`, {\n method: 'POST',\n body: JSON.stringify(options),\n });\n return response.json() as Promise<KvListResponse>;\n },\n };\n\n const db: DbClient = {\n async find(collection: string, filter: any = {}, options: DbFindOptions = {}) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}`, {\n method: 'POST',\n body: JSON.stringify({ filter, options }),\n });\n return response.json() as Promise<any[]>;\n },\n\n async findOne(collection: string, filter: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/findOne`, {\n method: 'POST',\n body: JSON.stringify(filter),\n });\n if (response.status === 404) return null;\n return response.json();\n },\n\n async insertOne(collection: string, document: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}`, {\n method: 'PUT',\n body: JSON.stringify(document),\n });\n const result = await response.json() as { id: string };\n return result.id;\n },\n\n async updateOne(collection: string, filter: any, update: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/update`, {\n method: 'POST',\n body: JSON.stringify({ filter, update }),\n });\n return response.json() as Promise<{ modified: number }>;\n },\n\n async deleteOne(collection: string, filter: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/delete`, {\n method: 'DELETE',\n body: JSON.stringify(filter),\n });\n return response.json() as Promise<{ deleted: number }>;\n },\n };\n\n return { kv, db };\n}","import type { Plugin, ProxyOptions } from 'vite';\n\nexport interface MaravillaPluginOptions {\n /**\n * URL of the Maravilla dev server. Used both for env-var injection and\n * as the proxy target for `/_auth`, `/_assets`, `/_rt`, and `/api`.\n * @default 'http://localhost:3001'\n */\n devServerUrl?: string;\n\n /**\n * Tenant ID for development.\n * @default 'dev-tenant'\n */\n tenant?: string;\n\n /**\n * Extra URL prefixes to forward to the platform dev server, in addition\n * to the defaults (`/_auth`, `/_assets`, `/_rt`, `/api`). Each prefix is\n * proxied with `changeOrigin: true`. Use this for custom platform routes.\n */\n additionalProxies?: string[];\n\n /**\n * Disable the dev-server proxy entirely. Useful if your project routes\n * platform requests through some other mechanism.\n * @default false\n */\n disableProxy?: boolean;\n}\n\n/**\n * URL prefixes the dev-server owns. Anything the browser navigates to\n * directly (auth pages, assets) or upgrades (realtime ws) must be proxied\n * here so it lands on the platform server instead of the framework dev\n * server. Mirrors the prefixes registered in\n * `crates/dev-server/src/router.rs`.\n *\n * `/api/*` is intentionally enumerated by sub-prefix instead of the\n * blanket `/api`. The framework's own `+server.ts` routes (e.g.\n * `src/routes/api/v/[...key]/+server.ts` in demo's photo proxy) live\n * under `/api/...` too — proxying them all to the dev-server would\n * 404 every tenant-defined API route. Each entry below corresponds to\n * a router.rs registration.\n */\nconst DEFAULT_PROXY_PREFIXES = [\n '/_auth',\n '/_assets',\n '/_rt',\n '/api/kv',\n '/api/db',\n '/api/storage',\n '/api/realtime',\n '/api/workflows',\n '/api/push',\n '/api/media',\n '/api/maravilla',\n '/api/platform',\n '/api/tenant',\n '/api/events',\n] as const;\n\nexport function maravilla(options: MaravillaPluginOptions = {}): Plugin {\n const devServerUrl = options.devServerUrl || 'http://localhost:3001';\n const tenant = options.tenant || 'dev-tenant';\n const disableProxy = options.disableProxy ?? false;\n const additionalProxies = options.additionalProxies ?? [];\n\n return {\n name: 'vite-plugin-maravilla',\n\n config() {\n if (disableProxy) return undefined;\n const proxy: Record<string, ProxyOptions> = {};\n for (const prefix of [...DEFAULT_PROXY_PREFIXES, ...additionalProxies]) {\n // `/_rt/*` includes the WebSocket upgrade path `/_rt/ws`; enabling\n // `ws: true` is harmless for the plain HTTP routes under `/_rt`.\n proxy[prefix] = {\n target: devServerUrl,\n changeOrigin: true,\n ws: prefix === '/_rt',\n };\n }\n return { server: { proxy } };\n },\n\n async configureServer(server) {\n // Set environment variables that @maravilla/platform will use\n process.env.MARAVILLA_DEV_SERVER = devServerUrl;\n process.env.MARAVILLA_TENANT = tenant;\n\n console.log(`[maravilla] Platform APIs configured for ${devServerUrl}`);\n console.log('[maravilla] Using tenant:', tenant);\n if (!disableProxy) {\n const proxied = [...DEFAULT_PROXY_PREFIXES, ...additionalProxies].join(', ');\n console.log(`[maravilla] Proxying ${proxied} → ${devServerUrl}`);\n }\n\n // Auto-open a per-request AsyncLocalStorage scope before any\n // framework hook runs. This is the dev-mode equivalent of the\n // runtime's per-request REQUEST_CTX (tokio::task_local! at\n // crates/runtime/src/request_ctx.rs). With this in place,\n // tenant code calls `platform.auth.setCurrentUser(token)` from\n // SvelteKit's `handle` hook (or RR/Remix loaders) without any\n // Maravilla-specific wrapper — same code as prod.\n //\n // Loaded via dynamic import so non-Maravilla apps that pull\n // this plugin in for the proxy alone don't pay the cost until\n // a request actually arrives.\n const { runWithRequest } = await import('@maravilla-labs/platform');\n server.middlewares.use((_req, _res, next) => {\n runWithRequest(() => {\n next();\n }).catch((err) => {\n // Connect's next() doesn't return a Promise we can await;\n // any unhandled error from downstream surfaces here only\n // if runWithRequest's own machinery rejects. Re-throw on\n // the next tick so it shows up in the dev console.\n setImmediate(() => { throw err; });\n });\n });\n\n // Hook into SSR module resolution to ensure platform is available\n server.ssrLoadModule = new Proxy(server.ssrLoadModule, {\n async apply(target, thisArg, args) {\n const result = await Reflect.apply(target, thisArg, args);\n\n // Clear platform cache before each SSR module load to ensure fresh instance\n if ((globalThis as any).__maravilla_platform) {\n delete (globalThis as any).__maravilla_platform;\n }\n\n return result;\n }\n });\n }\n };\n}\n\n// Functions plugin\nexport { maravillaFunctions } from './functions.js';\nexport type { FunctionsPluginOptions } from './functions.js';\n\n// Legacy exports for backwards compatibility\nexport { createPlatformClient } from './client.js';\nexport type { PlatformClient, KvClient, DbClient } from './client.js';\n"],"mappings":"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","names":[]}
1
+ {"version":3,"sources":["../src/functions.ts","../src/client.ts","../src/index.ts"],"sourcesContent":["import type { Plugin, ViteDevServer } from 'vite';\nimport { buildFunctions, developmentServer } from '@maravilla-labs/functions';\nimport { join } from 'node:path';\n\nexport interface FunctionsPluginOptions {\n functionsDir?: string;\n functionsPort?: number;\n watch?: boolean;\n}\n\n/**\n * Vite plugin for Maravilla functions development\n * Handles building and serving functions with hot reload\n */\nexport function maravillaFunctions(options: FunctionsPluginOptions = {}): Plugin {\n const {\n functionsDir = 'functions',\n functionsPort = 3003,\n watch = true,\n } = options;\n\n let functionsBundle: any = null;\n let cleanupWatcher: (() => void) | null = null;\n\n return {\n name: 'vite-plugin-maravilla-functions',\n \n async configureServer(server: ViteDevServer) {\n const resolvedFunctionsDir = join(server.config.root, functionsDir);\n \n // Check if functions directory exists\n try {\n await import('node:fs/promises').then(fs => fs.access(resolvedFunctionsDir));\n } catch {\n console.log(`[maravilla-functions] No functions directory found at ${resolvedFunctionsDir}`);\n return;\n }\n\n console.log(`[maravilla-functions] Found functions at ${resolvedFunctionsDir}`);\n\n // Build functions initially\n try {\n functionsBundle = await buildFunctions({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n production: false,\n minify: false,\n });\n\n console.log(`[maravilla-functions] Built ${functionsBundle.functions} functions:`);\n functionsBundle.routes.forEach((route: any) => {\n console.log(` ${route.path} → ${route.name} (${route.methods.join(', ')})`);\n });\n } catch (error) {\n console.error('[maravilla-functions] Build failed:', error);\n return;\n }\n\n // Set up dev server with watcher\n if (watch) {\n const cleanup = await developmentServer({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n watch: true,\n onRebuild: async () => {\n // Reload functions bundle\n functionsBundle = await buildFunctions({\n functionsDir: resolvedFunctionsDir,\n outputDir: join(server.config.root, '.maravilla/functions-dev'),\n production: false,\n minify: false,\n });\n \n // Notify Vite of the change\n server.ws.send({\n type: 'custom',\n event: 'maravilla:functions-reload',\n data: { routes: functionsBundle.routes }\n });\n }\n });\n cleanupWatcher = cleanup || null;\n }\n\n // Add middleware to handle function requests\n server.middlewares.use(async (req, res, next) => {\n // Only handle /api routes\n if (!req.url?.startsWith('/api')) {\n return next();\n }\n\n // Check if this matches a function route\n const matchedRoute = functionsBundle?.routes.find((route: any) => {\n const routePath = route.path;\n const requestPath = req.url?.split('?')[0];\n return routePath === requestPath || \n (routePath === '/api' && requestPath === '/api') ||\n (routePath === '/api' && requestPath === '/api/');\n });\n\n if (!matchedRoute) {\n return next();\n }\n\n // Check if method is allowed\n const method = req.method?.toUpperCase() || 'GET';\n if (!matchedRoute.methods.includes(method)) {\n res.statusCode = 405;\n res.setHeader('Allow', matchedRoute.methods.join(', '));\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({\n error: 'Method not allowed',\n allowed: matchedRoute.methods\n }));\n return;\n }\n\n console.log(`[maravilla-functions] ${method} ${req.url} → ${matchedRoute.name}`);\n\n // For now, execute the function in a simple way\n // In production, this would use the V8 isolate runtime\n try {\n const functionModule = await import(\n join(server.config.root, '.maravilla/functions-dev/functions.js')\n );\n\n // Create a mock Request object\n const url = new URL(req.url!, `http://localhost:${server.config.server?.port || 5173}`);\n const headers = new Headers();\n Object.entries(req.headers).forEach(([key, value]) => {\n if (typeof value === 'string') {\n headers.append(key, value);\n } else if (Array.isArray(value)) {\n value.forEach(v => headers.append(key, v));\n }\n });\n\n // Get body if present\n let body = null;\n if (req.method && ['POST', 'PUT', 'PATCH'].includes(req.method)) {\n const MAX_BODY_SIZE = 10 * 1024 * 1024; // 10MB\n body = await new Promise<string>((resolve, reject) => {\n let data = '';\n let size = 0;\n req.on('data', (chunk: Buffer | string) => {\n size += typeof chunk === 'string' ? Buffer.byteLength(chunk) : chunk.length;\n if (size > MAX_BODY_SIZE) {\n req.destroy(new Error('Request body too large'));\n reject(new Error('Request body too large'));\n return;\n }\n data += chunk;\n });\n req.on('end', () => resolve(data));\n req.on('error', reject);\n }).catch((err) => {\n res.statusCode = 413;\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({ error: 'Payload too large', message: err.message }));\n return null;\n });\n if (body === null) return;\n }\n\n const request = {\n url: url.toString(),\n method: req.method || 'GET',\n headers,\n json: body ? () => Promise.resolve(JSON.parse(body)) : undefined,\n text: () => Promise.resolve(body || ''),\n };\n\n // Call the function handler\n const response = await functionModule.handleFunctionRequest(request);\n\n // Send response\n res.statusCode = response.status || 200;\n Object.entries(response.headers || {}).forEach(([key, value]) => {\n res.setHeader(key, value as string);\n });\n \n if (response instanceof Response) {\n const responseBody = await response.text();\n res.end(responseBody);\n } else {\n res.end(JSON.stringify(response));\n }\n } catch (error: any) {\n console.error('[maravilla-functions] Execution error:', error);\n res.statusCode = 500;\n res.setHeader('Content-Type', 'application/json');\n res.end(JSON.stringify({\n error: 'Function execution failed',\n message: error.message,\n stack: process.env.NODE_ENV === 'development' ? error.stack : undefined\n }));\n }\n });\n },\n\n async closeBundle() {\n if (cleanupWatcher) {\n cleanupWatcher();\n }\n }\n };\n}","/**\n * Platform client for communicating with the dev server\n */\n\nexport interface PlatformClient {\n kv: KvClient;\n db: DbClient;\n}\n\nexport interface KvClient {\n get(namespace: string, key: string): Promise<any>;\n put(namespace: string, key: string, value: any, ttl?: number): Promise<void>;\n delete(namespace: string, key: string): Promise<void>;\n list(namespace: string, options?: KvListOptions): Promise<KvListResponse>;\n}\n\nexport interface DbClient {\n find(collection: string, filter?: any, options?: DbFindOptions): Promise<any[]>;\n findOne(collection: string, filter: any): Promise<any | null>;\n insertOne(collection: string, document: any): Promise<string>;\n updateOne(collection: string, filter: any, update: any): Promise<{ modified: number }>;\n deleteOne(collection: string, filter: any): Promise<{ deleted: number }>;\n}\n\nexport interface KvListOptions {\n prefix?: string;\n limit?: number;\n cursor?: string;\n}\n\nexport interface KvListResponse {\n success: boolean;\n result: Array<{ name: string; expiration?: number; metadata?: any }>;\n result_info: {\n count: number;\n cursor?: string;\n };\n}\n\nexport interface DbFindOptions {\n limit?: number;\n skip?: number;\n sort?: any;\n}\n\nexport function createPlatformClient(baseUrl: string, tenant?: string): PlatformClient {\n const tenantId = tenant || 'dev-tenant';\n const headers = {\n 'Content-Type': 'application/json',\n 'X-Tenant-Id': tenantId,\n };\n\n const fetchWithError = async (url: string, options: RequestInit = {}) => {\n try {\n const response = await fetch(url, {\n ...options,\n headers: {\n ...headers,\n ...options.headers,\n },\n });\n\n if (!response.ok && response.status !== 404) {\n const error = await response.text();\n throw new Error(`Platform API error: ${response.status} - ${error}`);\n }\n\n return response;\n } catch (error) {\n if (error instanceof TypeError && error.message.includes('fetch')) {\n throw new Error(\n `Failed to connect to Maravilla dev server at ${baseUrl}. ` +\n 'Please ensure the dev server is running (cargo run -p dev-server)'\n );\n }\n throw error;\n }\n };\n\n const kv: KvClient = {\n async get(namespace: string, key: string) {\n const response = await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`);\n if (response.status === 404) return null;\n return response.json();\n },\n\n async put(namespace: string, key: string, value: any, ttl?: number) {\n const requestHeaders: Record<string, string> = { ...headers };\n if (ttl) {\n requestHeaders['X-TTL'] = ttl.toString();\n }\n\n await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`, {\n method: 'PUT',\n headers: requestHeaders,\n body: JSON.stringify(value),\n });\n },\n\n async delete(namespace: string, key: string) {\n await fetchWithError(`${baseUrl}/api/kv/${namespace}/${key}`, {\n method: 'DELETE',\n });\n },\n\n async list(namespace: string, options: KvListOptions = {}) {\n const response = await fetchWithError(`${baseUrl}/api/kv/${namespace}`, {\n method: 'POST',\n body: JSON.stringify(options),\n });\n return response.json() as Promise<KvListResponse>;\n },\n };\n\n const db: DbClient = {\n async find(collection: string, filter: any = {}, options: DbFindOptions = {}) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}`, {\n method: 'POST',\n body: JSON.stringify({ filter, options }),\n });\n return response.json() as Promise<any[]>;\n },\n\n async findOne(collection: string, filter: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/findOne`, {\n method: 'POST',\n body: JSON.stringify(filter),\n });\n if (response.status === 404) return null;\n return response.json();\n },\n\n async insertOne(collection: string, document: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}`, {\n method: 'PUT',\n body: JSON.stringify(document),\n });\n const result = await response.json() as { id: string };\n return result.id;\n },\n\n async updateOne(collection: string, filter: any, update: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/update`, {\n method: 'POST',\n body: JSON.stringify({ filter, update }),\n });\n return response.json() as Promise<{ modified: number }>;\n },\n\n async deleteOne(collection: string, filter: any) {\n const response = await fetchWithError(`${baseUrl}/api/db/${collection}/delete`, {\n method: 'DELETE',\n body: JSON.stringify(filter),\n });\n return response.json() as Promise<{ deleted: number }>;\n },\n };\n\n return { kv, db };\n}","import type { Plugin, ProxyOptions } from 'vite';\n\nexport interface MaravillaPluginOptions {\n /**\n * URL of the Maravilla dev server. Used both for env-var injection and\n * as the proxy target for `/_auth`, `/_assets`, `/_rt`, and `/api`.\n * @default 'http://localhost:3001'\n */\n devServerUrl?: string;\n\n /**\n * Tenant ID for development.\n * @default 'dev-tenant'\n */\n tenant?: string;\n\n /**\n * Extra URL prefixes to forward to the platform dev server, in addition\n * to the defaults (`/_auth`, `/_assets`, `/_rt`, `/api`). Each prefix is\n * proxied with `changeOrigin: true`. Use this for custom platform routes.\n */\n additionalProxies?: string[];\n\n /**\n * Disable the dev-server proxy entirely. Useful if your project routes\n * platform requests through some other mechanism.\n * @default false\n */\n disableProxy?: boolean;\n}\n\n/**\n * URL prefixes the dev-server owns. Anything the browser navigates to\n * directly (auth pages, assets) or upgrades (realtime ws) must be proxied\n * here so it lands on the platform server instead of the framework dev\n * server. Mirrors the prefixes registered in\n * `crates/dev-server/src/router.rs`.\n *\n * `/api/*` is intentionally enumerated by sub-prefix instead of the\n * blanket `/api`. The framework's own `+server.ts` routes (e.g.\n * `src/routes/api/v/[...key]/+server.ts` in demo's photo proxy) live\n * under `/api/...` too — proxying them all to the dev-server would\n * 404 every tenant-defined API route. Each entry below corresponds to\n * a router.rs registration.\n */\nconst DEFAULT_PROXY_PREFIXES = [\n '/_auth',\n '/_assets',\n '/_rt',\n '/api/kv',\n '/api/db',\n '/api/storage',\n '/api/realtime',\n '/api/workflows',\n '/api/push',\n '/api/media',\n '/api/maravilla',\n '/api/platform',\n '/api/tenant',\n '/api/events',\n] as const;\n\nexport function maravilla(options: MaravillaPluginOptions = {}): Plugin {\n const devServerUrl = options.devServerUrl || 'http://localhost:3001';\n const tenant = options.tenant || 'dev-tenant';\n const disableProxy = options.disableProxy ?? false;\n const additionalProxies = options.additionalProxies ?? [];\n\n return {\n name: 'vite-plugin-maravilla',\n\n config() {\n if (disableProxy) return undefined;\n const proxy: Record<string, ProxyOptions> = {};\n for (const prefix of [...DEFAULT_PROXY_PREFIXES, ...additionalProxies]) {\n // `/_rt/*` includes the WebSocket upgrade path `/_rt/ws`; enabling\n // `ws: true` is harmless for the plain HTTP routes under `/_rt`.\n proxy[prefix] = {\n target: devServerUrl,\n changeOrigin: true,\n ws: prefix === '/_rt',\n };\n }\n return { server: { proxy } };\n },\n\n async configureServer(server) {\n // Set environment variables that @maravilla/platform will use\n process.env.MARAVILLA_DEV_SERVER = devServerUrl;\n process.env.MARAVILLA_TENANT = tenant;\n\n console.log(`[maravilla] Platform APIs configured for ${devServerUrl}`);\n console.log('[maravilla] Using tenant:', tenant);\n if (!disableProxy) {\n const proxied = [...DEFAULT_PROXY_PREFIXES, ...additionalProxies].join(', ');\n console.log(`[maravilla] Proxying ${proxied} → ${devServerUrl}`);\n }\n\n // Auto-open a per-request AsyncLocalStorage scope before any\n // framework hook runs. This is the dev-mode equivalent of the\n // runtime's per-request REQUEST_CTX (tokio::task_local! at\n // crates/runtime/src/request_ctx.rs). With this in place,\n // tenant code calls `platform.auth.setCurrentUser(token)` from\n // SvelteKit's `handle` hook (or RR/Remix loaders) without any\n // Maravilla-specific wrapper — same code as prod.\n //\n // Loaded via dynamic import so non-Maravilla apps that pull\n // this plugin in for the proxy alone don't pay the cost until\n // a request actually arrives.\n const { runWithRequest } = await import('@maravilla-labs/platform');\n server.middlewares.use((_req, _res, next) => {\n runWithRequest(() => {\n next();\n }).catch((err) => {\n // Connect's next() doesn't return a Promise we can await;\n // any unhandled error from downstream surfaces here only\n // if runWithRequest's own machinery rejects. Re-throw on\n // the next tick so it shows up in the dev console.\n setImmediate(() => { throw err; });\n });\n });\n\n // Hook into SSR module resolution to ensure platform is available\n server.ssrLoadModule = new Proxy(server.ssrLoadModule, {\n async apply(target, thisArg, args) {\n const result = await Reflect.apply(target, thisArg, args);\n\n // Clear platform cache before each SSR module load to ensure fresh instance\n if ((globalThis as any).__maravilla_platform) {\n delete (globalThis as any).__maravilla_platform;\n }\n\n return result;\n }\n });\n\n // Dev-mode events auto-fire dispatcher. Subscribes to dev-server's\n // REN SSE and invokes any matching handler from `.maravilla/events.js`\n // (synthesized from `transforms:` config + hand-written `events/*.ts`).\n // Mirrors prod's `EventDispatcher` so STORAGE.put on a configured\n // pattern auto-runs transcode/thumbnail/etc. without an explicit\n // call from app code.\n try {\n const { startEventDispatcher } = await import('./event-dispatcher.js');\n const teardown = await startEventDispatcher({\n projectDir: process.cwd(),\n devServerUrl,\n tenant,\n });\n // Vite's `server.httpServer.once('close')` fires on `vite dev` exit.\n server.httpServer?.once('close', () => teardown());\n } catch (err) {\n console.error(\n '[maravilla][events] failed to start dispatcher:',\n (err as Error)?.message ?? err,\n );\n }\n }\n };\n}\n\n// Functions plugin\nexport { maravillaFunctions } from './functions.js';\nexport type { FunctionsPluginOptions } from './functions.js';\n\n// Legacy exports for backwards compatibility\nexport { createPlatformClient } from './client.js';\nexport type { PlatformClient, KvClient, DbClient } from './client.js';\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@maravilla-labs/vite-plugin",
3
- "version": "0.3.8",
3
+ "version": "0.3.10",
4
4
  "description": "Vite plugin for Maravilla Runtime development",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
@@ -24,7 +24,7 @@
24
24
  "node-fetch": "^3.3.2"
25
25
  },
26
26
  "devDependencies": {
27
- "@maravilla-labs/platform": "^0.3.8",
27
+ "@maravilla-labs/platform": "^0.3.7",
28
28
  "@types/node": "^20.10.5",
29
29
  "tsup": "^8.0.1",
30
30
  "typescript": "^5.3.3",