appflare 0.2.30 → 0.2.32

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (140) hide show
  1. package/Documentation.md +758 -758
  2. package/cli/commands/index.ts +238 -238
  3. package/cli/generate.ts +243 -178
  4. package/cli/index.ts +120 -120
  5. package/cli/load-config.ts +184 -184
  6. package/cli/schema-compiler.ts +1183 -1183
  7. package/cli/templates/auth/README.md +156 -156
  8. package/cli/templates/auth/config.ts +61 -61
  9. package/cli/templates/auth/route-config.ts +1 -1
  10. package/cli/templates/auth/route-handler.ts +1 -1
  11. package/cli/templates/auth/route-request-utils.ts +5 -5
  12. package/cli/templates/auth/route.config.ts +18 -18
  13. package/cli/templates/auth/route.handler.ts +18 -18
  14. package/cli/templates/auth/route.request-utils.ts +55 -55
  15. package/cli/templates/auth/route.ts +14 -14
  16. package/cli/templates/core/README.md +266 -266
  17. package/cli/templates/core/app-creation.ts +19 -19
  18. package/cli/templates/core/client/appflare.ts +112 -112
  19. package/cli/templates/core/client/handlers/index.ts +748 -748
  20. package/cli/templates/core/client/handlers.ts +1 -1
  21. package/cli/templates/core/client/index.ts +7 -7
  22. package/cli/templates/core/client/storage.ts +195 -195
  23. package/cli/templates/core/client/types.ts +186 -186
  24. package/cli/templates/core/client-modules/appflare.ts +1 -1
  25. package/cli/templates/core/client-modules/handlers.ts +1 -1
  26. package/cli/templates/core/client-modules/index.ts +1 -1
  27. package/cli/templates/core/client-modules/storage.ts +1 -1
  28. package/cli/templates/core/client-modules/types.ts +1 -1
  29. package/cli/templates/core/client.artifacts.ts +39 -39
  30. package/cli/templates/core/client.ts +4 -4
  31. package/cli/templates/core/drizzle.ts +15 -15
  32. package/cli/templates/core/export.ts +14 -14
  33. package/cli/templates/core/handlers.route.ts +24 -24
  34. package/cli/templates/core/handlers.ts +1 -1
  35. package/cli/templates/core/imports.ts +9 -9
  36. package/cli/templates/core/server.ts +38 -38
  37. package/cli/templates/core/types.ts +6 -6
  38. package/cli/templates/core/wrangler.ts +109 -109
  39. package/cli/templates/dashboard/builders/functions/index.ts +17 -17
  40. package/cli/templates/dashboard/builders/functions/render-page/header.ts +20 -20
  41. package/cli/templates/dashboard/builders/functions/render-page/index.ts +33 -33
  42. package/cli/templates/dashboard/builders/functions/render-page/request-panel.ts +171 -171
  43. package/cli/templates/dashboard/builders/functions/render-page/result-panel.ts +85 -85
  44. package/cli/templates/dashboard/builders/functions/render-page/scripts.ts +554 -554
  45. package/cli/templates/dashboard/builders/navigation.ts +122 -122
  46. package/cli/templates/dashboard/builders/storage/index.ts +13 -13
  47. package/cli/templates/dashboard/builders/storage/routes/create-directory-route.ts +29 -29
  48. package/cli/templates/dashboard/builders/storage/routes/delete-route.ts +18 -18
  49. package/cli/templates/dashboard/builders/storage/routes/download-route.ts +23 -23
  50. package/cli/templates/dashboard/builders/storage/routes/index.ts +22 -22
  51. package/cli/templates/dashboard/builders/storage/routes/list-route.ts +25 -25
  52. package/cli/templates/dashboard/builders/storage/routes/preview-route.ts +21 -21
  53. package/cli/templates/dashboard/builders/storage/routes/upload-route.ts +21 -21
  54. package/cli/templates/dashboard/builders/storage/runtime/helpers.ts +72 -72
  55. package/cli/templates/dashboard/builders/storage/runtime/storage-page.ts +130 -130
  56. package/cli/templates/dashboard/builders/table-routes/common/drawer-panel.ts +27 -27
  57. package/cli/templates/dashboard/builders/table-routes/common/pagination.ts +30 -30
  58. package/cli/templates/dashboard/builders/table-routes/common/search-bar.ts +23 -23
  59. package/cli/templates/dashboard/builders/table-routes/fragments.ts +217 -217
  60. package/cli/templates/dashboard/builders/table-routes/helpers.ts +45 -45
  61. package/cli/templates/dashboard/builders/table-routes/index.ts +8 -8
  62. package/cli/templates/dashboard/builders/table-routes/table/actions-cell.ts +71 -71
  63. package/cli/templates/dashboard/builders/table-routes/table/get-route.ts +291 -291
  64. package/cli/templates/dashboard/builders/table-routes/table/index.ts +80 -80
  65. package/cli/templates/dashboard/builders/table-routes/table/post-routes.ts +163 -163
  66. package/cli/templates/dashboard/builders/table-routes/table-route.ts +7 -7
  67. package/cli/templates/dashboard/builders/table-routes/users/get-route.ts +69 -69
  68. package/cli/templates/dashboard/builders/table-routes/users/html/modals.ts +57 -57
  69. package/cli/templates/dashboard/builders/table-routes/users/html/page.ts +27 -27
  70. package/cli/templates/dashboard/builders/table-routes/users/html/table.ts +128 -128
  71. package/cli/templates/dashboard/builders/table-routes/users/index.ts +32 -32
  72. package/cli/templates/dashboard/builders/table-routes/users/post-routes.ts +150 -150
  73. package/cli/templates/dashboard/builders/table-routes/users/redirect.ts +14 -14
  74. package/cli/templates/dashboard/builders/table-routes/users-route.ts +10 -10
  75. package/cli/templates/dashboard/components/dashboard-home.ts +23 -23
  76. package/cli/templates/dashboard/components/layout.ts +388 -388
  77. package/cli/templates/dashboard/components/login-page.ts +65 -65
  78. package/cli/templates/dashboard/index.ts +61 -61
  79. package/cli/templates/dashboard/types.ts +9 -9
  80. package/cli/templates/handlers/README.md +353 -353
  81. package/cli/templates/handlers/auth.ts +37 -37
  82. package/cli/templates/handlers/execution.ts +42 -42
  83. package/cli/templates/handlers/generators/context/context-creation.ts +101 -101
  84. package/cli/templates/handlers/generators/context/error-helpers.ts +11 -11
  85. package/cli/templates/handlers/generators/context/scheduler.ts +24 -24
  86. package/cli/templates/handlers/generators/context/storage-api.ts +82 -82
  87. package/cli/templates/handlers/generators/context/storage-helpers.ts +59 -59
  88. package/cli/templates/handlers/generators/context/types.ts +40 -40
  89. package/cli/templates/handlers/generators/context.ts +43 -43
  90. package/cli/templates/handlers/generators/execution.ts +15 -15
  91. package/cli/templates/handlers/generators/handlers.ts +13 -13
  92. package/cli/templates/handlers/generators/registration/modules/cron.ts +26 -26
  93. package/cli/templates/handlers/generators/registration/modules/realtime/auth.ts +75 -75
  94. package/cli/templates/handlers/generators/registration/modules/realtime/durable-object.ts +144 -144
  95. package/cli/templates/handlers/generators/registration/modules/realtime/index.ts +14 -14
  96. package/cli/templates/handlers/generators/registration/modules/realtime/publisher.ts +102 -102
  97. package/cli/templates/handlers/generators/registration/modules/realtime/routes.ts +164 -164
  98. package/cli/templates/handlers/generators/registration/modules/realtime/types.ts +30 -30
  99. package/cli/templates/handlers/generators/registration/modules/realtime/utils.ts +516 -516
  100. package/cli/templates/handlers/generators/registration/modules/scheduler.ts +56 -56
  101. package/cli/templates/handlers/generators/registration/modules/storage.ts +199 -199
  102. package/cli/templates/handlers/generators/registration/sections.ts +210 -210
  103. package/cli/templates/handlers/generators/types/context.ts +92 -92
  104. package/cli/templates/handlers/generators/types/core.ts +106 -106
  105. package/cli/templates/handlers/generators/types/operations.ts +135 -135
  106. package/cli/templates/handlers/generators/types/query-definitions/filter-and-where-types.ts +281 -259
  107. package/cli/templates/handlers/generators/types/query-definitions/query-api-types.ts +135 -135
  108. package/cli/templates/handlers/generators/types/query-definitions/query-helper-functions.ts +1103 -1031
  109. package/cli/templates/handlers/generators/types/query-definitions/schema-and-table-types.ts +278 -246
  110. package/cli/templates/handlers/generators/types/query-definitions.ts +13 -13
  111. package/cli/templates/handlers/generators/types/query-runtime/handled-error.ts +13 -13
  112. package/cli/templates/handlers/generators/types/query-runtime/runtime-aggregate-and-footer.ts +174 -174
  113. package/cli/templates/handlers/generators/types/query-runtime/runtime-read.ts +157 -121
  114. package/cli/templates/handlers/generators/types/query-runtime/runtime-setup.ts +45 -45
  115. package/cli/templates/handlers/generators/types/query-runtime/runtime-write.ts +697 -697
  116. package/cli/templates/handlers/generators/types/query-runtime.ts +15 -15
  117. package/cli/templates/handlers/index.ts +43 -43
  118. package/cli/templates/handlers/operations.ts +116 -116
  119. package/cli/templates/handlers/registration.ts +91 -91
  120. package/cli/templates/handlers/types.ts +15 -15
  121. package/cli/templates/handlers/utils.ts +48 -48
  122. package/cli/types.ts +110 -110
  123. package/cli/utils/handler-discovery.ts +466 -466
  124. package/cli/utils/json-utils.ts +24 -24
  125. package/cli/utils/path-utils.ts +19 -19
  126. package/cli/utils/schema-discovery.ts +399 -399
  127. package/dist/cli/index.d.mts +2 -0
  128. package/dist/cli/index.d.ts +2 -0
  129. package/dist/cli/index.js +366 -203
  130. package/dist/cli/index.mjs +366 -203
  131. package/index.ts +18 -18
  132. package/package.json +58 -58
  133. package/react/index.ts +5 -5
  134. package/react/use-infinite-query.ts +252 -252
  135. package/react/use-mutation.ts +89 -89
  136. package/react/use-query.ts +207 -207
  137. package/schema.ts +415 -415
  138. package/test-better-auth-hash.ts +2 -2
  139. package/tsconfig.json +6 -6
  140. package/tsup.config.ts +82 -82
@@ -1,554 +1,554 @@
1
- import { DiscoveredHandlerOperation } from "../../../../../utils/handler-discovery";
2
-
3
- export function renderScripts(h: DiscoveredHandlerOperation): string {
4
- const isQuery = h.kind === "query";
5
- const routePath = h.routePath;
6
- // handlerName matches the key used in realtimeQueryHandlers registry
7
- const realtimeQueryName = h.handlerName ?? h.routePath;
8
- // Only queries can be subscribed to via realtime
9
- const isRealtimeSupported = isQuery;
10
-
11
- return `
12
- <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/atom-one-dark.min.css">
13
- <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script>
14
-
15
- <script>
16
- // ── Tab switching ──────────────────────────────────────────────────────────
17
- window._currentResultTab = 'body';
18
-
19
- window.switchResultTab = function(tab) {
20
- window._currentResultTab = tab;
21
- ['body', 'headers', 'info', 'realtime'].forEach(function(t) {
22
- var pane = document.getElementById('result-tab-' + t);
23
- var btn = document.getElementById('tab-' + (t === 'headers' ? 'resp-headers' : t));
24
- if (pane) pane.classList.toggle('hidden', t !== tab);
25
- if (btn) btn.classList.toggle('tab-active', t === tab);
26
- });
27
- };
28
-
29
- window.switchRequestTab = function(tab) {
30
- ['args', 'auth', 'headers'].forEach(function(t) {
31
- var pane = document.getElementById('request-tab-' + t);
32
- var btn = document.getElementById('req-tab-btn-' + t);
33
- if (pane) pane.classList.toggle('hidden', t !== tab);
34
- if (btn) btn.classList.toggle('tab-active', t === tab);
35
- });
36
- };
37
-
38
- // ── Header key-value rows ──────────────────────────────────────────────────
39
- var _headerRowId = 0;
40
-
41
- window.addHeaderRow = function(key, value) {
42
- key = key || '';
43
- value = value || '';
44
- var id = ++_headerRowId;
45
- var container = document.getElementById('headers-rows');
46
-
47
- var row = document.createElement('div');
48
- row.id = 'header-row-' + id;
49
- row.className = 'flex items-center gap-2';
50
-
51
- var keyInput = document.createElement('input');
52
- keyInput.type = 'text';
53
- keyInput.placeholder = 'Header-Name';
54
- keyInput.setAttribute('data-hdr-key', '');
55
- keyInput.value = key;
56
- keyInput.className = 'input input-xs input-bordered font-mono w-2/5 bg-base-200/30 focus:bg-base-100 focus:border-primary transition-all rounded-lg border-base-200';
57
-
58
- var valInput = document.createElement('input');
59
- valInput.type = 'text';
60
- valInput.placeholder = 'value';
61
- valInput.setAttribute('data-hdr-val', '');
62
- valInput.value = value;
63
- valInput.className = 'input input-xs input-bordered font-mono flex-1 bg-base-200/30 focus:bg-base-100 focus:border-primary transition-all rounded-lg border-base-200';
64
-
65
- var removeBtn = document.createElement('button');
66
- removeBtn.type = 'button';
67
- removeBtn.className = 'btn btn-xs btn-ghost text-error opacity-40 hover:opacity-100 px-1.5';
68
- removeBtn.innerHTML = '<iconify-icon icon="solar:close-circle-linear" width="14" height="14"></iconify-icon>';
69
- removeBtn.onclick = function() { row.remove(); };
70
-
71
- row.appendChild(keyInput);
72
- row.appendChild(valInput);
73
- row.appendChild(removeBtn);
74
- container.appendChild(row);
75
- };
76
-
77
- // ── Collect args from server-rendered inputs ───────────────────────────────
78
- function collectArgs() {
79
- var result = {};
80
- document.querySelectorAll('#args-rows [data-arg-key]').forEach(function(el) {
81
- var key = el.getAttribute('data-arg-key');
82
- var type = el.getAttribute('data-arg-type');
83
- if (!key) return;
84
- var value;
85
- if (type === 'boolean') {
86
- value = el.checked;
87
- } else if (type === 'number') {
88
- var raw = el.value.trim();
89
- value = raw === '' ? undefined : Number(raw);
90
- } else {
91
- value = el.value;
92
- }
93
- if (value !== undefined && value !== '') {
94
- result[key] = value;
95
- }
96
- });
97
- return result;
98
- }
99
-
100
- function collectHeaders() {
101
- var result = {};
102
- document.querySelectorAll('#headers-rows [data-hdr-key]').forEach(function(keyEl) {
103
- var k = keyEl.value.trim();
104
- var v = keyEl.parentElement.querySelector('[data-hdr-val]').value;
105
- if (k) result[k] = v;
106
- });
107
- return result;
108
- }
109
-
110
- // ── Token visibility ──────────────────────────────────────────────────────
111
- window.toggleTokenVisibility = function() {
112
- var inp = document.getElementById('bearer-token-input');
113
- var icon = document.getElementById('token-eye-icon');
114
- if (!inp) return;
115
- if (inp.type === 'password') {
116
- inp.type = 'text';
117
- if (icon) icon.setAttribute('icon', 'solar:eye-closed-linear');
118
- } else {
119
- inp.type = 'password';
120
- if (icon) icon.setAttribute('icon', 'solar:eye-linear');
121
- }
122
- };
123
-
124
- // ── Render helpers ─────────────────────────────────────────────────────────
125
- function esc(str) {
126
- return String(str).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
127
- }
128
-
129
- function showBodyLoading() {
130
- var empty = document.getElementById('result-empty-state');
131
- if (empty) empty.classList.add('hidden');
132
- var content = document.getElementById('result-body-content');
133
- if (content) {
134
- content.classList.remove('hidden');
135
- content.innerHTML = '<div class="flex items-center justify-center h-full min-h-[200px]"><span class="loading loading-ring loading-md text-primary"></span></div>';
136
- }
137
- }
138
-
139
- function showBodyError(msg) {
140
- var empty = document.getElementById('result-empty-state');
141
- if (empty) empty.classList.add('hidden');
142
- var content = document.getElementById('result-body-content');
143
- if (content) {
144
- content.classList.remove('hidden');
145
- content.innerHTML = '<div class="p-5 text-error"><p class="font-bold mb-1 text-xs uppercase tracking-wider">Error</p><pre class="text-sm whitespace-pre-wrap">' + esc(msg) + '</pre></div>';
146
- }
147
- }
148
-
149
- function renderBodyJson(jsonStr) {
150
- var empty = document.getElementById('result-empty-state');
151
- if (empty) empty.classList.add('hidden');
152
- var content = document.getElementById('result-body-content');
153
- if (!content) return;
154
- content.classList.remove('hidden');
155
- content.innerHTML = '<pre style="margin:0;padding:1rem"><code class="language-json">' + esc(jsonStr) + '</code></pre>';
156
- if (window.hljs) hljs.highlightAll();
157
- }
158
-
159
- function renderBodyText(text) {
160
- var empty = document.getElementById('result-empty-state');
161
- if (empty) empty.classList.add('hidden');
162
- var content = document.getElementById('result-body-content');
163
- if (!content) return;
164
- content.classList.remove('hidden');
165
- content.innerHTML = '<pre style="margin:0;padding:1rem;white-space:pre-wrap;word-break:break-all">' + esc(text) + '</pre>';
166
- }
167
-
168
- function renderStatusBadge(status, elapsed) {
169
- var badge = document.getElementById('res-status-badge');
170
- var elBadge = document.getElementById('res-elapsed-badge');
171
- if (badge) {
172
- badge.className = 'badge badge-sm font-mono font-bold ' + (status < 300 ? 'badge-success' : status < 400 ? 'badge-warning' : 'badge-error');
173
- badge.textContent = status;
174
- badge.classList.remove('hidden');
175
- }
176
- if (elBadge) {
177
- elBadge.textContent = elapsed + 'ms';
178
- elBadge.classList.remove('hidden');
179
- }
180
- }
181
-
182
- function renderResponseHeaders(headers) {
183
- var el = document.getElementById('result-headers-content');
184
- if (!el) return;
185
- var entries = Object.entries(headers);
186
- if (!entries.length) {
187
- el.innerHTML = '<p class="text-xs opacity-40 italic p-2">No headers returned.</p>';
188
- return;
189
- }
190
- var rows = entries.map(function(pair) {
191
- return '<tr><td class="font-mono text-xs font-medium opacity-70">' + esc(pair[0]) + '</td><td class="font-mono text-xs opacity-50 break-all">' + esc(pair[1]) + '</td></tr>';
192
- }).join('');
193
- el.innerHTML = '<table class="table table-xs w-full"><thead><tr>' +
194
- '<th class="text-[10px] uppercase tracking-wider opacity-40 font-bold">Header</th>' +
195
- '<th class="text-[10px] uppercase tracking-wider opacity-40 font-bold">Value</th>' +
196
- '</tr></thead><tbody>' + rows + '</tbody></table>';
197
- }
198
-
199
- function renderInfo(data) {
200
- var el = document.getElementById('result-info-content');
201
- if (!el) return;
202
- var items = [
203
- { label: 'Status', value: data.status + ' ' + data.statusText, color: data.status < 300 ? 'text-success' : data.status < 400 ? 'text-warning' : 'text-error' },
204
- { label: 'Elapsed', value: data.elapsed + ' ms', color: '' },
205
- { label: 'Method', value: data.method, color: '' },
206
- { label: 'URL', value: data.url, color: '' },
207
- { label: 'Content-Type', value: data.contentType || '—', color: '' },
208
- ];
209
- var html = '<div class="flex flex-col gap-3 p-4">';
210
- items.forEach(function(item) {
211
- html += '<div class="flex flex-col gap-0.5">';
212
- html += '<span class="text-[10px] font-bold uppercase tracking-wider opacity-40">' + esc(item.label) + '</span>';
213
- html += '<span class="font-mono text-sm break-all ' + (item.color || 'opacity-70') + '">' + esc(String(item.value)) + '</span>';
214
- html += '</div>';
215
- });
216
- html += '</div>';
217
- el.innerHTML = html;
218
- }
219
-
220
- // ── Main execution ─────────────────────────────────────────────────────────
221
- window.executeFn = async function() {
222
- // If realtime is enabled, connect/reconnect after execute completes
223
- var shouldConnectRealtime = _rtEnabled && _isRealtimeSupported;
224
- var indicator = document.getElementById('execution-indicator');
225
- var headerErr = document.getElementById('headers-error');
226
-
227
- var args = collectArgs();
228
- var customHeaders = collectHeaders();
229
- var token = (document.getElementById('bearer-token-input') || {}).value || '';
230
-
231
- if (headerErr) headerErr.classList.add('hidden');
232
-
233
- // Show loading
234
- if (indicator) indicator.classList.remove('opacity-0');
235
- showBodyLoading();
236
-
237
- var isQuery = ${isQuery};
238
- var method = isQuery ? 'GET' : 'POST';
239
- var pathWithQuery = '${routePath}';
240
-
241
- if (isQuery && Object.keys(args).length > 0) {
242
- var params = new URLSearchParams();
243
- Object.entries(args).forEach(function(entry) {
244
- var key = entry[0];
245
- var value = entry[1];
246
- if (value !== undefined && value !== null) {
247
- if (Array.isArray(value)) {
248
- value.forEach(function(v) { params.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v)); });
249
- } else if (typeof value === 'object') {
250
- params.append(key, JSON.stringify(value));
251
- } else {
252
- params.append(key, String(value));
253
- }
254
- }
255
- });
256
- var qs = params.toString();
257
- if (qs) pathWithQuery += '?' + qs;
258
- }
259
-
260
- var fetchHeaders = Object.assign({ 'Content-Type': 'application/json' }, customHeaders);
261
- if (token) fetchHeaders['Authorization'] = 'Bearer ' + token;
262
-
263
- var startTime = performance.now();
264
-
265
- try {
266
- var res = await fetch(pathWithQuery, {
267
- method: method,
268
- headers: fetchHeaders,
269
- body: method === 'POST' ? JSON.stringify(args) : undefined,
270
- });
271
-
272
- var elapsed = Math.round(performance.now() - startTime);
273
- var contentType = res.headers.get('content-type') || '';
274
- var rawBody = await res.text();
275
-
276
- var parsedBody = rawBody;
277
- if (rawBody && contentType.includes('application/json')) {
278
- try { parsedBody = JSON.parse(rawBody); } catch(e) { parsedBody = rawBody; }
279
- }
280
-
281
- renderStatusBadge(res.status, elapsed);
282
-
283
- if (typeof parsedBody === 'object') {
284
- renderBodyJson(JSON.stringify(parsedBody, null, 2));
285
- } else {
286
- renderBodyText(rawBody);
287
- }
288
-
289
- renderResponseHeaders(Object.fromEntries(res.headers));
290
- renderInfo({ status: res.status, statusText: res.statusText, elapsed: elapsed, method: method, url: pathWithQuery, contentType: contentType });
291
-
292
- } catch(err) {
293
- showBodyError('Fetch error: ' + (err.message || String(err)));
294
- } finally {
295
- if (indicator) indicator.classList.add('opacity-0');
296
- // Connect realtime after execute if toggle is on
297
- if (shouldConnectRealtime) {
298
- switchResultTab('realtime');
299
- connectRealtime();
300
- }
301
- }
302
- };
303
-
304
- // ── Realtime WebSocket ─────────────────────────────────────────────────────
305
- var _rtSocket = null;
306
- var _rtToken = null;
307
- var _rtEnabled = false;
308
- var _rtEventCount = 0;
309
- var _isRealtimeSupported = ${isRealtimeSupported};
310
- var _realtimeQueryName = '${realtimeQueryName}';
311
- var _rtPingInterval = null;
312
-
313
- function setRtStatus(status) {
314
- // status: 'disconnected' | 'connecting' | 'connected' | 'error'
315
- var dot = document.getElementById('rt-status-dot');
316
- var headerDot = document.getElementById('rt-dot');
317
- var text = document.getElementById('rt-status-text');
318
- var colorMap = {
319
- disconnected: 'bg-base-content/20',
320
- connecting: 'bg-warning animate-pulse',
321
- connected: 'bg-success animate-pulse',
322
- error: 'bg-error',
323
- };
324
- var headerColorMap = {
325
- disconnected: 'bg-base-content/20',
326
- connecting: 'bg-warning animate-pulse',
327
- connected: 'bg-success animate-pulse',
328
- error: 'bg-error',
329
- };
330
- var labelMap = {
331
- disconnected: 'Disconnected',
332
- connecting: 'Connecting…',
333
- connected: 'Connected',
334
- error: 'Error',
335
- };
336
- if (dot) {
337
- dot.className = 'w-2 h-2 rounded-full transition-all duration-300 ' + (colorMap[status] || colorMap.disconnected);
338
- }
339
- if (headerDot) {
340
- headerDot.className = 'w-1.5 h-1.5 rounded-full transition-colors duration-300 ' + (headerColorMap[status] || headerColorMap.disconnected);
341
- }
342
- if (text) {
343
- text.textContent = labelMap[status] || status;
344
- }
345
- }
346
-
347
- function appendRtEvent(eventName, payload, ts) {
348
- var log = document.getElementById('rt-log');
349
- var empty = document.getElementById('rt-empty-state');
350
- if (!log) return;
351
- if (empty) empty.classList.add('hidden');
352
-
353
- _rtEventCount++;
354
- var countEl = document.getElementById('rt-event-count');
355
- if (countEl) {
356
- countEl.textContent = _rtEventCount + ' event' + (_rtEventCount !== 1 ? 's' : '');
357
- countEl.classList.remove('hidden');
358
- }
359
-
360
- var row = document.createElement('div');
361
- row.className = 'flex flex-col gap-0.5 p-2 rounded-lg bg-base-200/40 border border-base-200/60 hover:bg-base-200/70 transition-colors';
362
-
363
- var header = document.createElement('div');
364
- header.className = 'flex items-center justify-between';
365
-
366
- var nameSpan = document.createElement('span');
367
- nameSpan.className = 'font-semibold text-primary text-[11px] uppercase tracking-wider';
368
- nameSpan.textContent = eventName;
369
-
370
- var tsSpan = document.createElement('span');
371
- tsSpan.className = 'text-[10px] opacity-30 font-mono';
372
- tsSpan.textContent = new Date(ts || Date.now()).toLocaleTimeString();
373
-
374
- header.appendChild(nameSpan);
375
- header.appendChild(tsSpan);
376
- row.appendChild(header);
377
-
378
- if (payload !== undefined && payload !== null) {
379
- var pre = document.createElement('pre');
380
- pre.className = 'text-[11px] opacity-60 whitespace-pre-wrap break-all mt-0.5 max-h-32 overflow-auto';
381
- try {
382
- pre.textContent = typeof payload === 'object' ? JSON.stringify(payload, null, 2) : String(payload);
383
- } catch(e) {
384
- pre.textContent = String(payload);
385
- }
386
- row.appendChild(pre);
387
- }
388
-
389
- log.appendChild(row);
390
- log.scrollTop = log.scrollHeight;
391
- }
392
-
393
- function appendRtSystemMsg(msg, type) {
394
- var log = document.getElementById('rt-log');
395
- if (!log) return;
396
- var el = document.createElement('div');
397
- el.className = 'text-[10px] font-mono italic opacity-40 px-1 py-0.5 ' + (type === 'error' ? 'text-error opacity-70' : '');
398
- el.textContent = '— ' + msg + ' —';
399
- log.appendChild(el);
400
- log.scrollTop = log.scrollHeight;
401
- }
402
-
403
- function disconnectRealtime(silent) {
404
- if (_rtPingInterval) { clearInterval(_rtPingInterval); _rtPingInterval = null; }
405
- if (_rtSocket) {
406
- try { _rtSocket.close(); } catch(e) {}
407
- _rtSocket = null;
408
- }
409
- // Unsubscribe server-side if we have a token
410
- if (_rtToken) {
411
- var authToken = (document.getElementById('bearer-token-input') || {}).value || '';
412
- fetch('/realtime/unsubscribe', {
413
- method: 'POST',
414
- headers: { 'Content-Type': 'application/json' },
415
- body: JSON.stringify({ token: _rtToken, authToken: authToken }),
416
- }).catch(function() {});
417
- _rtToken = null;
418
- }
419
- setRtStatus('disconnected');
420
- if (!silent) appendRtSystemMsg('Disconnected');
421
- }
422
-
423
- async function connectRealtime() {
424
- if (!_isRealtimeSupported) {
425
- appendRtSystemMsg('Realtime is only available for queries', 'error');
426
- return;
427
- }
428
-
429
- disconnectRealtime(true);
430
- setRtStatus('connecting');
431
- appendRtSystemMsg('Subscribing…');
432
-
433
- var args = collectArgs();
434
- var authToken = (document.getElementById('bearer-token-input') || {}).value || '';
435
-
436
- var queryName = _realtimeQueryName;
437
-
438
- try {
439
- var subRes = await fetch('/realtime/subscribe', {
440
- method: 'POST',
441
- headers: { 'Content-Type': 'application/json' },
442
- body: JSON.stringify({ queryName: queryName, args: args, authToken: authToken }),
443
- });
444
-
445
- if (!subRes.ok) {
446
- var errBody = await subRes.json().catch(function() { return {}; });
447
- throw new Error(errBody.message || ('Subscribe failed: ' + subRes.status));
448
- }
449
-
450
- var subData = await subRes.json();
451
- _rtToken = subData.token;
452
-
453
- // Build WebSocket URL
454
- var wsInfo = subData.websocket;
455
- var wsUrl = wsInfo.url;
456
- // Append token params
457
- var sep = wsUrl.indexOf('?') >= 0 ? '&' : '?';
458
- wsUrl += sep + 'token=' + encodeURIComponent(_rtToken);
459
- if (authToken) wsUrl += '&authToken=' + encodeURIComponent(authToken);
460
-
461
- appendRtSystemMsg('Connecting WebSocket…');
462
-
463
- var ws = new WebSocket(wsUrl, wsInfo.protocol || undefined);
464
- _rtSocket = ws;
465
-
466
- ws.addEventListener('open', function() {
467
- setRtStatus('connected');
468
- appendRtSystemMsg('Connected');
469
- // Start ping keepalive
470
- _rtPingInterval = setInterval(function() {
471
- if (ws.readyState === 1) ws.send('ping');
472
- }, 25000);
473
- });
474
-
475
- ws.addEventListener('message', function(evt) {
476
- var data;
477
- try { data = JSON.parse(evt.data); } catch(e) { data = { event: 'raw', payload: evt.data }; }
478
- // Skip pong internal events
479
- if (data.event === 'pong') return;
480
- appendRtEvent(data.event || 'message', data.payload, Date.now());
481
- });
482
-
483
- ws.addEventListener('close', function(evt) {
484
- if (_rtPingInterval) { clearInterval(_rtPingInterval); _rtPingInterval = null; }
485
- _rtSocket = null;
486
- _rtToken = null;
487
- setRtStatus('disconnected');
488
- appendRtSystemMsg('Connection closed' + (evt.reason ? ': ' + evt.reason : ''));
489
- // Uncheck toggle
490
- var toggle = document.getElementById('realtime-toggle');
491
- if (toggle) toggle.checked = false;
492
- _rtEnabled = false;
493
- });
494
-
495
- ws.addEventListener('error', function() {
496
- setRtStatus('error');
497
- appendRtSystemMsg('WebSocket error', 'error');
498
- });
499
-
500
- } catch(err) {
501
- disconnectRealtime(true);
502
- setRtStatus('error');
503
- appendRtSystemMsg('Error: ' + (err.message || String(err)), 'error');
504
- // Uncheck toggle
505
- var toggle = document.getElementById('realtime-toggle');
506
- if (toggle) toggle.checked = false;
507
- _rtEnabled = false;
508
- }
509
- }
510
-
511
- window.toggleRealtime = function(enabled) {
512
- _rtEnabled = enabled;
513
- if (enabled) {
514
- // Switch to realtime tab to show the panel
515
- switchResultTab('realtime');
516
- // Don't connect yet — connection happens after executeFn() is called
517
- var log = document.getElementById('rt-log');
518
- var empty = document.getElementById('rt-empty-state');
519
- if (empty) {
520
- empty.innerHTML = '<iconify-icon icon="solar:pulse-2-bold-duotone" width="40" height="40"></iconify-icon><p class="text-xs font-semibold uppercase tracking-widest">Press Run to connect and start receiving updates</p>';
521
- empty.classList.remove('hidden');
522
- }
523
- } else {
524
- disconnectRealtime(false);
525
- }
526
- };
527
-
528
- window.clearRealtimeLog = function() {
529
- var log = document.getElementById('rt-log');
530
- if (!log) return;
531
- log.innerHTML = '';
532
- _rtEventCount = 0;
533
- var countEl = document.getElementById('rt-event-count');
534
- if (countEl) countEl.classList.add('hidden');
535
- // Re-show empty state if disconnected
536
- if (!_rtSocket) {
537
- var empty = document.createElement('div');
538
- empty.id = 'rt-empty-state';
539
- empty.className = 'flex flex-col items-center justify-center h-full opacity-20 text-center pointer-events-none gap-2';
540
- empty.innerHTML = '<iconify-icon icon="solar:pulse-2-bold-duotone" width="40" height="40"></iconify-icon><p class="text-xs font-semibold uppercase tracking-widest">Enable Realtime in Request Panel</p>';
541
- log.appendChild(empty);
542
- }
543
- };
544
-
545
- // ── Cleanup on HTMX navigation ─────────────────────────────────────────────
546
- document.addEventListener('htmx:beforeSwap', function() {
547
- disconnectRealtime(true);
548
- var toggle = document.getElementById('realtime-toggle');
549
- if (toggle) toggle.checked = false;
550
- _rtEnabled = false;
551
- });
552
- </script>
553
- `;
554
- }
1
+ import { DiscoveredHandlerOperation } from "../../../../../utils/handler-discovery";
2
+
3
+ export function renderScripts(h: DiscoveredHandlerOperation): string {
4
+ const isQuery = h.kind === "query";
5
+ const routePath = h.routePath;
6
+ // handlerName matches the key used in realtimeQueryHandlers registry
7
+ const realtimeQueryName = h.handlerName ?? h.routePath;
8
+ // Only queries can be subscribed to via realtime
9
+ const isRealtimeSupported = isQuery;
10
+
11
+ return `
12
+ <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/atom-one-dark.min.css">
13
+ <script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script>
14
+
15
+ <script>
16
+ // ── Tab switching ──────────────────────────────────────────────────────────
17
+ window._currentResultTab = 'body';
18
+
19
+ window.switchResultTab = function(tab) {
20
+ window._currentResultTab = tab;
21
+ ['body', 'headers', 'info', 'realtime'].forEach(function(t) {
22
+ var pane = document.getElementById('result-tab-' + t);
23
+ var btn = document.getElementById('tab-' + (t === 'headers' ? 'resp-headers' : t));
24
+ if (pane) pane.classList.toggle('hidden', t !== tab);
25
+ if (btn) btn.classList.toggle('tab-active', t === tab);
26
+ });
27
+ };
28
+
29
+ window.switchRequestTab = function(tab) {
30
+ ['args', 'auth', 'headers'].forEach(function(t) {
31
+ var pane = document.getElementById('request-tab-' + t);
32
+ var btn = document.getElementById('req-tab-btn-' + t);
33
+ if (pane) pane.classList.toggle('hidden', t !== tab);
34
+ if (btn) btn.classList.toggle('tab-active', t === tab);
35
+ });
36
+ };
37
+
38
+ // ── Header key-value rows ──────────────────────────────────────────────────
39
+ var _headerRowId = 0;
40
+
41
+ window.addHeaderRow = function(key, value) {
42
+ key = key || '';
43
+ value = value || '';
44
+ var id = ++_headerRowId;
45
+ var container = document.getElementById('headers-rows');
46
+
47
+ var row = document.createElement('div');
48
+ row.id = 'header-row-' + id;
49
+ row.className = 'flex items-center gap-2';
50
+
51
+ var keyInput = document.createElement('input');
52
+ keyInput.type = 'text';
53
+ keyInput.placeholder = 'Header-Name';
54
+ keyInput.setAttribute('data-hdr-key', '');
55
+ keyInput.value = key;
56
+ keyInput.className = 'input input-xs input-bordered font-mono w-2/5 bg-base-200/30 focus:bg-base-100 focus:border-primary transition-all rounded-lg border-base-200';
57
+
58
+ var valInput = document.createElement('input');
59
+ valInput.type = 'text';
60
+ valInput.placeholder = 'value';
61
+ valInput.setAttribute('data-hdr-val', '');
62
+ valInput.value = value;
63
+ valInput.className = 'input input-xs input-bordered font-mono flex-1 bg-base-200/30 focus:bg-base-100 focus:border-primary transition-all rounded-lg border-base-200';
64
+
65
+ var removeBtn = document.createElement('button');
66
+ removeBtn.type = 'button';
67
+ removeBtn.className = 'btn btn-xs btn-ghost text-error opacity-40 hover:opacity-100 px-1.5';
68
+ removeBtn.innerHTML = '<iconify-icon icon="solar:close-circle-linear" width="14" height="14"></iconify-icon>';
69
+ removeBtn.onclick = function() { row.remove(); };
70
+
71
+ row.appendChild(keyInput);
72
+ row.appendChild(valInput);
73
+ row.appendChild(removeBtn);
74
+ container.appendChild(row);
75
+ };
76
+
77
+ // ── Collect args from server-rendered inputs ───────────────────────────────
78
+ function collectArgs() {
79
+ var result = {};
80
+ document.querySelectorAll('#args-rows [data-arg-key]').forEach(function(el) {
81
+ var key = el.getAttribute('data-arg-key');
82
+ var type = el.getAttribute('data-arg-type');
83
+ if (!key) return;
84
+ var value;
85
+ if (type === 'boolean') {
86
+ value = el.checked;
87
+ } else if (type === 'number') {
88
+ var raw = el.value.trim();
89
+ value = raw === '' ? undefined : Number(raw);
90
+ } else {
91
+ value = el.value;
92
+ }
93
+ if (value !== undefined && value !== '') {
94
+ result[key] = value;
95
+ }
96
+ });
97
+ return result;
98
+ }
99
+
100
+ function collectHeaders() {
101
+ var result = {};
102
+ document.querySelectorAll('#headers-rows [data-hdr-key]').forEach(function(keyEl) {
103
+ var k = keyEl.value.trim();
104
+ var v = keyEl.parentElement.querySelector('[data-hdr-val]').value;
105
+ if (k) result[k] = v;
106
+ });
107
+ return result;
108
+ }
109
+
110
+ // ── Token visibility ──────────────────────────────────────────────────────
111
+ window.toggleTokenVisibility = function() {
112
+ var inp = document.getElementById('bearer-token-input');
113
+ var icon = document.getElementById('token-eye-icon');
114
+ if (!inp) return;
115
+ if (inp.type === 'password') {
116
+ inp.type = 'text';
117
+ if (icon) icon.setAttribute('icon', 'solar:eye-closed-linear');
118
+ } else {
119
+ inp.type = 'password';
120
+ if (icon) icon.setAttribute('icon', 'solar:eye-linear');
121
+ }
122
+ };
123
+
124
+ // ── Render helpers ─────────────────────────────────────────────────────────
125
+ function esc(str) {
126
+ return String(str).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
127
+ }
128
+
129
+ function showBodyLoading() {
130
+ var empty = document.getElementById('result-empty-state');
131
+ if (empty) empty.classList.add('hidden');
132
+ var content = document.getElementById('result-body-content');
133
+ if (content) {
134
+ content.classList.remove('hidden');
135
+ content.innerHTML = '<div class="flex items-center justify-center h-full min-h-[200px]"><span class="loading loading-ring loading-md text-primary"></span></div>';
136
+ }
137
+ }
138
+
139
+ function showBodyError(msg) {
140
+ var empty = document.getElementById('result-empty-state');
141
+ if (empty) empty.classList.add('hidden');
142
+ var content = document.getElementById('result-body-content');
143
+ if (content) {
144
+ content.classList.remove('hidden');
145
+ content.innerHTML = '<div class="p-5 text-error"><p class="font-bold mb-1 text-xs uppercase tracking-wider">Error</p><pre class="text-sm whitespace-pre-wrap">' + esc(msg) + '</pre></div>';
146
+ }
147
+ }
148
+
149
+ function renderBodyJson(jsonStr) {
150
+ var empty = document.getElementById('result-empty-state');
151
+ if (empty) empty.classList.add('hidden');
152
+ var content = document.getElementById('result-body-content');
153
+ if (!content) return;
154
+ content.classList.remove('hidden');
155
+ content.innerHTML = '<pre style="margin:0;padding:1rem"><code class="language-json">' + esc(jsonStr) + '</code></pre>';
156
+ if (window.hljs) hljs.highlightAll();
157
+ }
158
+
159
+ function renderBodyText(text) {
160
+ var empty = document.getElementById('result-empty-state');
161
+ if (empty) empty.classList.add('hidden');
162
+ var content = document.getElementById('result-body-content');
163
+ if (!content) return;
164
+ content.classList.remove('hidden');
165
+ content.innerHTML = '<pre style="margin:0;padding:1rem;white-space:pre-wrap;word-break:break-all">' + esc(text) + '</pre>';
166
+ }
167
+
168
+ function renderStatusBadge(status, elapsed) {
169
+ var badge = document.getElementById('res-status-badge');
170
+ var elBadge = document.getElementById('res-elapsed-badge');
171
+ if (badge) {
172
+ badge.className = 'badge badge-sm font-mono font-bold ' + (status < 300 ? 'badge-success' : status < 400 ? 'badge-warning' : 'badge-error');
173
+ badge.textContent = status;
174
+ badge.classList.remove('hidden');
175
+ }
176
+ if (elBadge) {
177
+ elBadge.textContent = elapsed + 'ms';
178
+ elBadge.classList.remove('hidden');
179
+ }
180
+ }
181
+
182
+ function renderResponseHeaders(headers) {
183
+ var el = document.getElementById('result-headers-content');
184
+ if (!el) return;
185
+ var entries = Object.entries(headers);
186
+ if (!entries.length) {
187
+ el.innerHTML = '<p class="text-xs opacity-40 italic p-2">No headers returned.</p>';
188
+ return;
189
+ }
190
+ var rows = entries.map(function(pair) {
191
+ return '<tr><td class="font-mono text-xs font-medium opacity-70">' + esc(pair[0]) + '</td><td class="font-mono text-xs opacity-50 break-all">' + esc(pair[1]) + '</td></tr>';
192
+ }).join('');
193
+ el.innerHTML = '<table class="table table-xs w-full"><thead><tr>' +
194
+ '<th class="text-[10px] uppercase tracking-wider opacity-40 font-bold">Header</th>' +
195
+ '<th class="text-[10px] uppercase tracking-wider opacity-40 font-bold">Value</th>' +
196
+ '</tr></thead><tbody>' + rows + '</tbody></table>';
197
+ }
198
+
199
+ function renderInfo(data) {
200
+ var el = document.getElementById('result-info-content');
201
+ if (!el) return;
202
+ var items = [
203
+ { label: 'Status', value: data.status + ' ' + data.statusText, color: data.status < 300 ? 'text-success' : data.status < 400 ? 'text-warning' : 'text-error' },
204
+ { label: 'Elapsed', value: data.elapsed + ' ms', color: '' },
205
+ { label: 'Method', value: data.method, color: '' },
206
+ { label: 'URL', value: data.url, color: '' },
207
+ { label: 'Content-Type', value: data.contentType || '—', color: '' },
208
+ ];
209
+ var html = '<div class="flex flex-col gap-3 p-4">';
210
+ items.forEach(function(item) {
211
+ html += '<div class="flex flex-col gap-0.5">';
212
+ html += '<span class="text-[10px] font-bold uppercase tracking-wider opacity-40">' + esc(item.label) + '</span>';
213
+ html += '<span class="font-mono text-sm break-all ' + (item.color || 'opacity-70') + '">' + esc(String(item.value)) + '</span>';
214
+ html += '</div>';
215
+ });
216
+ html += '</div>';
217
+ el.innerHTML = html;
218
+ }
219
+
220
+ // ── Main execution ─────────────────────────────────────────────────────────
221
+ window.executeFn = async function() {
222
+ // If realtime is enabled, connect/reconnect after execute completes
223
+ var shouldConnectRealtime = _rtEnabled && _isRealtimeSupported;
224
+ var indicator = document.getElementById('execution-indicator');
225
+ var headerErr = document.getElementById('headers-error');
226
+
227
+ var args = collectArgs();
228
+ var customHeaders = collectHeaders();
229
+ var token = (document.getElementById('bearer-token-input') || {}).value || '';
230
+
231
+ if (headerErr) headerErr.classList.add('hidden');
232
+
233
+ // Show loading
234
+ if (indicator) indicator.classList.remove('opacity-0');
235
+ showBodyLoading();
236
+
237
+ var isQuery = ${isQuery};
238
+ var method = isQuery ? 'GET' : 'POST';
239
+ var pathWithQuery = '${routePath}';
240
+
241
+ if (isQuery && Object.keys(args).length > 0) {
242
+ var params = new URLSearchParams();
243
+ Object.entries(args).forEach(function(entry) {
244
+ var key = entry[0];
245
+ var value = entry[1];
246
+ if (value !== undefined && value !== null) {
247
+ if (Array.isArray(value)) {
248
+ value.forEach(function(v) { params.append(key, typeof v === 'object' ? JSON.stringify(v) : String(v)); });
249
+ } else if (typeof value === 'object') {
250
+ params.append(key, JSON.stringify(value));
251
+ } else {
252
+ params.append(key, String(value));
253
+ }
254
+ }
255
+ });
256
+ var qs = params.toString();
257
+ if (qs) pathWithQuery += '?' + qs;
258
+ }
259
+
260
+ var fetchHeaders = Object.assign({ 'Content-Type': 'application/json' }, customHeaders);
261
+ if (token) fetchHeaders['Authorization'] = 'Bearer ' + token;
262
+
263
+ var startTime = performance.now();
264
+
265
+ try {
266
+ var res = await fetch(pathWithQuery, {
267
+ method: method,
268
+ headers: fetchHeaders,
269
+ body: method === 'POST' ? JSON.stringify(args) : undefined,
270
+ });
271
+
272
+ var elapsed = Math.round(performance.now() - startTime);
273
+ var contentType = res.headers.get('content-type') || '';
274
+ var rawBody = await res.text();
275
+
276
+ var parsedBody = rawBody;
277
+ if (rawBody && contentType.includes('application/json')) {
278
+ try { parsedBody = JSON.parse(rawBody); } catch(e) { parsedBody = rawBody; }
279
+ }
280
+
281
+ renderStatusBadge(res.status, elapsed);
282
+
283
+ if (typeof parsedBody === 'object') {
284
+ renderBodyJson(JSON.stringify(parsedBody, null, 2));
285
+ } else {
286
+ renderBodyText(rawBody);
287
+ }
288
+
289
+ renderResponseHeaders(Object.fromEntries(res.headers));
290
+ renderInfo({ status: res.status, statusText: res.statusText, elapsed: elapsed, method: method, url: pathWithQuery, contentType: contentType });
291
+
292
+ } catch(err) {
293
+ showBodyError('Fetch error: ' + (err.message || String(err)));
294
+ } finally {
295
+ if (indicator) indicator.classList.add('opacity-0');
296
+ // Connect realtime after execute if toggle is on
297
+ if (shouldConnectRealtime) {
298
+ switchResultTab('realtime');
299
+ connectRealtime();
300
+ }
301
+ }
302
+ };
303
+
304
+ // ── Realtime WebSocket ─────────────────────────────────────────────────────
305
+ var _rtSocket = null;
306
+ var _rtToken = null;
307
+ var _rtEnabled = false;
308
+ var _rtEventCount = 0;
309
+ var _isRealtimeSupported = ${isRealtimeSupported};
310
+ var _realtimeQueryName = '${realtimeQueryName}';
311
+ var _rtPingInterval = null;
312
+
313
+ function setRtStatus(status) {
314
+ // status: 'disconnected' | 'connecting' | 'connected' | 'error'
315
+ var dot = document.getElementById('rt-status-dot');
316
+ var headerDot = document.getElementById('rt-dot');
317
+ var text = document.getElementById('rt-status-text');
318
+ var colorMap = {
319
+ disconnected: 'bg-base-content/20',
320
+ connecting: 'bg-warning animate-pulse',
321
+ connected: 'bg-success animate-pulse',
322
+ error: 'bg-error',
323
+ };
324
+ var headerColorMap = {
325
+ disconnected: 'bg-base-content/20',
326
+ connecting: 'bg-warning animate-pulse',
327
+ connected: 'bg-success animate-pulse',
328
+ error: 'bg-error',
329
+ };
330
+ var labelMap = {
331
+ disconnected: 'Disconnected',
332
+ connecting: 'Connecting…',
333
+ connected: 'Connected',
334
+ error: 'Error',
335
+ };
336
+ if (dot) {
337
+ dot.className = 'w-2 h-2 rounded-full transition-all duration-300 ' + (colorMap[status] || colorMap.disconnected);
338
+ }
339
+ if (headerDot) {
340
+ headerDot.className = 'w-1.5 h-1.5 rounded-full transition-colors duration-300 ' + (headerColorMap[status] || headerColorMap.disconnected);
341
+ }
342
+ if (text) {
343
+ text.textContent = labelMap[status] || status;
344
+ }
345
+ }
346
+
347
+ function appendRtEvent(eventName, payload, ts) {
348
+ var log = document.getElementById('rt-log');
349
+ var empty = document.getElementById('rt-empty-state');
350
+ if (!log) return;
351
+ if (empty) empty.classList.add('hidden');
352
+
353
+ _rtEventCount++;
354
+ var countEl = document.getElementById('rt-event-count');
355
+ if (countEl) {
356
+ countEl.textContent = _rtEventCount + ' event' + (_rtEventCount !== 1 ? 's' : '');
357
+ countEl.classList.remove('hidden');
358
+ }
359
+
360
+ var row = document.createElement('div');
361
+ row.className = 'flex flex-col gap-0.5 p-2 rounded-lg bg-base-200/40 border border-base-200/60 hover:bg-base-200/70 transition-colors';
362
+
363
+ var header = document.createElement('div');
364
+ header.className = 'flex items-center justify-between';
365
+
366
+ var nameSpan = document.createElement('span');
367
+ nameSpan.className = 'font-semibold text-primary text-[11px] uppercase tracking-wider';
368
+ nameSpan.textContent = eventName;
369
+
370
+ var tsSpan = document.createElement('span');
371
+ tsSpan.className = 'text-[10px] opacity-30 font-mono';
372
+ tsSpan.textContent = new Date(ts || Date.now()).toLocaleTimeString();
373
+
374
+ header.appendChild(nameSpan);
375
+ header.appendChild(tsSpan);
376
+ row.appendChild(header);
377
+
378
+ if (payload !== undefined && payload !== null) {
379
+ var pre = document.createElement('pre');
380
+ pre.className = 'text-[11px] opacity-60 whitespace-pre-wrap break-all mt-0.5 max-h-32 overflow-auto';
381
+ try {
382
+ pre.textContent = typeof payload === 'object' ? JSON.stringify(payload, null, 2) : String(payload);
383
+ } catch(e) {
384
+ pre.textContent = String(payload);
385
+ }
386
+ row.appendChild(pre);
387
+ }
388
+
389
+ log.appendChild(row);
390
+ log.scrollTop = log.scrollHeight;
391
+ }
392
+
393
+ function appendRtSystemMsg(msg, type) {
394
+ var log = document.getElementById('rt-log');
395
+ if (!log) return;
396
+ var el = document.createElement('div');
397
+ el.className = 'text-[10px] font-mono italic opacity-40 px-1 py-0.5 ' + (type === 'error' ? 'text-error opacity-70' : '');
398
+ el.textContent = '— ' + msg + ' —';
399
+ log.appendChild(el);
400
+ log.scrollTop = log.scrollHeight;
401
+ }
402
+
403
+ function disconnectRealtime(silent) {
404
+ if (_rtPingInterval) { clearInterval(_rtPingInterval); _rtPingInterval = null; }
405
+ if (_rtSocket) {
406
+ try { _rtSocket.close(); } catch(e) {}
407
+ _rtSocket = null;
408
+ }
409
+ // Unsubscribe server-side if we have a token
410
+ if (_rtToken) {
411
+ var authToken = (document.getElementById('bearer-token-input') || {}).value || '';
412
+ fetch('/realtime/unsubscribe', {
413
+ method: 'POST',
414
+ headers: { 'Content-Type': 'application/json' },
415
+ body: JSON.stringify({ token: _rtToken, authToken: authToken }),
416
+ }).catch(function() {});
417
+ _rtToken = null;
418
+ }
419
+ setRtStatus('disconnected');
420
+ if (!silent) appendRtSystemMsg('Disconnected');
421
+ }
422
+
423
+ async function connectRealtime() {
424
+ if (!_isRealtimeSupported) {
425
+ appendRtSystemMsg('Realtime is only available for queries', 'error');
426
+ return;
427
+ }
428
+
429
+ disconnectRealtime(true);
430
+ setRtStatus('connecting');
431
+ appendRtSystemMsg('Subscribing…');
432
+
433
+ var args = collectArgs();
434
+ var authToken = (document.getElementById('bearer-token-input') || {}).value || '';
435
+
436
+ var queryName = _realtimeQueryName;
437
+
438
+ try {
439
+ var subRes = await fetch('/realtime/subscribe', {
440
+ method: 'POST',
441
+ headers: { 'Content-Type': 'application/json' },
442
+ body: JSON.stringify({ queryName: queryName, args: args, authToken: authToken }),
443
+ });
444
+
445
+ if (!subRes.ok) {
446
+ var errBody = await subRes.json().catch(function() { return {}; });
447
+ throw new Error(errBody.message || ('Subscribe failed: ' + subRes.status));
448
+ }
449
+
450
+ var subData = await subRes.json();
451
+ _rtToken = subData.token;
452
+
453
+ // Build WebSocket URL
454
+ var wsInfo = subData.websocket;
455
+ var wsUrl = wsInfo.url;
456
+ // Append token params
457
+ var sep = wsUrl.indexOf('?') >= 0 ? '&' : '?';
458
+ wsUrl += sep + 'token=' + encodeURIComponent(_rtToken);
459
+ if (authToken) wsUrl += '&authToken=' + encodeURIComponent(authToken);
460
+
461
+ appendRtSystemMsg('Connecting WebSocket…');
462
+
463
+ var ws = new WebSocket(wsUrl, wsInfo.protocol || undefined);
464
+ _rtSocket = ws;
465
+
466
+ ws.addEventListener('open', function() {
467
+ setRtStatus('connected');
468
+ appendRtSystemMsg('Connected');
469
+ // Start ping keepalive
470
+ _rtPingInterval = setInterval(function() {
471
+ if (ws.readyState === 1) ws.send('ping');
472
+ }, 25000);
473
+ });
474
+
475
+ ws.addEventListener('message', function(evt) {
476
+ var data;
477
+ try { data = JSON.parse(evt.data); } catch(e) { data = { event: 'raw', payload: evt.data }; }
478
+ // Skip pong internal events
479
+ if (data.event === 'pong') return;
480
+ appendRtEvent(data.event || 'message', data.payload, Date.now());
481
+ });
482
+
483
+ ws.addEventListener('close', function(evt) {
484
+ if (_rtPingInterval) { clearInterval(_rtPingInterval); _rtPingInterval = null; }
485
+ _rtSocket = null;
486
+ _rtToken = null;
487
+ setRtStatus('disconnected');
488
+ appendRtSystemMsg('Connection closed' + (evt.reason ? ': ' + evt.reason : ''));
489
+ // Uncheck toggle
490
+ var toggle = document.getElementById('realtime-toggle');
491
+ if (toggle) toggle.checked = false;
492
+ _rtEnabled = false;
493
+ });
494
+
495
+ ws.addEventListener('error', function() {
496
+ setRtStatus('error');
497
+ appendRtSystemMsg('WebSocket error', 'error');
498
+ });
499
+
500
+ } catch(err) {
501
+ disconnectRealtime(true);
502
+ setRtStatus('error');
503
+ appendRtSystemMsg('Error: ' + (err.message || String(err)), 'error');
504
+ // Uncheck toggle
505
+ var toggle = document.getElementById('realtime-toggle');
506
+ if (toggle) toggle.checked = false;
507
+ _rtEnabled = false;
508
+ }
509
+ }
510
+
511
+ window.toggleRealtime = function(enabled) {
512
+ _rtEnabled = enabled;
513
+ if (enabled) {
514
+ // Switch to realtime tab to show the panel
515
+ switchResultTab('realtime');
516
+ // Don't connect yet — connection happens after executeFn() is called
517
+ var log = document.getElementById('rt-log');
518
+ var empty = document.getElementById('rt-empty-state');
519
+ if (empty) {
520
+ empty.innerHTML = '<iconify-icon icon="solar:pulse-2-bold-duotone" width="40" height="40"></iconify-icon><p class="text-xs font-semibold uppercase tracking-widest">Press Run to connect and start receiving updates</p>';
521
+ empty.classList.remove('hidden');
522
+ }
523
+ } else {
524
+ disconnectRealtime(false);
525
+ }
526
+ };
527
+
528
+ window.clearRealtimeLog = function() {
529
+ var log = document.getElementById('rt-log');
530
+ if (!log) return;
531
+ log.innerHTML = '';
532
+ _rtEventCount = 0;
533
+ var countEl = document.getElementById('rt-event-count');
534
+ if (countEl) countEl.classList.add('hidden');
535
+ // Re-show empty state if disconnected
536
+ if (!_rtSocket) {
537
+ var empty = document.createElement('div');
538
+ empty.id = 'rt-empty-state';
539
+ empty.className = 'flex flex-col items-center justify-center h-full opacity-20 text-center pointer-events-none gap-2';
540
+ empty.innerHTML = '<iconify-icon icon="solar:pulse-2-bold-duotone" width="40" height="40"></iconify-icon><p class="text-xs font-semibold uppercase tracking-widest">Enable Realtime in Request Panel</p>';
541
+ log.appendChild(empty);
542
+ }
543
+ };
544
+
545
+ // ── Cleanup on HTMX navigation ─────────────────────────────────────────────
546
+ document.addEventListener('htmx:beforeSwap', function() {
547
+ disconnectRealtime(true);
548
+ var toggle = document.getElementById('realtime-toggle');
549
+ if (toggle) toggle.checked = false;
550
+ _rtEnabled = false;
551
+ });
552
+ </script>
553
+ `;
554
+ }