@meshxdata/fops 0.1.52 → 0.1.54

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 (86) hide show
  1. package/CHANGELOG.md +559 -0
  2. package/package.json +2 -6
  3. package/src/agent/agent.js +6 -0
  4. package/src/commands/setup.js +34 -0
  5. package/src/fleet-registry.js +38 -2
  6. package/src/plugins/__test-fixtures__/fake-plugin.js +2 -0
  7. package/src/plugins/__test-fixtures__/no-register-plugin.js +2 -0
  8. package/src/plugins/__test-fixtures__/with-register/index.js +2 -0
  9. package/src/plugins/__test-fixtures__/without-register/index.js +2 -0
  10. package/src/plugins/api.js +4 -0
  11. package/src/plugins/builtins/docker-compose.js +65 -0
  12. package/src/plugins/bundled/fops-plugin-azure/index.js +4 -0
  13. package/src/plugins/bundled/fops-plugin-azure/lib/azure-aks-core.js +44 -53
  14. package/src/plugins/bundled/fops-plugin-azure/lib/azure-aks-storage.js +2 -2
  15. package/src/plugins/bundled/fops-plugin-azure/lib/azure-cost.js +52 -22
  16. package/src/plugins/bundled/fops-plugin-azure/lib/azure-helpers.js +6 -2
  17. package/src/plugins/bundled/fops-plugin-azure/lib/azure-ops.js +113 -7
  18. package/src/plugins/bundled/fops-plugin-azure/lib/azure-provision-init.js +13 -4
  19. package/src/plugins/bundled/fops-plugin-azure/lib/azure-provision.js +91 -14
  20. package/src/plugins/bundled/fops-plugin-azure/lib/azure-service.js +507 -0
  21. package/src/plugins/bundled/fops-plugin-azure/lib/azure-sync.js +146 -7
  22. package/src/plugins/bundled/fops-plugin-azure/lib/azure.js +1 -1
  23. package/src/plugins/bundled/fops-plugin-azure/lib/commands/vm-cmds.js +61 -0
  24. package/src/plugins/bundled/fops-plugin-cloud/api.js +712 -0
  25. package/src/plugins/bundled/fops-plugin-cloud/fops.plugin.json +6 -0
  26. package/src/plugins/bundled/fops-plugin-cloud/index.js +208 -0
  27. package/src/plugins/bundled/fops-plugin-cloud/lib/azure-provider.js +81 -0
  28. package/src/plugins/bundled/fops-plugin-cloud/lib/provider.js +50 -0
  29. package/src/plugins/bundled/fops-plugin-cloud/ui/dist/assets/favicon-C49brna2.svg +15 -0
  30. package/src/plugins/bundled/fops-plugin-cloud/ui/dist/assets/index-CVqQ_kKW.js +65 -0
  31. package/src/plugins/bundled/fops-plugin-cloud/ui/dist/assets/index-DZetahP3.css +1 -0
  32. package/src/plugins/bundled/fops-plugin-cloud/ui/dist/index.html +28 -0
  33. package/src/plugins/bundled/fops-plugin-cloud/ui/index.html +27 -0
  34. package/src/plugins/bundled/fops-plugin-cloud/ui/package-lock.json +2634 -0
  35. package/src/plugins/bundled/fops-plugin-cloud/ui/package.json +29 -0
  36. package/src/plugins/bundled/fops-plugin-cloud/ui/postcss.config.cjs +5 -0
  37. package/src/plugins/bundled/fops-plugin-cloud/ui/src/App.jsx +32 -0
  38. package/src/plugins/bundled/fops-plugin-cloud/ui/src/api/client.js +114 -0
  39. package/src/plugins/bundled/fops-plugin-cloud/ui/src/api/queries.js +111 -0
  40. package/src/plugins/bundled/fops-plugin-cloud/ui/src/components/LogPanel.jsx +162 -0
  41. package/src/plugins/bundled/fops-plugin-cloud/ui/src/components/ThemeToggle.jsx +46 -0
  42. package/src/plugins/bundled/fops-plugin-cloud/ui/src/css/additional-styles/utility-patterns.css +147 -0
  43. package/src/plugins/bundled/fops-plugin-cloud/ui/src/css/style.css +138 -0
  44. package/src/plugins/bundled/fops-plugin-cloud/ui/src/favicon.svg +15 -0
  45. package/src/plugins/bundled/fops-plugin-cloud/ui/src/lib/utils.ts +19 -0
  46. package/src/plugins/bundled/fops-plugin-cloud/ui/src/main.jsx +25 -0
  47. package/src/plugins/bundled/fops-plugin-cloud/ui/src/pages/Audit.jsx +164 -0
  48. package/src/plugins/bundled/fops-plugin-cloud/ui/src/pages/Costs.jsx +305 -0
  49. package/src/plugins/bundled/fops-plugin-cloud/ui/src/pages/CreateResource.jsx +285 -0
  50. package/src/plugins/bundled/fops-plugin-cloud/ui/src/pages/Fleet.jsx +307 -0
  51. package/src/plugins/bundled/fops-plugin-cloud/ui/src/pages/Resources.jsx +229 -0
  52. package/src/plugins/bundled/fops-plugin-cloud/ui/src/partials/Header.jsx +132 -0
  53. package/src/plugins/bundled/fops-plugin-cloud/ui/src/partials/Sidebar.jsx +174 -0
  54. package/src/plugins/bundled/fops-plugin-cloud/ui/src/partials/SidebarLinkGroup.jsx +21 -0
  55. package/src/plugins/bundled/fops-plugin-cloud/ui/src/utils/AuthContext.jsx +170 -0
  56. package/src/plugins/bundled/fops-plugin-cloud/ui/src/utils/Info.jsx +49 -0
  57. package/src/plugins/bundled/fops-plugin-cloud/ui/src/utils/ThemeContext.jsx +37 -0
  58. package/src/plugins/bundled/fops-plugin-cloud/ui/src/utils/Transition.jsx +116 -0
  59. package/src/plugins/bundled/fops-plugin-cloud/ui/src/utils/Utils.js +63 -0
  60. package/src/plugins/bundled/fops-plugin-cloud/ui/vite.config.js +23 -0
  61. package/src/plugins/bundled/fops-plugin-foundation/test-helpers.js +65 -0
  62. package/src/plugins/loader.js +34 -1
  63. package/src/plugins/registry.js +15 -0
  64. package/src/plugins/schemas.js +17 -0
  65. package/src/project.js +1 -1
  66. package/src/serve.js +196 -2
  67. package/src/shell.js +21 -1
  68. package/src/web/admin.html.js +236 -0
  69. package/src/web/api.js +73 -0
  70. package/src/web/dist/assets/index-BphVaAUd.css +1 -0
  71. package/src/web/dist/assets/index-CSckLzuG.js +129 -0
  72. package/src/web/dist/index.html +2 -2
  73. package/src/web/frontend/index.html +16 -0
  74. package/src/web/frontend/src/App.jsx +445 -0
  75. package/src/web/frontend/src/components/ChatView.jsx +910 -0
  76. package/src/web/frontend/src/components/InputBox.jsx +523 -0
  77. package/src/web/frontend/src/components/Sidebar.jsx +410 -0
  78. package/src/web/frontend/src/components/StatusBar.jsx +37 -0
  79. package/src/web/frontend/src/components/TabBar.jsx +87 -0
  80. package/src/web/frontend/src/hooks/useWebSocket.js +412 -0
  81. package/src/web/frontend/src/index.css +78 -0
  82. package/src/web/frontend/src/main.jsx +6 -0
  83. package/src/web/frontend/vite.config.js +21 -0
  84. package/src/web/server.js +64 -1
  85. package/src/web/dist/assets/index-NXC8Hvnp.css +0 -1
  86. package/src/web/dist/assets/index-QH1N4ejK.js +0 -112
@@ -269,6 +269,7 @@ export function getAdminHtml() {
269
269
  <button data-tab="tools">Tools & Agents</button>
270
270
  <button data-tab="doctor">Doctor</button>
271
271
  <button data-tab="fleet">Fleet</button>
272
+ <button data-tab="costs">Costs</button>
272
273
  <button data-tab="audit">Audit</button>
273
274
  <button data-tab="meshes">Meshes & Landscape</button>
274
275
  <button data-tab="sessions">Sessions</button>
@@ -381,6 +382,46 @@ export function getAdminHtml() {
381
382
  </div>
382
383
  </div>
383
384
 
385
+ <!-- Cost Explorer -->
386
+ <div id="tab-costs" class="panel">
387
+ <div style="margin-bottom:12px;display:flex;gap:8px;align-items:center;flex-wrap:wrap">
388
+ <button class="btn btn-sm" onclick="loadCosts()">Refresh</button>
389
+ <select id="cost-days" onchange="loadCosts()" style="font-size:12px;padding:4px 8px;background:var(--bg-card);border:1px solid var(--border);border-radius:var(--radius-sm);color:var(--text)">
390
+ <option value="7">Last 7 days</option>
391
+ <option value="14">Last 14 days</option>
392
+ <option value="30" selected>Last 30 days</option>
393
+ <option value="90">Last 90 days</option>
394
+ </select>
395
+ <span id="cost-status" style="font-size:12px;color:var(--text-dim)"></span>
396
+ </div>
397
+ <div id="cost-summary" class="grid grid-4" style="margin-bottom:20px"></div>
398
+ <div style="display:grid;grid-template-columns:1fr 1fr;gap:20px">
399
+ <div>
400
+ <h3 style="font-size:12px;font-weight:600;color:var(--text-dim);text-transform:uppercase;letter-spacing:0.5px;margin-bottom:8px">Cost by Subscription</h3>
401
+ <div id="cost-subscriptions"></div>
402
+ </div>
403
+ <div>
404
+ <h3 style="font-size:12px;font-weight:600;color:var(--text-dim);text-transform:uppercase;letter-spacing:0.5px;margin-bottom:8px">Cost by Service</h3>
405
+ <div id="cost-services"></div>
406
+ </div>
407
+ </div>
408
+ <div style="margin-top:20px">
409
+ <h3 style="font-size:12px;font-weight:600;color:var(--text-dim);text-transform:uppercase;letter-spacing:0.5px;margin-bottom:8px">Budgets</h3>
410
+ <div id="cost-budgets"></div>
411
+ </div>
412
+ <div style="margin-top:20px">
413
+ <h3 style="font-size:12px;font-weight:600;color:var(--text-dim);text-transform:uppercase;letter-spacing:0.5px;margin-bottom:8px">Waste & Savings</h3>
414
+ <div style="display:grid;grid-template-columns:1fr 1fr;gap:20px">
415
+ <div id="cost-waste"></div>
416
+ <div id="cost-advisor"></div>
417
+ </div>
418
+ </div>
419
+ <div style="margin-top:20px">
420
+ <h3 style="font-size:12px;font-weight:600;color:var(--text-dim);text-transform:uppercase;letter-spacing:0.5px;margin-bottom:8px">SLA &amp; Availability</h3>
421
+ <div id="cost-sla"></div>
422
+ </div>
423
+ </div>
424
+
384
425
  <!-- Audit (from scrape: VM status, containers, services, sessions) -->
385
426
  <div id="tab-audit" class="panel">
386
427
  <div style="margin-bottom:12px;display:flex;gap:8px;align-items:center;flex-wrap:wrap">
@@ -2294,6 +2335,201 @@ function filterAuditContent() {
2294
2335
  el.innerHTML = temp.innerHTML;
2295
2336
  }
2296
2337
 
2338
+ // ── Cost Explorer ────────────────────────────────────────────
2339
+
2340
+ let _costCache = null;
2341
+ async function loadCosts() {
2342
+ const days = document.getElementById('cost-days').value;
2343
+ const status = document.getElementById('cost-status');
2344
+ status.textContent = 'Loading...';
2345
+
2346
+ try {
2347
+ // Fetch all cost data in parallel via tool execution
2348
+ const [allSubs, budgets, waste, advisor] = await Promise.all([
2349
+ api('/tools/azure_cost_all_subscriptions', { method: 'POST', headers: {'Content-Type':'application/json'}, body: JSON.stringify({ days: parseInt(days) }) }).catch(() => null),
2350
+ api('/tools/azure_budgets', { method: 'POST', headers: {'Content-Type':'application/json'}, body: JSON.stringify({}) }).catch(() => null),
2351
+ api('/tools/azure_vm_waste', { method: 'POST', headers: {'Content-Type':'application/json'}, body: JSON.stringify({}) }).catch(() => null),
2352
+ api('/tools/azure_advisor_recommendations', { method: 'POST', headers: {'Content-Type':'application/json'}, body: JSON.stringify({ category: 'Cost' }) }).catch(() => null),
2353
+ ]);
2354
+
2355
+ // SLA: fleet health + service uptime from each VM's Prometheus
2356
+ let sla = null;
2357
+ try {
2358
+ const fleet = await api('/fleet');
2359
+ if (fleet?.vms) {
2360
+ sla = { vms: [] };
2361
+ for (const vm of fleet.vms) {
2362
+ const entry = { vm: vm.vm, status: vm.status, services: vm.services || {} };
2363
+ // Query Prometheus avg_over_time(up) for each VM via grafana proxy
2364
+ try {
2365
+ const upRes = await api('/grafana/' + encodeURIComponent(vm.vm) + '/prometheus?query=' + encodeURIComponent('avg_over_time(up[' + days + 'd])'));
2366
+ if (upRes?.data?.result) {
2367
+ entry.uptimes = {};
2368
+ for (const r of upRes.data.result) {
2369
+ const job = r.metric?.job || '?';
2370
+ const pct = (parseFloat(r.value?.[1] || '0') * 100).toFixed(2);
2371
+ entry.uptimes[job] = parseFloat(pct);
2372
+ }
2373
+ }
2374
+ } catch {}
2375
+ sla.vms.push(entry);
2376
+ }
2377
+ }
2378
+ } catch {}
2379
+
2380
+ _costCache = { allSubs, budgets, waste, advisor, sla };
2381
+ renderCosts();
2382
+ status.textContent = 'Updated ' + new Date().toLocaleTimeString();
2383
+ } catch (err) {
2384
+ status.textContent = 'Error: ' + err.message;
2385
+ }
2386
+ }
2387
+
2388
+ function fmtCurrency(v) {
2389
+ if (v == null) return '-';
2390
+ const n = typeof v === 'string' ? parseFloat(v) : v;
2391
+ if (isNaN(n)) return '-';
2392
+ return '$' + n.toLocaleString(undefined, { minimumFractionDigits: 2, maximumFractionDigits: 2 });
2393
+ }
2394
+
2395
+ function renderCosts() {
2396
+ if (!_costCache) return;
2397
+ const { allSubs, budgets, waste, advisor } = _costCache;
2398
+
2399
+ // Parse tool results (tool execution returns { result: ... } or { tool, result, stdout })
2400
+ const costData = allSubs?.result || allSubs?.stdout ? JSON.parse(allSubs.stdout || '{}') : allSubs || {};
2401
+ const budgetData = budgets?.result || budgets?.stdout ? JSON.parse(budgets.stdout || '[]') : budgets || [];
2402
+ const wasteData = waste?.result || waste?.stdout ? JSON.parse(waste.stdout || '{}') : waste || {};
2403
+ const advisorData = advisor?.result || advisor?.stdout ? JSON.parse(advisor.stdout || '[]') : advisor || [];
2404
+
2405
+ // Summary cards
2406
+ const summary = document.getElementById('cost-summary');
2407
+ const totalCost = costData.totalCost || costData.total || 0;
2408
+ const subCount = costData.subscriptions?.length || 0;
2409
+ const wasteCount = (wasteData.stoppedVMs?.length || 0) + (wasteData.unattachedDisks?.length || 0) + (wasteData.unusedPublicIPs?.length || 0);
2410
+ const advisorCount = Array.isArray(advisorData) ? advisorData.length : 0;
2411
+ summary.innerHTML =
2412
+ '<div class="card"><h3>Total Spend</h3><div class="value">' + fmtCurrency(totalCost) + '</div></div>' +
2413
+ '<div class="card"><h3>Subscriptions</h3><div class="value">' + subCount + '</div></div>' +
2414
+ '<div class="card"><h3>Waste Items</h3><div class="value">' + wasteCount + '</div></div>' +
2415
+ '<div class="card"><h3>Recommendations</h3><div class="value">' + advisorCount + '</div></div>';
2416
+
2417
+ // Subscriptions table
2418
+ const subsEl = document.getElementById('cost-subscriptions');
2419
+ if (costData.subscriptions?.length) {
2420
+ let html = '<table class="tbl"><thead><tr><th>Subscription</th><th>Cost</th><th>Currency</th></tr></thead><tbody>';
2421
+ for (const s of costData.subscriptions) {
2422
+ html += '<tr><td>' + esc(s.name || s.subscriptionId || '?') + '</td><td>' + fmtCurrency(s.cost || s.totalCost) + '</td><td>' + (s.currency || 'USD') + '</td></tr>';
2423
+ }
2424
+ html += '</tbody></table>';
2425
+ subsEl.innerHTML = html;
2426
+ } else {
2427
+ subsEl.innerHTML = '<div class="empty">No subscription data</div>';
2428
+ }
2429
+
2430
+ // Service breakdown (from first subscription or top-level)
2431
+ const servicesEl = document.getElementById('cost-services');
2432
+ const serviceBreakdown = costData.services || costData.byService || (costData.subscriptions?.[0]?.services) || [];
2433
+ if (serviceBreakdown.length) {
2434
+ let html = '<table class="tbl"><thead><tr><th>Service</th><th>Cost</th></tr></thead><tbody>';
2435
+ const sorted = [...serviceBreakdown].sort((a, b) => (b.cost || 0) - (a.cost || 0));
2436
+ for (const s of sorted.slice(0, 15)) {
2437
+ html += '<tr><td>' + esc(s.name || s.service || '?') + '</td><td>' + fmtCurrency(s.cost) + '</td></tr>';
2438
+ }
2439
+ html += '</tbody></table>';
2440
+ servicesEl.innerHTML = html;
2441
+ } else {
2442
+ servicesEl.innerHTML = '<div class="empty">No service breakdown</div>';
2443
+ }
2444
+
2445
+ // Budgets
2446
+ const budgetsEl = document.getElementById('cost-budgets');
2447
+ const budgetList = Array.isArray(budgetData) ? budgetData : budgetData.budgets || [];
2448
+ if (budgetList.length) {
2449
+ let html = '<table class="tbl"><thead><tr><th>Budget</th><th>Limit</th><th>Spent</th><th>%</th></tr></thead><tbody>';
2450
+ for (const b of budgetList) {
2451
+ const pct = b.limit ? ((b.spent || b.currentSpend || 0) / b.limit * 100).toFixed(0) : '?';
2452
+ const color = pct > 90 ? 'var(--red)' : pct > 70 ? 'var(--yellow)' : 'var(--green)';
2453
+ html += '<tr><td>' + esc(b.name || '?') + '</td><td>' + fmtCurrency(b.limit || b.amount) + '</td><td>' + fmtCurrency(b.spent || b.currentSpend) + '</td><td style="color:' + color + '">' + pct + '%</td></tr>';
2454
+ }
2455
+ html += '</tbody></table>';
2456
+ budgetsEl.innerHTML = html;
2457
+ } else {
2458
+ budgetsEl.innerHTML = '<div class="empty">No budgets configured</div>';
2459
+ }
2460
+
2461
+ // Waste
2462
+ const wasteEl = document.getElementById('cost-waste');
2463
+ const wasteItems = [];
2464
+ for (const vm of (wasteData.stoppedVMs || [])) wasteItems.push({ type: 'Stopped VM', name: vm.name || vm, cost: vm.monthlyCost });
2465
+ for (const d of (wasteData.unattachedDisks || [])) wasteItems.push({ type: 'Unattached Disk', name: d.name || d, cost: d.monthlyCost });
2466
+ for (const ip of (wasteData.unusedPublicIPs || [])) wasteItems.push({ type: 'Unused IP', name: ip.name || ip, cost: ip.monthlyCost });
2467
+ if (wasteItems.length) {
2468
+ let html = '<table class="tbl"><thead><tr><th>Type</th><th>Resource</th><th>Est. Cost</th></tr></thead><tbody>';
2469
+ for (const w of wasteItems) {
2470
+ html += '<tr><td>' + esc(w.type) + '</td><td>' + esc(w.name) + '</td><td>' + (w.cost ? fmtCurrency(w.cost) + '/mo' : '-') + '</td></tr>';
2471
+ }
2472
+ html += '</tbody></table>';
2473
+ wasteEl.innerHTML = html;
2474
+ } else {
2475
+ wasteEl.innerHTML = '<div class="empty" style="color:var(--green)">No wasted resources found</div>';
2476
+ }
2477
+
2478
+ // Advisor recommendations
2479
+ const advisorEl = document.getElementById('cost-advisor');
2480
+ const recs = Array.isArray(advisorData) ? advisorData : advisorData.recommendations || [];
2481
+ if (recs.length) {
2482
+ let html = '<table class="tbl"><thead><tr><th>Recommendation</th><th>Impact</th><th>Savings</th></tr></thead><tbody>';
2483
+ for (const r of recs.slice(0, 10)) {
2484
+ html += '<tr><td>' + esc(r.shortDescription || r.recommendation || r.name || '?') + '</td><td>' + esc(r.impact || '-') + '</td><td>' + (r.savingsAmount ? fmtCurrency(r.savingsAmount) + '/yr' : '-') + '</td></tr>';
2485
+ }
2486
+ html += '</tbody></table>';
2487
+ advisorEl.innerHTML = html;
2488
+ } else {
2489
+ advisorEl.innerHTML = '<div class="empty">No cost recommendations</div>';
2490
+ }
2491
+
2492
+ // SLA & Availability
2493
+ const slaEl = document.getElementById('cost-sla');
2494
+ const sla = _costCache.sla;
2495
+ if (sla?.vms?.length) {
2496
+ const SLA_TARGET = 99.9; // SLA target percentage
2497
+ let html = '<table class="tbl"><thead><tr><th>Environment</th><th>Status</th><th>Service</th><th>Uptime</th><th>SLA</th></tr></thead><tbody>';
2498
+ for (const vm of sla.vms) {
2499
+ const uptimes = vm.uptimes || {};
2500
+ const services = Object.keys(uptimes).length ? Object.keys(uptimes) : Object.keys(vm.services || {});
2501
+ if (!services.length) {
2502
+ const statusColor = vm.status === 'healthy' ? 'var(--green)' : vm.status === 'degraded' ? 'var(--yellow)' : 'var(--red)';
2503
+ html += '<tr><td>' + esc(vm.vm) + '</td><td style="color:' + statusColor + '">' + esc(vm.status || '?') + '</td><td>-</td><td>-</td><td>-</td></tr>';
2504
+ continue;
2505
+ }
2506
+ let first = true;
2507
+ for (const svc of services) {
2508
+ const pct = uptimes[svc];
2509
+ const pctStr = pct != null ? pct.toFixed(2) + '%' : '-';
2510
+ const breach = pct != null && pct < SLA_TARGET;
2511
+ const pctColor = breach ? 'var(--red)' : pct != null && pct < 99.95 ? 'var(--yellow)' : 'var(--green)';
2512
+ const slaStatus = breach ? '⚠ BREACH' : pct != null ? '✓' : '-';
2513
+ const slaColor = breach ? 'color:var(--red);font-weight:600' : '';
2514
+ const statusColor = vm.status === 'healthy' ? 'var(--green)' : vm.status === 'degraded' ? 'var(--yellow)' : 'var(--red)';
2515
+ html += '<tr>';
2516
+ html += '<td>' + (first ? esc(vm.vm) : '') + '</td>';
2517
+ html += '<td' + (first ? ' style="color:' + statusColor + '"' : '') + '>' + (first ? esc(vm.status || '?') : '') + '</td>';
2518
+ html += '<td>' + esc(svc) + '</td>';
2519
+ html += '<td style="color:' + pctColor + '">' + pctStr + '</td>';
2520
+ html += '<td style="' + slaColor + '">' + slaStatus + '</td>';
2521
+ html += '</tr>';
2522
+ first = false;
2523
+ }
2524
+ }
2525
+ html += '</tbody></table>';
2526
+ html += '<div style="margin-top:8px;font-size:11px;color:var(--text-dim)">SLA target: ' + SLA_TARGET + '% · Based on Prometheus <code>up</code> metric</div>';
2527
+ slaEl.innerHTML = html;
2528
+ } else {
2529
+ slaEl.innerHTML = '<div class="empty">Fleet data not available — start with <code>fops serve --scrape</code></div>';
2530
+ }
2531
+ }
2532
+
2297
2533
  // ── Init ────────────────────────────────────────────────────
2298
2534
  init();
2299
2535
  </script>
package/src/web/api.js CHANGED
@@ -491,5 +491,78 @@ export function createApiRoutes(core, shared = {}) {
491
491
  return c.json(result);
492
492
  });
493
493
 
494
+ // ---------------------------------------------------------------------------
495
+ // POST /agent — Send a message to an agent and wait for the full response.
496
+ // Body: { agent?: string, message: string, timeout?: number }
497
+ // Returns: { ok: true, response: string, sessionId: string }
498
+ // ---------------------------------------------------------------------------
499
+ api.post("/agent", async (c) => {
500
+ const body = await c.req.json().catch(() => ({}));
501
+ const { agent, message, timeout: timeoutMs } = body;
502
+ if (!message) return c.json({ error: "message is required" }, 400);
503
+
504
+ const maxWait = Math.min(Number(timeoutMs) || 300_000, 600_000); // default 5m, max 10m
505
+
506
+ // Create a temporary session for this request
507
+ const sessionId = core.createSession(agent || undefined);
508
+
509
+ try {
510
+ const response = await new Promise((resolve, reject) => {
511
+ const timer = setTimeout(() => {
512
+ core.off(E.STREAM_END, onEnd);
513
+ core.off(E.ERROR, onError);
514
+ core.off(E.TOOL_CONFIRM, onConfirm);
515
+ reject(new Error("Agent response timed out"));
516
+ }, maxWait);
517
+
518
+ const onEnd = (data) => {
519
+ if (data.sessionId !== sessionId) return;
520
+ clearTimeout(timer);
521
+ core.off(E.STREAM_END, onEnd);
522
+ core.off(E.ERROR, onError);
523
+ core.off(E.TOOL_CONFIRM, onConfirm);
524
+ resolve(data.fullText || "");
525
+ };
526
+
527
+ const onError = (data) => {
528
+ if (data.sessionId !== sessionId) return;
529
+ clearTimeout(timer);
530
+ core.off(E.STREAM_END, onEnd);
531
+ core.off(E.ERROR, onError);
532
+ core.off(E.TOOL_CONFIRM, onConfirm);
533
+ reject(new Error(data.message || "Agent error"));
534
+ };
535
+
536
+ // Auto-approve tool confirmations for headless API calls
537
+ const onConfirm = (data) => {
538
+ if (data.sessionId !== sessionId) return;
539
+ if (core._pendingConfirmResolve) {
540
+ core._pendingConfirmResolve(true);
541
+ }
542
+ };
543
+
544
+ core.on(E.STREAM_END, onEnd);
545
+ core.on(E.ERROR, onError);
546
+ core.on(E.TOOL_CONFIRM, onConfirm);
547
+
548
+ core.sendMessage(sessionId, message).catch((err) => {
549
+ clearTimeout(timer);
550
+ core.off(E.STREAM_END, onEnd);
551
+ core.off(E.ERROR, onError);
552
+ core.off(E.TOOL_CONFIRM, onConfirm);
553
+ reject(err);
554
+ });
555
+ });
556
+
557
+ // Clean up the session
558
+ try { core.closeSession(sessionId); } catch {}
559
+
560
+ return c.json({ ok: true, response, sessionId });
561
+ } catch (err) {
562
+ try { core.closeSession(sessionId); } catch {}
563
+ return c.json({ error: err.message }, 500);
564
+ }
565
+ });
566
+
494
567
  return api;
495
568
  }
@@ -0,0 +1 @@
1
+ /*! tailwindcss v4.2.1 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-duration:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-red-300:oklch(80.8% .114 19.571);--color-red-400:oklch(70.4% .191 22.216);--color-red-500:oklch(63.7% .237 25.331);--color-yellow-300:oklch(90.5% .182 98.111);--color-yellow-400:oklch(85.2% .199 91.936);--color-emerald-300:oklch(84.5% .143 164.978);--color-emerald-400:oklch(76.5% .177 163.223);--color-emerald-500:oklch(69.6% .17 162.48);--color-cyan-300:oklch(86.5% .127 207.078);--color-cyan-400:oklch(78.9% .154 211.53);--color-purple-300:oklch(82.7% .119 306.383);--color-purple-400:oklch(71.4% .203 305.504);--color-purple-500:oklch(62.7% .265 303.9);--color-fuchsia-300:oklch(83.3% .145 321.434);--color-fuchsia-400:oklch(74% .238 322.16);--color-white:#fff;--spacing:.25rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height: 1.5 ;--text-6xl:3.75rem;--text-6xl--line-height:1;--font-weight-normal:400;--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-normal:0em;--tracking-wide:.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-relaxed:1.625;--radius-sm:.25rem;--radius-md:.375rem;--radius-lg:.5rem;--radius-xl:.75rem;--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite;--blur-2xl:40px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono)}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.absolute{position:absolute}.relative{position:relative}.inset-0{inset:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.-top-9{top:calc(var(--spacing) * -9)}.top-full{top:100%}.right-2{right:calc(var(--spacing) * 2)}.right-4{right:calc(var(--spacing) * 4)}.right-8{right:calc(var(--spacing) * 8)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-full{bottom:100%}.left-0{left:calc(var(--spacing) * 0)}.left-2{left:calc(var(--spacing) * 2)}.left-4{left:calc(var(--spacing) * 4)}.left-8{left:calc(var(--spacing) * 8)}.z-10{z-index:10}.z-50{z-index:50}.mx-1{margin-inline:calc(var(--spacing) * 1)}.mx-4{margin-inline:calc(var(--spacing) * 4)}.my-0\.5{margin-block:calc(var(--spacing) * .5)}.my-2{margin-block:calc(var(--spacing) * 2)}.my-3{margin-block:calc(var(--spacing) * 3)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-2\.5{margin-top:calc(var(--spacing) * 2.5)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-auto{margin-top:auto}.mr-2{margin-right:calc(var(--spacing) * 2)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-1\.5{margin-bottom:calc(var(--spacing) * 1.5)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-6{margin-bottom:calc(var(--spacing) * 6)}.ml-0\.5{margin-left:calc(var(--spacing) * .5)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-8{margin-left:calc(var(--spacing) * 8)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-2{height:calc(var(--spacing) * 2)}.h-3{height:calc(var(--spacing) * 3)}.h-3\.5{height:calc(var(--spacing) * 3.5)}.h-8{height:calc(var(--spacing) * 8)}.h-\[2px\]{height:2px}.h-\[5px\]{height:5px}.h-\[6px\]{height:6px}.h-\[14px\]{height:14px}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-20{max-height:calc(var(--spacing) * 20)}.max-h-52{max-height:calc(var(--spacing) * 52)}.w-1\.5{width:calc(var(--spacing) * 1.5)}.w-1\/2{width:50%}.w-3\.5{width:calc(var(--spacing) * 3.5)}.w-8{width:calc(var(--spacing) * 8)}.w-12{width:calc(var(--spacing) * 12)}.w-60{width:calc(var(--spacing) * 60)}.w-\[2px\]{width:2px}.w-\[5px\]{width:5px}.w-\[6px\]{width:6px}.w-full{width:100%}.w-px{width:1px}.max-w-\[240px\]{max-width:240px}.max-w-\[260px\]{max-width:260px}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-14{min-width:calc(var(--spacing) * 14)}.min-w-\[220px\]{min-width:220px}.flex-1{flex:1}.shrink-0{flex-shrink:0}.border-collapse{border-collapse:collapse}.rotate-90{rotate:90deg}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-grab{cursor:grab}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.resize-none{resize:none}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-end{align-items:flex-end}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-2\.5{gap:calc(var(--spacing) * 2.5)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}:where(.space-y-0\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * .5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * .5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 5) * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-sm{border-radius:var(--radius-sm)}.rounded-xl{border-radius:var(--radius-xl)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-l-2{border-left-style:var(--tw-border-style);border-left-width:2px}.border-\[\#1a1a28\]{border-color:#1a1a28}.border-\[\#1a1a28\]\/50{border-color:#1a1a2880}.border-\[\#2e2e40\]{border-color:#2e2e40}.border-\[\#5a5a78\]{border-color:#5a5a78}.border-\[\#818cf8\]\/20{border-color:#818cf833}.border-\[\#818cf8\]\/30{border-color:#818cf84d}.border-\[\#26263a\]{border-color:#26263a}.border-\[\#303045\]{border-color:#303045}.border-\[\#f97316\]{border-color:#f97316}.border-\[\#f97316\]\/20{border-color:#f9731633}.border-\[\#fb923c\]{border-color:#fb923c}.border-\[\#fbbf24\]\/20{border-color:#fbbf2433}.border-\[\#fbbf24\]\/35{border-color:#fbbf2459}.border-\[\#fdba74\]{border-color:#fdba74}.border-cyan-400\/30{border-color:#00d2ef4d}@supports (color:color-mix(in lab,red,red)){.border-cyan-400\/30{border-color:color-mix(in oklab,var(--color-cyan-400) 30%,transparent)}}.border-emerald-400\/20{border-color:#00d29433}@supports (color:color-mix(in lab,red,red)){.border-emerald-400\/20{border-color:color-mix(in oklab,var(--color-emerald-400) 20%,transparent)}}.border-emerald-400\/30{border-color:#00d2944d}@supports (color:color-mix(in lab,red,red)){.border-emerald-400\/30{border-color:color-mix(in oklab,var(--color-emerald-400) 30%,transparent)}}.border-emerald-400\/50{border-color:#00d29480}@supports (color:color-mix(in lab,red,red)){.border-emerald-400\/50{border-color:color-mix(in oklab,var(--color-emerald-400) 50%,transparent)}}.border-emerald-500\/20{border-color:#00bb7f33}@supports (color:color-mix(in lab,red,red)){.border-emerald-500\/20{border-color:color-mix(in oklab,var(--color-emerald-500) 20%,transparent)}}.border-purple-400\/25{border-color:#c07eff40}@supports (color:color-mix(in lab,red,red)){.border-purple-400\/25{border-color:color-mix(in oklab,var(--color-purple-400) 25%,transparent)}}.border-purple-500\/20{border-color:#ac4bff33}@supports (color:color-mix(in lab,red,red)){.border-purple-500\/20{border-color:color-mix(in oklab,var(--color-purple-500) 20%,transparent)}}.border-red-400\/20{border-color:#ff656833}@supports (color:color-mix(in lab,red,red)){.border-red-400\/20{border-color:color-mix(in oklab,var(--color-red-400) 20%,transparent)}}.border-red-400\/50{border-color:#ff656880}@supports (color:color-mix(in lab,red,red)){.border-red-400\/50{border-color:color-mix(in oklab,var(--color-red-400) 50%,transparent)}}.border-red-500\/20{border-color:#fb2c3633}@supports (color:color-mix(in lab,red,red)){.border-red-500\/20{border-color:color-mix(in oklab,var(--color-red-500) 20%,transparent)}}.border-transparent{border-color:#0000}.border-yellow-400\/30{border-color:#fac8004d}@supports (color:color-mix(in lab,red,red)){.border-yellow-400\/30{border-color:color-mix(in oklab,var(--color-yellow-400) 30%,transparent)}}.border-t-purple-400{border-top-color:var(--color-purple-400)}.bg-\[\#0a0a10\]{background-color:#0a0a10}.bg-\[\#0c0c12\]{background-color:#0c0c12}.bg-\[\#0c0c14\]{background-color:#0c0c14}.bg-\[\#0e0e16\]{background-color:#0e0e16}.bg-\[\#0f0f16\]{background-color:#0f0f16}.bg-\[\#1a1a28\]{background-color:#1a1a28}.bg-\[\#1e1e2e\]{background-color:#1e1e2e}.bg-\[\#3a3a50\]{background-color:#3a3a50}.bg-\[\#818cf8\]{background-color:#818cf8}.bg-\[\#06060a\]{background-color:#06060a}.bg-\[\#08080c\]{background-color:#08080c}.bg-\[\#12121a\]{background-color:#12121a}.bg-\[\#14141c\]{background-color:#14141c}.bg-\[\#18181f\]{background-color:#18181f}.bg-\[\#ea580c\]{background-color:#ea580c}.bg-\[\#f97316\]{background-color:#f97316}.bg-\[\#f97316\]\/8{background-color:#f9731614}.bg-\[\#f97316\]\/20{background-color:#f9731633}.bg-\[\#fbbf24\]\/8{background-color:#fbbf2414}.bg-emerald-400{background-color:var(--color-emerald-400)}.bg-emerald-400\/8{background-color:#00d29414}@supports (color:color-mix(in lab,red,red)){.bg-emerald-400\/8{background-color:color-mix(in oklab,var(--color-emerald-400) 8%,transparent)}}.bg-emerald-500\/5{background-color:#00bb7f0d}@supports (color:color-mix(in lab,red,red)){.bg-emerald-500\/5{background-color:color-mix(in oklab,var(--color-emerald-500) 5%,transparent)}}.bg-emerald-500\/10{background-color:#00bb7f1a}@supports (color:color-mix(in lab,red,red)){.bg-emerald-500\/10{background-color:color-mix(in oklab,var(--color-emerald-500) 10%,transparent)}}.bg-red-400{background-color:var(--color-red-400)}.bg-red-400\/8{background-color:#ff656814}@supports (color:color-mix(in lab,red,red)){.bg-red-400\/8{background-color:color-mix(in oklab,var(--color-red-400) 8%,transparent)}}.bg-red-400\/50{background-color:#ff656880}@supports (color:color-mix(in lab,red,red)){.bg-red-400\/50{background-color:color-mix(in oklab,var(--color-red-400) 50%,transparent)}}.bg-red-400\/60{background-color:#ff656899}@supports (color:color-mix(in lab,red,red)){.bg-red-400\/60{background-color:color-mix(in oklab,var(--color-red-400) 60%,transparent)}}.bg-red-500\/5{background-color:#fb2c360d}@supports (color:color-mix(in lab,red,red)){.bg-red-500\/5{background-color:color-mix(in oklab,var(--color-red-500) 5%,transparent)}}.bg-red-500\/10{background-color:#fb2c361a}@supports (color:color-mix(in lab,red,red)){.bg-red-500\/10{background-color:color-mix(in oklab,var(--color-red-500) 10%,transparent)}}.bg-transparent{background-color:#0000}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.px-1{padding-inline:calc(var(--spacing) * 1)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0{padding-block:calc(var(--spacing) * 0)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-5{padding-block:calc(var(--spacing) * 5)}.py-\[1px\]{padding-block:1px}.py-\[2px\]{padding-block:2px}.py-\[3px\]{padding-block:3px}.pr-3{padding-right:calc(var(--spacing) * 3)}.pb-0\.5{padding-bottom:calc(var(--spacing) * .5)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-3{padding-left:calc(var(--spacing) * 3)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-5{padding-left:calc(var(--spacing) * 5)}.pl-\[6px\]{padding-left:6px}.text-center{text-align:center}.text-left{text-align:left}.align-middle{vertical-align:middle}.font-mono{font-family:var(--font-mono)}.text-6xl{font-size:var(--text-6xl);line-height:var(--tw-leading,var(--text-6xl--line-height))}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[8px\]{font-size:8px}.text-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-normal{--tw-font-weight:var(--font-weight-normal);font-weight:var(--font-weight-normal)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.2em\]{--tw-tracking:.2em;letter-spacing:.2em}.tracking-normal{--tw-tracking:var(--tracking-normal);letter-spacing:var(--tracking-normal)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.break-words{overflow-wrap:break-word}.break-all{word-break:break-all}.whitespace-pre-wrap{white-space:pre-wrap}.text-\[\#1e1e2a\]{color:#1e1e2a}.text-\[\#2e2e40\]{color:#2e2e40}.text-\[\#3a3a50\]{color:#3a3a50}.text-\[\#4e4e63\]{color:#4e4e63}.text-\[\#5a5a70\]{color:#5a5a70}.text-\[\#6b6b80\]{color:#6b6b80}.text-\[\#6e7a8a\]{color:#6e7a8a}.text-\[\#7f7f95\]{color:#7f7f95}.text-\[\#8b8b9e\]{color:#8b8b9e}.text-\[\#818cf8\]{color:#818cf8}.text-\[\#08080c\]{color:#08080c}.text-\[\#18181f\]{color:#18181f}.text-\[\#26263a\]{color:#26263a}.text-\[\#a0a0b0\]{color:#a0a0b0}.text-\[\#b5b5c4\]{color:#b5b5c4}.text-\[\#c0bfc6\]{color:#c0bfc6}.text-\[\#e0dfe4\]{color:#e0dfe4}.text-\[\#f5f5f7\]{color:#f5f5f7}.text-\[\#f97316\]{color:#f97316}.text-\[\#fb923c\]{color:#fb923c}.text-\[\#fbbf24\]{color:#fbbf24}.text-cyan-300\/70{color:#53eafdb3}@supports (color:color-mix(in lab,red,red)){.text-cyan-300\/70{color:color-mix(in oklab,var(--color-cyan-300) 70%,transparent)}}.text-cyan-400{color:var(--color-cyan-400)}.text-emerald-300{color:var(--color-emerald-300)}.text-emerald-300\/70{color:#5ee9b5b3}@supports (color:color-mix(in lab,red,red)){.text-emerald-300\/70{color:color-mix(in oklab,var(--color-emerald-300) 70%,transparent)}}.text-emerald-400{color:var(--color-emerald-400)}.text-fuchsia-300\/80{color:#f2a9ffcc}@supports (color:color-mix(in lab,red,red)){.text-fuchsia-300\/80{color:color-mix(in oklab,var(--color-fuchsia-300) 80%,transparent)}}.text-fuchsia-400{color:var(--color-fuchsia-400)}.text-purple-300{color:var(--color-purple-300)}.text-red-300{color:var(--color-red-300)}.text-red-400{color:var(--color-red-400)}.text-white{color:var(--color-white)}.text-yellow-300\/70{color:#ffe02ab3}@supports (color:color-mix(in lab,red,red)){.text-yellow-300\/70{color:color-mix(in oklab,var(--color-yellow-300) 70%,transparent)}}.text-yellow-400{color:var(--color-yellow-400)}.normal-case{text-transform:none}.uppercase{text-transform:uppercase}.italic{font-style:italic}.underline{text-decoration-line:underline}.decoration-\[\#f97316\]\/30{text-decoration-color:#f973164d}.underline-offset-2{text-underline-offset:2px}.opacity-0{opacity:0}.opacity-60{opacity:.6}.opacity-\[0\.07\]{opacity:.07}.shadow-2xl{--tw-shadow:0 25px 50px -12px var(--tw-shadow-color,#00000040);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_1px_rgba\(255\,255\,255\,0\.06\)\]{--tw-shadow:0 0 0 1px var(--tw-shadow-color,#ffffff0f);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_12px_rgba\(249\,115\,22\,0\.3\)\]{--tw-shadow:0 0 12px var(--tw-shadow-color,#f973164d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_6px_24px_rgba\(0\,0\,0\,0\.35\)\]{--tw-shadow:0 6px 24px var(--tw-shadow-color,#00000059);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a), 0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.blur-2xl{--tw-blur:blur(var(--blur-2xl));filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-150{--tw-duration:.15s;transition-duration:.15s}.duration-200{--tw-duration:.2s;transition-duration:.2s}.duration-300{--tw-duration:.3s;transition-duration:.3s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}@media(hover:hover){.group-hover\:opacity-100:is(:where(.group):hover *){opacity:1}}.placeholder\:text-\[\#2e2e40\]::placeholder{color:#2e2e40}.focus-within\:border-\[\#f97316\]\/50:focus-within{border-color:#f9731680}.focus-within\:shadow-\[0_0_20px_rgba\(249\,115\,22\,0\.08\)\]:focus-within{--tw-shadow:0 0 20px var(--tw-shadow-color,#f9731614);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}@media(hover:hover){.hover\:border-\[\#f97316\]:hover{border-color:#f97316}.hover\:border-\[\#f97316\]\/40:hover{border-color:#f9731666}.hover\:bg-\[\#1a1a28\]:hover{background-color:#1a1a28}.hover\:bg-\[\#1c1c28\]:hover{background-color:#1c1c28}.hover\:bg-\[\#1f1f2c\]:hover{background-color:#1f1f2c}.hover\:bg-\[\#12121a\]:hover{background-color:#12121a}.hover\:bg-\[\#14141c\]:hover{background-color:#14141c}.hover\:bg-\[\#18181f\]:hover{background-color:#18181f}.hover\:bg-\[\#f97316\]:hover{background-color:#f97316}.hover\:bg-\[\#f97316\]\/20:hover{background-color:#f9731633}.hover\:bg-\[\#fb923c\]:hover{background-color:#fb923c}.hover\:bg-emerald-500\/20:hover{background-color:#00bb7f33}@supports (color:color-mix(in lab,red,red)){.hover\:bg-emerald-500\/20:hover{background-color:color-mix(in oklab,var(--color-emerald-500) 20%,transparent)}}.hover\:bg-red-500\/20:hover{background-color:#fb2c3633}@supports (color:color-mix(in lab,red,red)){.hover\:bg-red-500\/20:hover{background-color:color-mix(in oklab,var(--color-red-500) 20%,transparent)}}.hover\:text-\[\#8b8b9e\]:hover{color:#8b8b9e}.hover\:text-\[\#e0dfe4\]:hover{color:#e0dfe4}.hover\:text-\[\#f87171\]:hover{color:#f87171}.hover\:text-\[\#f97316\]:hover{color:#f97316}.hover\:text-\[\#fb923c\]:hover{color:#fb923c}.hover\:decoration-\[\#fb923c\]\/50:hover{text-decoration-color:#fb923c80}}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.active\:cursor-grabbing:active{cursor:grabbing}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}}body{color:#e0dfe4;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;background:#08080c;height:100dvh;margin:0;padding:0;font-family:Sora,system-ui,-apple-system,sans-serif;overflow:hidden}#root{flex-direction:column;height:100dvh;display:flex}code,pre,.font-mono{font-family:IBM Plex Mono,JetBrains Mono,Fira Code,monospace}::selection{color:#e0dfe4;background:#f973164d}::-webkit-scrollbar{width:5px}::-webkit-scrollbar-track{background:0 0}::-webkit-scrollbar-thumb{background:#26263a;border-radius:10px}::-webkit-scrollbar-thumb:hover{background:#3f3f56}@keyframes pulse{50%{opacity:.5}}.animate-pulse{animation:2s cubic-bezier(.4,0,.6,1) infinite pulse}@keyframes fadeIn{0%{opacity:0;transform:translateY(6px)}to{opacity:1;transform:translateY(0)}}.animate-fade-in{animation:.25s ease-out fadeIn}@keyframes slideIn{0%{opacity:0;transform:translate(-6px)}to{opacity:1;transform:translate(0)}}.animate-slide-in{animation:.2s ease-out slideIn}@keyframes cursorBlink{0%,to{opacity:1}50%{opacity:0}}.animate-cursor{animation:1s step-end infinite cursorBlink}@keyframes glowPulse{0%,to{box-shadow:0 0 8px #f9731626}50%{box-shadow:0 0 16px #f973164d}}.noise:after{content:"";pointer-events:none;opacity:.02;z-index:9999;background-image:url("data:image/svg+xml,%3Csvg viewBox='0 0 256 256' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.85' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)'/%3E%3C/svg%3E");position:fixed;top:0;right:0;bottom:0;left:0}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}