@syncular/console 0.0.4-26 → 0.0.5-42

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 (66) hide show
  1. package/dist/App.d.ts +1 -1
  2. package/dist/App.d.ts.map +1 -1
  3. package/dist/hooks/ConnectionContext.d.ts.map +1 -1
  4. package/dist/hooks/ConnectionContext.js +5 -14
  5. package/dist/hooks/ConnectionContext.js.map +1 -1
  6. package/dist/hooks/useConsoleApi.d.ts +11 -1
  7. package/dist/hooks/useConsoleApi.d.ts.map +1 -1
  8. package/dist/hooks/useConsoleApi.js +78 -0
  9. package/dist/hooks/useConsoleApi.js.map +1 -1
  10. package/dist/hooks/useLiveEvents.d.ts.map +1 -1
  11. package/dist/hooks/useLiveEvents.js +94 -0
  12. package/dist/hooks/useLiveEvents.js.map +1 -1
  13. package/dist/layout.d.ts.map +1 -1
  14. package/dist/layout.js +1 -0
  15. package/dist/layout.js.map +1 -1
  16. package/dist/lib/api.d.ts +1 -1
  17. package/dist/lib/api.d.ts.map +1 -1
  18. package/dist/lib/api.js +36 -4
  19. package/dist/lib/api.js.map +1 -1
  20. package/dist/lib/types.d.ts +13 -0
  21. package/dist/lib/types.d.ts.map +1 -1
  22. package/dist/pages/Config.d.ts +3 -1
  23. package/dist/pages/Config.d.ts.map +1 -1
  24. package/dist/pages/Config.js +2 -3
  25. package/dist/pages/Config.js.map +1 -1
  26. package/dist/pages/Fleet.d.ts +3 -1
  27. package/dist/pages/Fleet.d.ts.map +1 -1
  28. package/dist/pages/Fleet.js +6 -3
  29. package/dist/pages/Fleet.js.map +1 -1
  30. package/dist/pages/Storage.d.ts +2 -0
  31. package/dist/pages/Storage.d.ts.map +1 -0
  32. package/dist/pages/Storage.js +103 -0
  33. package/dist/pages/Storage.js.map +1 -0
  34. package/dist/pages/index.d.ts +1 -0
  35. package/dist/pages/index.d.ts.map +1 -1
  36. package/dist/pages/index.js +1 -0
  37. package/dist/pages/index.js.map +1 -1
  38. package/dist/routeTree.d.ts +1 -1
  39. package/dist/routeTree.d.ts.map +1 -1
  40. package/dist/routeTree.js +2 -0
  41. package/dist/routeTree.js.map +1 -1
  42. package/dist/routes/storage.d.ts +2 -0
  43. package/dist/routes/storage.d.ts.map +1 -0
  44. package/dist/routes/storage.js +9 -0
  45. package/dist/routes/storage.js.map +1 -0
  46. package/dist/styles.css +1 -1
  47. package/package.json +8 -8
  48. package/src/hooks/ConnectionContext.tsx +5 -15
  49. package/src/hooks/useConsoleApi.ts +103 -0
  50. package/src/hooks/useLiveEvents.ts +115 -1
  51. package/src/layout.tsx +8 -1
  52. package/src/lib/api.ts +38 -5
  53. package/src/lib/types.ts +17 -0
  54. package/src/pages/Config.tsx +2 -1
  55. package/src/pages/Fleet.tsx +19 -17
  56. package/src/pages/Storage.tsx +277 -0
  57. package/src/pages/index.ts +1 -0
  58. package/src/routeTree.ts +2 -0
  59. package/src/routes/storage.tsx +9 -0
  60. package/src/styles/globals.css +4 -1
  61. package/web-dist/assets/index-BhPtRvK0.css +1 -0
  62. package/web-dist/assets/index-Fyq7dTrO.js +86 -0
  63. package/web-dist/console.css +1 -1
  64. package/web-dist/index.html +2 -2
  65. package/web-dist/assets/index-CTkQp6YC.js +0 -86
  66. package/web-dist/assets/index-j_U2SoXa.css +0 -1
package/dist/styles.css CHANGED
@@ -1 +1 @@
1
- @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-space-y-reverse:0;--tw-border-style:solid;--tw-font-weight:initial;--tw-tracking:initial;--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-ease:initial}}}:root,:host{--color-neutral-100:oklch(97% 0 0);--color-neutral-200:oklch(92.2% 0 0);--color-neutral-300:oklch(87% 0 0);--color-neutral-400:oklch(70.8% 0 0);--color-neutral-500:oklch(55.6% 0 0);--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);--font-weight-medium:500;--tracking-wide:.025em;--tracking-wider:.05em;--ease-out:cubic-bezier(0,0,.2,1)}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.container{width:100%}@media(min-width:40rem){.container{max-width:40rem}}@media(min-width:48rem){.container{max-width:48rem}}@media(min-width:64rem){.container{max-width:64rem}}@media(min-width:80rem){.container{max-width:80rem}}@media(min-width:96rem){.container{max-width:96rem}}.-mt-2{margin-top:calc(var(--spacing)*-2)}.mb-2{margin-bottom:calc(var(--spacing)*2)}.mb-4{margin-bottom:calc(var(--spacing)*4)}.block{display:block}.flex{display:flex}.grid{display:grid}.table{display:table}.h-7{height:calc(var(--spacing)*7)}.h-\[200px\]{height:200px}.h-full{height:100%}.h-screen{height:100vh}.max-h-\[70vh\]{max-height:70vh}.min-h-full{min-height:100%}.min-h-screen{min-height:100vh}.w-\[110px\]{width:110px}.max-w-\[220px\]{max-width:220px}.min-w-0{min-width:calc(var(--spacing)*0)}.flex-1{flex:1}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.flex-col{flex-direction:column}.items-center{align-items:center}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing)*1)}.gap-2{gap:calc(var(--spacing)*2)}.gap-3{gap:calc(var(--spacing)*3)}.gap-4{gap:calc(var(--spacing)*4)}.gap-5{gap:calc(var(--spacing)*5)}: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-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*3)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*3)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*4)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*4)*calc(1 - var(--tw-space-y-reverse)))}.overflow-auto{overflow:auto}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-md{border-radius:var(--syncular-radius-md)}.border{border-style:var(--tw-border-style);border-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-border{border-color:var(--syncular-color-border)}.bg-background{background-color:var(--syncular-color-background)}.bg-surface{background-color:var(--syncular-color-surface)}.p-2{padding:calc(var(--spacing)*2)}.p-3{padding:calc(var(--spacing)*3)}.p-4{padding:calc(var(--spacing)*4)}.px-2{padding-inline:calc(var(--spacing)*2)}.px-3{padding-inline:calc(var(--spacing)*3)}.px-5{padding-inline:calc(var(--spacing)*5)}.px-6{padding-inline:calc(var(--spacing)*6)}.py-1{padding-block:calc(var(--spacing)*1)}.py-2{padding-block:calc(var(--spacing)*2)}.py-4{padding-block:calc(var(--spacing)*4)}.py-5{padding-block:calc(var(--spacing)*5)}.py-8{padding-block:calc(var(--spacing)*8)}.py-10{padding-block:calc(var(--spacing)*10)}.py-16{padding-block:calc(var(--spacing)*16)}.py-24{padding-block:calc(var(--spacing)*24)}.pt-2{padding-top:calc(var(--spacing)*2)}.pt-\[42px\]{padding-top:42px}.pb-4{padding-bottom:calc(var(--spacing)*4)}.pb-5{padding-bottom:calc(var(--spacing)*5)}.pb-\[32px\]{padding-bottom:32px}.font-mono{font-family:var(--syncular-font-mono)}.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-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.break-all{word-break:break-all}.whitespace-nowrap{white-space:nowrap}.text-flow{color:var(--syncular-color-flow)}.text-foreground{color:var(--syncular-color-foreground)}.text-foreground-muted{color:var(--syncular-color-foreground-muted)}.text-neutral-100{color:var(--color-neutral-100)}.text-neutral-200{color:var(--color-neutral-200)}.text-neutral-300{color:var(--color-neutral-300)}.text-neutral-400{color:var(--color-neutral-400)}.text-neutral-500{color:var(--color-neutral-500)}.text-offline{color:var(--syncular-color-offline)}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.underline{text-decoration-line:underline}.underline-offset-4{text-underline-offset:4px}.filter{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,)}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}@media(min-width:64rem){.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}@media(min-width:80rem){.xl\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}:where(.syncular-console-root,.syncular-ui-root){--syncular-color-background:#0c0c0c;--syncular-color-surface:#0c0c0c;--syncular-color-panel:#111;--syncular-color-panel-alt:#0e0e0e;--syncular-color-border:#1e1e1e;--syncular-color-border-bright:#2a2a2a;--syncular-color-healthy:#22c55e;--syncular-color-syncing:#f59e0b;--syncular-color-offline:#ef4444;--syncular-color-flow:#3b82f6;--syncular-color-relay:#8b5cf6;--syncular-color-encrypt:#f472b6;--syncular-color-foreground:#e5e5e5;--syncular-color-foreground-muted:#737373;--syncular-color-primary:#22c55e;--syncular-color-destructive:#ef4444;--syncular-color-secondary:#1e1e1e;--syncular-font-display:"Inter Tight",system-ui,sans-serif;--syncular-font-mono:"JetBrains Mono",ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--syncular-radius-sm:4px;--syncular-radius-md:6px;--syncular-radius-lg:8px;--syncular-radius-xl:10px;--syncular-glow-healthy:0 0 6px #22c55e;--syncular-glow-syncing:0 0 6px #f59e0b;--syncular-glow-offline:0 0 6px #ef4444;--syncular-glow-flow:0 0 6px #3b82f6;--syncular-glow-relay:0 0 6px #8b5cf6;--syncular-glow-encrypt:0 0 6px #f472b6;--color-background:var(--syncular-color-background);--color-surface:var(--syncular-color-surface);--color-panel:var(--syncular-color-panel);--color-panel-alt:var(--syncular-color-panel-alt);--color-border:var(--syncular-color-border);--color-border-bright:var(--syncular-color-border-bright);--color-healthy:var(--syncular-color-healthy);--color-syncing:var(--syncular-color-syncing);--color-offline:var(--syncular-color-offline);--color-flow:var(--syncular-color-flow);--color-relay:var(--syncular-color-relay);--color-encrypt:var(--syncular-color-encrypt);--color-foreground:var(--syncular-color-foreground);--color-foreground-muted:var(--syncular-color-foreground-muted);--color-primary:var(--syncular-color-primary);--color-destructive:var(--syncular-color-destructive);--color-secondary:var(--syncular-color-secondary);--font-display:var(--syncular-font-display);--font-mono:var(--syncular-font-mono);--radius-sm:var(--syncular-radius-sm);--radius-md:var(--syncular-radius-md);--radius-lg:var(--syncular-radius-lg);--radius-xl:var(--syncular-radius-xl);--sync-glow-healthy:var(--syncular-glow-healthy);--sync-glow-syncing:var(--syncular-glow-syncing);--sync-glow-offline:var(--syncular-glow-offline);--sync-glow-flow:var(--syncular-glow-flow);--sync-glow-relay:var(--syncular-glow-relay);--sync-glow-encrypt:var(--syncular-glow-encrypt);background:var(--syncular-color-background);color:var(--syncular-color-foreground);font-family:var(--syncular-font-display),ui-sans-serif,system-ui,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;margin:0}:where(.syncular-console-root,.syncular-ui-root) .font-mono{font-family:var(--syncular-font-mono),ui-monospace,monospace}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar{width:3px;height:3px}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar-track{background:0 0}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar-thumb{background:#333;border-radius:2px}@keyframes dotPulse{0%,to{opacity:1}50%{opacity:.4}}@keyframes dashFlow{0%{stroke-dashoffset:20px}to{stroke-dashoffset:0}}@keyframes streamSlide{0%{opacity:0;transform:translateY(-6px)}15%{opacity:1;transform:translateY(0)}}@keyframes pageIn{0%{opacity:0;transform:translateY(6px)}to{opacity:1;transform:translateY(0)}}@keyframes scanSweep{0%{opacity:0;top:0}10%{opacity:.6}90%{opacity:.6}to{opacity:0;top:100%}}@keyframes heartbeat{0%,40%,to{transform:scaleY(1)}10%{transform:scaleY(2.5)}25%{transform:scaleY(1.5)}}@keyframes fadeUp{0%{opacity:0;transform:translateY(12px)}to{opacity:1;transform:translateY(0)}}:where(.syncular-console-root,.syncular-ui-root) .line-active{stroke-dasharray:4 2;animation:1s linear infinite dashFlow}:where(.syncular-console-root,.syncular-ui-root) .stream-entry{animation:.35s ease-out streamSlide}:where(.syncular-console-root,.syncular-ui-root) .dot-pulse{animation:1.4s ease-in-out infinite dotPulse}@keyframes flowPulse{0%{opacity:.15}50%{opacity:.6}to{opacity:.15}}:where(.syncular-console-root,.syncular-ui-root) .flow-arrow{animation:2s ease-in-out infinite flowPulse}@keyframes mediaLoad{0%{opacity:0;transform:scale(.92)}to{opacity:1;transform:scale(1)}}:where(.syncular-console-root,.syncular-ui-root) .dot-grid{background-image:radial-gradient(circle at 1px 1px,#1a1a1a 1px,#0000 0);background-size:24px 24px}:where(.syncular-console-root,.syncular-ui-root) .scan-line{pointer-events:none;background:linear-gradient(90deg,#0000,#3b82f64d,#0000);height:1px;animation:6s ease-in-out infinite scanSweep;position:absolute;left:0;right:0}:where(.syncular-console-root){color-scheme:dark}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@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-ease{syntax:"*";inherits:false}
1
+ @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-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--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-ordinal:initial;--tw-slashed-zero:initial;--tw-numeric-figure:initial;--tw-numeric-spacing:initial;--tw-numeric-fraction: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-outline-style:solid;--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-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-ease:initial;--tw-content:""}}}@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";--color-red-500:oklch(63.7% .237 25.331);--color-amber-300:oklch(87.9% .169 91.605);--color-amber-400:oklch(82.8% .189 84.429);--color-amber-500:oklch(76.9% .188 70.08);--color-emerald-400:oklch(76.5% .177 163.223);--color-emerald-500:oklch(69.6% .17 162.48);--color-blue-400:oklch(70.7% .165 254.624);--color-blue-600:oklch(54.6% .245 262.881);--color-violet-400:oklch(70.2% .183 293.541);--color-purple-500:oklch(62.7% .265 303.9);--color-purple-600:oklch(55.8% .288 302.321);--color-pink-400:oklch(71.8% .202 349.761);--color-neutral-100:oklch(97% 0 0);--color-neutral-200:oklch(92.2% 0 0);--color-neutral-300:oklch(87% 0 0);--color-neutral-400:oklch(70.8% 0 0);--color-neutral-500:oklch(55.6% 0 0);--color-neutral-600:oklch(43.9% 0 0);--color-neutral-700:oklch(37.1% 0 0);--color-neutral-800:oklch(26.9% 0 0);--color-neutral-900:oklch(20.5% 0 0);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-lg:32rem;--container-2xl:42rem;--container-3xl:48rem;--container-4xl:56rem;--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-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--text-3xl:1.875rem;--text-3xl--line-height: 1.2 ;--text-4xl:2.25rem;--text-4xl--line-height:calc(2.5 / 2.25);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-wide:.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-tight:1.25;--leading-relaxed:1.625;--ease-out:cubic-bezier(0, 0, .2, 1);--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite;--blur-sm:8px;--blur-lg:16px;--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(--syncular-font-mono);--color-border:var(--syncular-color-border);--color-healthy:var(--syncular-color-healthy);--color-offline:var(--syncular-color-offline);--color-flow:var(--syncular-color-flow)}}@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%;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]){appearance:button}::file-selector-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{.pointer-events-none{pointer-events:none}.visible{visibility:visible}.sr-only{clip-path:inset(50%);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.inset-0{inset:calc(var(--spacing) * 0)}.inset-x-0{inset-inline:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.top-0{top:calc(var(--spacing) * 0)}.top-0\.5{top:calc(var(--spacing) * .5)}.top-1\/2{top:50%}.top-3{top:calc(var(--spacing) * 3)}.top-4{top:calc(var(--spacing) * 4)}.right-0{right:calc(var(--spacing) * 0)}.right-5{right:calc(var(--spacing) * 5)}.-bottom-2\.5{bottom:calc(var(--spacing) * -2.5)}.bottom-0{bottom:calc(var(--spacing) * 0)}.left-0{left:calc(var(--spacing) * 0)}.left-0\.5{left:calc(var(--spacing) * .5)}.left-1\/2{left:50%}.left-4{left:calc(var(--spacing) * 4)}.left-5{left:calc(var(--spacing) * 5)}.z-10{z-index:10}.z-50{z-index:50}.z-100{z-index:100}.container{width:100%}@media(min-width:40rem){.container{max-width:40rem}}@media(min-width:48rem){.container{max-width:48rem}}@media(min-width:64rem){.container{max-width:64rem}}@media(min-width:80rem){.container{max-width:80rem}}@media(min-width:96rem){.container{max-width:96rem}}.m-0{margin:calc(var(--spacing) * 0)}.mx-auto{margin-inline:auto}.my-0\.5{margin-block:calc(var(--spacing) * .5)}.-mt-2{margin-top:calc(var(--spacing) * -2)}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-1\.5{margin-top:calc(var(--spacing) * 1.5)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-6{margin-top:calc(var(--spacing) * 6)}.mt-10{margin-top:calc(var(--spacing) * 10)}.mr-1{margin-right:calc(var(--spacing) * 1)}.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-4{margin-bottom:calc(var(--spacing) * 4)}.mb-5{margin-bottom:calc(var(--spacing) * 5)}.mb-16{margin-bottom:calc(var(--spacing) * 16)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-2{margin-left:calc(var(--spacing) * 2)}.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}.table{display:table}.aspect-square{aspect-ratio:1}.size-2\.5{width:calc(var(--spacing) * 2.5);height:calc(var(--spacing) * 2.5)}.size-3\.5{width:calc(var(--spacing) * 3.5);height:calc(var(--spacing) * 3.5)}.h-0\.5{height:calc(var(--spacing) * .5)}.h-1{height:calc(var(--spacing) * 1)}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-2{height:calc(var(--spacing) * 2)}.h-2\.5{height:calc(var(--spacing) * 2.5)}.h-3{height:calc(var(--spacing) * 3)}.h-4{height:calc(var(--spacing) * 4)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-7{height:calc(var(--spacing) * 7)}.h-8{height:calc(var(--spacing) * 8)}.h-10{height:calc(var(--spacing) * 10)}.h-12{height:calc(var(--spacing) * 12)}.h-16{height:calc(var(--spacing) * 16)}.h-\[3px\]{height:3px}.h-\[5px\]{height:5px}.h-\[14px\]{height:14px}.h-\[15px\]{height:15px}.h-\[18px\]{height:18px}.h-\[42px\]{height:42px}.h-\[120px\]{height:120px}.h-\[200px\]{height:200px}.h-\[400px\]{height:400px}.h-auto{height:auto}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-\[70vh\]{max-height:70vh}.min-h-\[2px\]{min-height:2px}.min-h-\[80px\]{min-height:80px}.min-h-\[120px\]{min-height:120px}.min-h-\[280px\]{min-height:280px}.min-h-full{min-height:100%}.min-h-screen{min-height:100vh}.w-0{width:calc(var(--spacing) * 0)}.w-0\.5{width:calc(var(--spacing) * .5)}.w-1{width:calc(var(--spacing) * 1)}.w-1\.5{width:calc(var(--spacing) * 1.5)}.w-2{width:calc(var(--spacing) * 2)}.w-2\.5{width:calc(var(--spacing) * 2.5)}.w-3{width:calc(var(--spacing) * 3)}.w-4{width:calc(var(--spacing) * 4)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-7{width:calc(var(--spacing) * 7)}.w-8{width:calc(var(--spacing) * 8)}.w-10{width:calc(var(--spacing) * 10)}.w-28{width:calc(var(--spacing) * 28)}.w-32{width:calc(var(--spacing) * 32)}.w-36{width:calc(var(--spacing) * 36)}.w-48{width:calc(var(--spacing) * 48)}.w-\[5px\]{width:5px}.w-\[14px\]{width:14px}.w-\[15px\]{width:15px}.w-\[34px\]{width:34px}.w-\[50px\]{width:50px}.w-\[55px\]{width:55px}.w-\[60px\]{width:60px}.w-\[65px\]{width:65px}.w-\[70px\]{width:70px}.w-\[80px\]{width:80px}.w-\[90px\]{width:90px}.w-\[100px\]{width:100px}.w-\[110px\]{width:110px}.w-\[120px\]{width:120px}.w-\[140px\]{width:140px}.w-\[400px\]{width:400px}.w-full{width:100%}.w-px{width:1px}.max-w-2xl{max-width:var(--container-2xl)}.max-w-3xl{max-width:var(--container-3xl)}.max-w-4xl{max-width:var(--container-4xl)}.max-w-\[200px\]{max-width:200px}.max-w-\[220px\]{max-width:220px}.max-w-\[320px\]{max-width:320px}.max-w-\[660px\]{max-width:660px}.max-w-\[700px\]{max-width:700px}.max-w-\[1400px\]{max-width:1400px}.max-w-lg{max-width:var(--container-lg)}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-\[8rem\]{min-width:8rem}.min-w-\[700px\]{min-width:700px}.flex-1{flex:1}.flex-shrink-0,.shrink-0{flex-shrink:0}.-translate-x-1\/2{--tw-translate-x: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-y-1\/2{--tw-translate-y: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-\[dotPulse_1\.5s_ease-in-out_infinite\]{animation:1.5s ease-in-out infinite dotPulse}.animate-\[dotPulse_1s_ease-in-out_infinite\]{animation:1s ease-in-out infinite dotPulse}.animate-\[heartbeat_1\.2s_ease-in-out_infinite\]{animation:1.2s ease-in-out infinite heartbeat}.animate-\[scanSweep_6s_ease-in-out_infinite\]{animation:6s ease-in-out infinite scanSweep}.animate-\[streamSlide_0\.35s_ease-out\]{animation:.35s ease-out streamSlide}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-copy{cursor:copy}.cursor-default{cursor:default}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.resize-none{resize:none}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.grid-cols-\[80px_1fr_100px_80px\]{grid-template-columns:80px 1fr 100px 80px}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.items-stretch{align-items:stretch}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.gap-0{gap:calc(var(--spacing) * 0)}.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)}.gap-5{gap:calc(var(--spacing) * 5)}.gap-6{gap:calc(var(--spacing) * 6)}.gap-8{gap:calc(var(--spacing) * 8)}.gap-\[1px\]{gap:1px}.gap-\[2px\]{gap:2px}: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-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * 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-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 4) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 4) * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-\[2px\]{border-radius:2px}.rounded-\[3px\]{border-radius:3px}.rounded-\[4px\]{border-radius:4px}.rounded-\[6px\]{border-radius:6px}.rounded-\[10px\]{border-radius:10px}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--syncular-radius-lg)}.rounded-md{border-radius:var(--syncular-radius-md)}.rounded-sm{border-radius:var(--syncular-radius-sm)}.rounded-t{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.rounded-t-sm{border-top-left-radius:var(--syncular-radius-sm);border-top-right-radius:var(--syncular-radius-sm)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-\[1\.5px\]{border-style:var(--tw-border-style);border-width:1.5px}.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-dashed{--tw-border-style:dashed;border-style:dashed}.border-none{--tw-border-style:none;border-style:none}.border-\[\#1a1a1a\]{border-color:#1a1a1a}.border-\[\#141414\]{border-color:#141414}.border-\[\#161616\]{border-color:#161616}.border-\[--color-border\]{border-color:--color-border}.border-amber-300\/30{border-color:#ffd2364d}@supports (color:color-mix(in lab,red,red)){.border-amber-300\/30{border-color:color-mix(in oklab,var(--color-amber-300) 30%,transparent)}}.border-border{border-color:var(--syncular-color-border)}.border-border-bright{border-color:var(--syncular-color-border-bright)}.border-border\/50{border-color:var(--syncular-color-border)}@supports (color:color-mix(in lab,red,red)){.border-border\/50{border-color:color-mix(in oklab,var(--syncular-color-border) 50%,transparent)}}.border-encrypt\/15{border-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.border-encrypt\/15{border-color:color-mix(in oklab,var(--syncular-color-encrypt) 15%,transparent)}}.border-encrypt\/20{border-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.border-encrypt\/20{border-color:color-mix(in oklab,var(--syncular-color-encrypt) 20%,transparent)}}.border-encrypt\/30{border-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.border-encrypt\/30{border-color:color-mix(in oklab,var(--syncular-color-encrypt) 30%,transparent)}}.border-flow,.border-flow\/20{border-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.border-flow\/20{border-color:color-mix(in oklab,var(--syncular-color-flow) 20%,transparent)}}.border-flow\/30{border-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.border-flow\/30{border-color:color-mix(in oklab,var(--syncular-color-flow) 30%,transparent)}}.border-flow\/40{border-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.border-flow\/40{border-color:color-mix(in oklab,var(--syncular-color-flow) 40%,transparent)}}.border-healthy,.border-healthy\/20{border-color:var(--syncular-color-healthy)}@supports (color:color-mix(in lab,red,red)){.border-healthy\/20{border-color:color-mix(in oklab,var(--syncular-color-healthy) 20%,transparent)}}.border-neutral-700{border-color:var(--color-neutral-700)}.border-offline,.border-offline\/20{border-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.border-offline\/20{border-color:color-mix(in oklab,var(--syncular-color-offline) 20%,transparent)}}.border-offline\/25{border-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.border-offline\/25{border-color:color-mix(in oklab,var(--syncular-color-offline) 25%,transparent)}}.border-relay\/20{border-color:var(--syncular-color-relay)}@supports (color:color-mix(in lab,red,red)){.border-relay\/20{border-color:color-mix(in oklab,var(--syncular-color-relay) 20%,transparent)}}.border-syncing\/20{border-color:var(--syncular-color-syncing)}@supports (color:color-mix(in lab,red,red)){.border-syncing\/20{border-color:color-mix(in oklab,var(--syncular-color-syncing) 20%,transparent)}}.border-syncing\/30{border-color:var(--syncular-color-syncing)}@supports (color:color-mix(in lab,red,red)){.border-syncing\/30{border-color:color-mix(in oklab,var(--syncular-color-syncing) 30%,transparent)}}.border-transparent{border-color:#0000}.border-violet-400\/20{border-color:#a685ff33}@supports (color:color-mix(in lab,red,red)){.border-violet-400\/20{border-color:color-mix(in oklab,var(--color-violet-400) 20%,transparent)}}.bg-\[\#1e1e1e\]{background-color:#1e1e1e}.bg-\[--color-bg\]{background-color:--color-bg}.bg-\[radial-gradient\(ellipse_at_35\%_50\%\,rgba\(59\,130\,246\,0\.06\)_0\%\,transparent_50\%\)\,radial-gradient\(ellipse_at_68\%_50\%\,rgba\(139\,92\,246\,0\.04\)_0\%\,transparent_40\%\)\,\#0c0c0c\]{background-color:radial-gradient(ellipse at 35% 50%,#3b82f60f 0%,transparent 50%),radial-gradient(ellipse at 68% 50%,#8b5cf60a 0%,transparent 40%),#0c0c0c}.bg-amber-400{background-color:var(--color-amber-400)}.bg-background{background-color:var(--syncular-color-background)}.bg-black\/60{background-color:#0009}@supports (color:color-mix(in lab,red,red)){.bg-black\/60{background-color:color-mix(in oklab,var(--color-black) 60%,transparent)}}.bg-black\/70{background-color:#000000b3}@supports (color:color-mix(in lab,red,red)){.bg-black\/70{background-color:color-mix(in oklab,var(--color-black) 70%,transparent)}}.bg-border{background-color:var(--syncular-color-border)}.bg-border-bright{background-color:var(--syncular-color-border-bright)}.bg-encrypt,.bg-encrypt\/10{background-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.bg-encrypt\/10{background-color:color-mix(in oklab,var(--syncular-color-encrypt) 10%,transparent)}}.bg-encrypt\/15{background-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.bg-encrypt\/15{background-color:color-mix(in oklab,var(--syncular-color-encrypt) 15%,transparent)}}.bg-encrypt\/\[0\.05\]{background-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.bg-encrypt\/\[0\.05\]{background-color:color-mix(in oklab,var(--syncular-color-encrypt) 5%,transparent)}}.bg-encrypt\/\[0\.08\]{background-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.bg-encrypt\/\[0\.08\]{background-color:color-mix(in oklab,var(--syncular-color-encrypt) 8%,transparent)}}.bg-flow,.bg-flow\/10{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/10{background-color:color-mix(in oklab,var(--syncular-color-flow) 10%,transparent)}}.bg-flow\/15{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/15{background-color:color-mix(in oklab,var(--syncular-color-flow) 15%,transparent)}}.bg-flow\/40{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/40{background-color:color-mix(in oklab,var(--syncular-color-flow) 40%,transparent)}}.bg-flow\/60{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/60{background-color:color-mix(in oklab,var(--syncular-color-flow) 60%,transparent)}}.bg-flow\/70{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/70{background-color:color-mix(in oklab,var(--syncular-color-flow) 70%,transparent)}}.bg-flow\/\[0\.02\]{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/\[0\.02\]{background-color:color-mix(in oklab,var(--syncular-color-flow) 2%,transparent)}}.bg-flow\/\[0\.05\]{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/\[0\.05\]{background-color:color-mix(in oklab,var(--syncular-color-flow) 5%,transparent)}}.bg-flow\/\[0\.06\]{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/\[0\.06\]{background-color:color-mix(in oklab,var(--syncular-color-flow) 6%,transparent)}}.bg-flow\/\[0\.08\]{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.bg-flow\/\[0\.08\]{background-color:color-mix(in oklab,var(--syncular-color-flow) 8%,transparent)}}.bg-healthy,.bg-healthy\/10{background-color:var(--syncular-color-healthy)}@supports (color:color-mix(in lab,red,red)){.bg-healthy\/10{background-color:color-mix(in oklab,var(--syncular-color-healthy) 10%,transparent)}}.bg-healthy\/15{background-color:var(--syncular-color-healthy)}@supports (color:color-mix(in lab,red,red)){.bg-healthy\/15{background-color:color-mix(in oklab,var(--syncular-color-healthy) 15%,transparent)}}.bg-healthy\/60{background-color:var(--syncular-color-healthy)}@supports (color:color-mix(in lab,red,red)){.bg-healthy\/60{background-color:color-mix(in oklab,var(--syncular-color-healthy) 60%,transparent)}}.bg-healthy\/70{background-color:var(--syncular-color-healthy)}@supports (color:color-mix(in lab,red,red)){.bg-healthy\/70{background-color:color-mix(in oklab,var(--syncular-color-healthy) 70%,transparent)}}.bg-neutral-500{background-color:var(--color-neutral-500)}.bg-neutral-600{background-color:var(--color-neutral-600)}.bg-neutral-700{background-color:var(--color-neutral-700)}.bg-neutral-800{background-color:var(--color-neutral-800)}.bg-neutral-900{background-color:var(--color-neutral-900)}.bg-offline,.bg-offline\/5{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.bg-offline\/5{background-color:color-mix(in oklab,var(--syncular-color-offline) 5%,transparent)}}.bg-offline\/10{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.bg-offline\/10{background-color:color-mix(in oklab,var(--syncular-color-offline) 10%,transparent)}}.bg-offline\/\[0\.04\]{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.bg-offline\/\[0\.04\]{background-color:color-mix(in oklab,var(--syncular-color-offline) 4%,transparent)}}.bg-panel{background-color:var(--syncular-color-panel)}.bg-panel-alt{background-color:var(--syncular-color-panel-alt)}.bg-pink-400\/10{background-color:#fb64b61a}@supports (color:color-mix(in lab,red,red)){.bg-pink-400\/10{background-color:color-mix(in oklab,var(--color-pink-400) 10%,transparent)}}.bg-purple-600{background-color:var(--color-purple-600)}.bg-relay,.bg-relay\/10{background-color:var(--syncular-color-relay)}@supports (color:color-mix(in lab,red,red)){.bg-relay\/10{background-color:color-mix(in oklab,var(--syncular-color-relay) 10%,transparent)}}.bg-relay\/15{background-color:var(--syncular-color-relay)}@supports (color:color-mix(in lab,red,red)){.bg-relay\/15{background-color:color-mix(in oklab,var(--syncular-color-relay) 15%,transparent)}}.bg-surface,.bg-surface\/88{background-color:var(--syncular-color-surface)}@supports (color:color-mix(in lab,red,red)){.bg-surface\/88{background-color:color-mix(in oklab,var(--syncular-color-surface) 88%,transparent)}}.bg-surface\/92{background-color:var(--syncular-color-surface)}@supports (color:color-mix(in lab,red,red)){.bg-surface\/92{background-color:color-mix(in oklab,var(--syncular-color-surface) 92%,transparent)}}.bg-syncing,.bg-syncing\/10{background-color:var(--syncular-color-syncing)}@supports (color:color-mix(in lab,red,red)){.bg-syncing\/10{background-color:color-mix(in oklab,var(--syncular-color-syncing) 10%,transparent)}}.bg-syncing\/15{background-color:var(--syncular-color-syncing)}@supports (color:color-mix(in lab,red,red)){.bg-syncing\/15{background-color:color-mix(in oklab,var(--syncular-color-syncing) 15%,transparent)}}.bg-syncing\/\[0\.05\]{background-color:var(--syncular-color-syncing)}@supports (color:color-mix(in lab,red,red)){.bg-syncing\/\[0\.05\]{background-color:color-mix(in oklab,var(--syncular-color-syncing) 5%,transparent)}}.bg-transparent{background-color:#0000}.bg-violet-400\/10{background-color:#a685ff1a}@supports (color:color-mix(in lab,red,red)){.bg-violet-400\/10{background-color:color-mix(in oklab,var(--color-violet-400) 10%,transparent)}}.bg-white{background-color:var(--color-white)}.bg-white\/\[0\.01\]{background-color:#ffffff03}@supports (color:color-mix(in lab,red,red)){.bg-white\/\[0\.01\]{background-color:color-mix(in oklab,var(--color-white) 1%,transparent)}}.bg-white\/\[0\.03\]{background-color:#ffffff08}@supports (color:color-mix(in lab,red,red)){.bg-white\/\[0\.03\]{background-color:color-mix(in oklab,var(--color-white) 3%,transparent)}}.bg-white\/\[0\.05\]{background-color:#ffffff0d}@supports (color:color-mix(in lab,red,red)){.bg-white\/\[0\.05\]{background-color:color-mix(in oklab,var(--color-white) 5%,transparent)}}.bg-\[image\:radial-gradient\(circle_at_1px_1px\,\#1a1a1a_1px\,transparent_0\)\]{background-image:radial-gradient(circle at 1px 1px,#1a1a1a 1px,#0000 0)}.bg-\[linear-gradient\(90deg\,transparent\,rgba\(59\,130\,246\,0\.3\)\,transparent\)\]{background-image:linear-gradient(90deg,#0000,#3b82f64d,#0000)}.bg-\[linear-gradient\(to_bottom\,rgba\(17\,17\,17\,0\.8\)\,\#0c0c0c\)\]{background-image:linear-gradient(#111c,#0c0c0c)}.bg-\[size\:24px_24px\]{background-size:24px 24px}.fill-foreground{fill:var(--syncular-color-foreground)}.fill-neutral-500{fill:var(--color-neutral-500)}.object-cover{object-fit:cover}.p-0{padding:calc(var(--spacing) * 0)}.p-1\.5{padding:calc(var(--spacing) * 1.5)}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-5{padding:calc(var(--spacing) * 5)}.p-6{padding:calc(var(--spacing) * 6)}.p-8{padding:calc(var(--spacing) * 8)}.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-3\.5{padding-inline:calc(var(--spacing) * 3.5)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.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-4{padding-block:calc(var(--spacing) * 4)}.py-5{padding-block:calc(var(--spacing) * 5)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-8{padding-block:calc(var(--spacing) * 8)}.py-10{padding-block:calc(var(--spacing) * 10)}.py-12{padding-block:calc(var(--spacing) * 12)}.py-16{padding-block:calc(var(--spacing) * 16)}.py-24{padding-block:calc(var(--spacing) * 24)}.py-\[3px\]{padding-block:3px}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pt-6{padding-top:calc(var(--spacing) * 6)}.pt-16{padding-top:calc(var(--spacing) * 16)}.pt-\[42px\]{padding-top:42px}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pb-3{padding-bottom:calc(var(--spacing) * 3)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pb-5{padding-bottom:calc(var(--spacing) * 5)}.pb-6{padding-bottom:calc(var(--spacing) * 6)}.pb-\[32px\]{padding-bottom:32px}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.font-display{font-family:var(--syncular-font-display)}.font-mono{font-family:var(--syncular-font-mono)}.font-sans{font-family:var(--font-sans)}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-3xl{font-size:var(--text-3xl);line-height:var(--tw-leading,var(--text-3xl--line-height))}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--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-\[12\.5px\]{font-size:12.5px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.leading-5{--tw-leading:calc(var(--spacing) * 5);line-height:calc(var(--spacing) * 5)}.leading-6{--tw-leading:calc(var(--spacing) * 6);line-height:calc(var(--spacing) * 6)}.leading-7{--tw-leading:calc(var(--spacing) * 7);line-height:calc(var(--spacing) * 7)}.leading-\[22px\]{--tw-leading:22px;line-height:22px}.leading-\[26px\]{--tw-leading:26px;line-height:26px}.leading-none{--tw-leading:1;line-height:1}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.leading-tight{--tw-leading:var(--leading-tight);line-height:var(--leading-tight)}.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-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.5px\]{--tw-tracking:.5px;letter-spacing:.5px}.tracking-\[1\.5px\]{--tw-tracking:1.5px;letter-spacing:1.5px}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.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-all{word-break:break-all}.text-ellipsis{text-overflow:ellipsis}.whitespace-nowrap{white-space:nowrap}.text-amber-300{color:var(--color-amber-300)}.text-amber-300\/80{color:#ffd236cc}@supports (color:color-mix(in lab,red,red)){.text-amber-300\/80{color:color-mix(in oklab,var(--color-amber-300) 80%,transparent)}}.text-amber-400{color:var(--color-amber-400)}.text-amber-500{color:var(--color-amber-500)}.text-blue-400{color:var(--color-blue-400)}.text-emerald-400{color:var(--color-emerald-400)}.text-emerald-500{color:var(--color-emerald-500)}.text-encrypt,.text-encrypt\/30{color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.text-encrypt\/30{color:color-mix(in oklab,var(--syncular-color-encrypt) 30%,transparent)}}.text-flow{color:var(--syncular-color-flow)}.text-foreground{color:var(--syncular-color-foreground)}.text-foreground-muted{color:var(--syncular-color-foreground-muted)}.text-healthy{color:var(--syncular-color-healthy)}.text-neutral-100{color:var(--color-neutral-100)}.text-neutral-200{color:var(--color-neutral-200)}.text-neutral-300{color:var(--color-neutral-300)}.text-neutral-400{color:var(--color-neutral-400)}.text-neutral-500{color:var(--color-neutral-500)}.text-neutral-600{color:var(--color-neutral-600)}.text-neutral-700{color:var(--color-neutral-700)}.text-offline{color:var(--syncular-color-offline)}.text-pink-400{color:var(--color-pink-400)}.text-red-500{color:var(--color-red-500)}.text-relay{color:var(--syncular-color-relay)}.text-syncing{color:var(--syncular-color-syncing)}.text-violet-400{color:var(--color-violet-400)}.text-violet-400\/60{color:#a685ff99}@supports (color:color-mix(in lab,red,red)){.text-violet-400\/60{color:color-mix(in oklab,var(--color-violet-400) 60%,transparent)}}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.tabular-nums{--tw-numeric-spacing:tabular-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.line-through{text-decoration-line:line-through}.underline{text-decoration-line:underline}.decoration-border{-webkit-text-decoration-color:var(--syncular-color-border);text-decoration-color:var(--syncular-color-border)}.decoration-neutral-700{-webkit-text-decoration-color:var(--color-neutral-700);text-decoration-color:var(--color-neutral-700)}.underline-offset-4{text-underline-offset:4px}.opacity-0{opacity:0}.opacity-25{opacity:.25}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-60{opacity:.6}.opacity-75{opacity:.75}.shadow-\[0_0_6px_rgba\(34\,197\,94\,0\.3\)\]{--tw-shadow:0 0 6px var(--tw-shadow-color,#22c55e4d);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_8px_rgba\(59\,130\,246\,0\.5\)\]{--tw-shadow:0 0 8px var(--tw-shadow-color,#3b82f680);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)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a), 0 8px 10px -6px 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)}.ring{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.filter{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,)}.backdrop-blur-\[4px\]{--tw-backdrop-blur:blur(4px);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-lg{--tw-backdrop-blur:blur(var(--blur-lg));-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.transition{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,-webkit-backdrop-filter,backdrop-filter,display,content-visibility,overlay,pointer-events;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-\[border-color\]{transition-property:border-color;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.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-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-200{--tw-duration:.2s;transition-duration:.2s}.duration-800{--tw-duration:.8s;transition-duration:.8s}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}.outline-none{--tw-outline-style:none;outline-style:none}.\[-ms-overflow-style\:none\]{-ms-overflow-style:none}.\[scrollbar-width\:none\]{scrollbar-width:none}@media(hover:hover){.group-hover\:opacity-100:is(:where(.group):hover *){opacity:1}}.placeholder\:text-neutral-600::placeholder{color:var(--color-neutral-600)}.before\:absolute:before{content:var(--tw-content);position:absolute}.before\:top-\[20\%\]:before{content:var(--tw-content);top:20%}.before\:bottom-\[20\%\]:before{content:var(--tw-content);bottom:20%}.before\:left-0:before{content:var(--tw-content);left:calc(var(--spacing) * 0)}.before\:w-\[2px\]:before{content:var(--tw-content);width:2px}.before\:bg-transparent:before{content:var(--tw-content);background-color:#0000}.before\:transition-colors:before{content:var(--tw-content);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))}.last\:border-b-0:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}@media(hover:hover){.hover\:scale-\[1\.15\]:hover{scale:1.15}.hover\:border-\[\#282828\]:hover{border-color:#282828}.hover\:border-border-bright:hover{border-color:var(--syncular-color-border-bright)}.hover\:border-flow\/40:hover{border-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.hover\:border-flow\/40:hover{border-color:color-mix(in oklab,var(--syncular-color-flow) 40%,transparent)}}.hover\:border-neutral-500:hover{border-color:var(--color-neutral-500)}.hover\:border-offline:hover,.hover\:border-offline\/50:hover{border-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.hover\:border-offline\/50:hover{border-color:color-mix(in oklab,var(--syncular-color-offline) 50%,transparent)}}.hover\:bg-blue-600:hover{background-color:var(--color-blue-600)}.hover\:bg-encrypt\/\[0\.14\]:hover{background-color:var(--syncular-color-encrypt)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-encrypt\/\[0\.14\]:hover{background-color:color-mix(in oklab,var(--syncular-color-encrypt) 14%,transparent)}}.hover\:bg-flow\/10:hover{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-flow\/10:hover{background-color:color-mix(in oklab,var(--syncular-color-flow) 10%,transparent)}}.hover\:bg-flow\/20:hover{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-flow\/20:hover{background-color:color-mix(in oklab,var(--syncular-color-flow) 20%,transparent)}}.hover\:bg-flow\/\[0\.02\]:hover{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-flow\/\[0\.02\]:hover{background-color:color-mix(in oklab,var(--syncular-color-flow) 2%,transparent)}}.hover\:bg-flow\/\[0\.12\]:hover{background-color:var(--syncular-color-flow)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-flow\/\[0\.12\]:hover{background-color:color-mix(in oklab,var(--syncular-color-flow) 12%,transparent)}}.hover\:bg-offline\/20:hover{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-offline\/20:hover{background-color:color-mix(in oklab,var(--syncular-color-offline) 20%,transparent)}}.hover\:bg-offline\/\[0\.06\]:hover{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-offline\/\[0\.06\]:hover{background-color:color-mix(in oklab,var(--syncular-color-offline) 6%,transparent)}}.hover\:bg-offline\/\[0\.08\]:hover{background-color:var(--syncular-color-offline)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-offline\/\[0\.08\]:hover{background-color:color-mix(in oklab,var(--syncular-color-offline) 8%,transparent)}}.hover\:bg-purple-500:hover{background-color:var(--color-purple-500)}.hover\:bg-white\/\[0\.02\]:hover{background-color:#ffffff05}@supports (color:color-mix(in lab,red,red)){.hover\:bg-white\/\[0\.02\]:hover{background-color:color-mix(in oklab,var(--color-white) 2%,transparent)}}.hover\:bg-white\/\[0\.03\]:hover{background-color:#ffffff08}@supports (color:color-mix(in lab,red,red)){.hover\:bg-white\/\[0\.03\]:hover{background-color:color-mix(in oklab,var(--color-white) 3%,transparent)}}.hover\:bg-white\/\[0\.015\]:hover{background-color:#ffffff04}@supports (color:color-mix(in lab,red,red)){.hover\:bg-white\/\[0\.015\]:hover{background-color:color-mix(in oklab,var(--color-white) 1.5%,transparent)}}.hover\:bg-white\/\[0\.018\]:hover{background-color:#ffffff05}@supports (color:color-mix(in lab,red,red)){.hover\:bg-white\/\[0\.018\]:hover{background-color:color-mix(in oklab,var(--color-white) 1.8%,transparent)}}.hover\:text-flow:hover{color:var(--syncular-color-flow)}.hover\:text-neutral-300:hover{color:var(--color-neutral-300)}.hover\:text-offline:hover{color:var(--syncular-color-offline)}.hover\:text-white:hover{color:var(--color-white)}.hover\:decoration-flow:hover{-webkit-text-decoration-color:var(--syncular-color-flow);text-decoration-color:var(--syncular-color-flow)}.hover\:before\:bg-flow:hover:before{content:var(--tw-content);background-color:var(--syncular-color-flow)}}.focus\:border-flow:focus{border-color:var(--syncular-color-flow)}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-50:disabled{opacity:.5}.data-\[checked\]\:translate-x-4[data-checked]{--tw-translate-x:calc(var(--spacing) * 4);translate:var(--tw-translate-x) var(--tw-translate-y)}.data-\[checked\]\:border-healthy[data-checked]{border-color:var(--syncular-color-healthy)}.data-\[checked\]\:bg-healthy[data-checked]{background-color:var(--syncular-color-healthy)}.data-\[checked\]\:shadow-\[0_0_6px_rgba\(34\,197\,94\,0\.3\)\][data-checked]{--tw-shadow:0 0 6px var(--tw-shadow-color,#22c55e4d);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.data-\[highlighted\]\:bg-white\/\[0\.03\][data-highlighted]{background-color:#ffffff08}@supports (color:color-mix(in lab,red,red)){.data-\[highlighted\]\:bg-white\/\[0\.03\][data-highlighted]{background-color:color-mix(in oklab,var(--color-white) 3%,transparent)}}.data-\[highlighted\]\:text-white[data-highlighted]{color:var(--color-white)}.data-\[pressed\]\:border-border-bright[data-pressed]{border-color:var(--syncular-color-border-bright)}.data-\[pressed\]\:bg-white\/\[0\.03\][data-pressed]{background-color:#ffffff08}@supports (color:color-mix(in lab,red,red)){.data-\[pressed\]\:bg-white\/\[0\.03\][data-pressed]{background-color:color-mix(in oklab,var(--color-white) 3%,transparent)}}.data-\[pressed\]\:text-white[data-pressed]{color:var(--color-white)}.data-\[selected\]\:border-border-bright[data-selected]{border-color:var(--syncular-color-border-bright)}.data-\[selected\]\:bg-white\/\[0\.03\][data-selected]{background-color:#ffffff08}@supports (color:color-mix(in lab,red,red)){.data-\[selected\]\:bg-white\/\[0\.03\][data-selected]{background-color:color-mix(in oklab,var(--color-white) 3%,transparent)}}.data-\[selected\]\:text-white[data-selected]{color:var(--color-white)}@media(min-width:40rem){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}}@media(min-width:48rem){.md\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.md\:p-12{padding:calc(var(--spacing) * 12)}.md\:text-3xl{font-size:var(--text-3xl);line-height:var(--tw-leading,var(--text-3xl--line-height))}.md\:text-4xl{font-size:var(--text-4xl);line-height:var(--tw-leading,var(--text-4xl--line-height))}}@media(min-width:64rem){.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.lg\:grid-cols-\[320px_1fr_320px\]{grid-template-columns:320px 1fr 320px}}@media(min-width:80rem){.xl\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}.\[\&_\.recharts-cartesian-axis-tick_text\]\:fill-neutral-500 .recharts-cartesian-axis-tick text{fill:var(--color-neutral-500)}.\[\&_\.recharts-cartesian-grid_line\[stroke\=\'\#ccc\'\]\]\:stroke-border\/50 .recharts-cartesian-grid line[stroke="#ccc"]{stroke:var(--syncular-color-border)}@supports (color:color-mix(in lab,red,red)){.\[\&_\.recharts-cartesian-grid_line\[stroke\=\'\#ccc\'\]\]\:stroke-border\/50 .recharts-cartesian-grid line[stroke="#ccc"]{stroke:color-mix(in oklab,var(--syncular-color-border) 50%,transparent)}}.\[\&_\.recharts-curve\.recharts-tooltip-cursor\]\:stroke-border .recharts-curve.recharts-tooltip-cursor{stroke:var(--syncular-color-border)}.\[\&_\.recharts-dot\[stroke\=\'\#fff\'\]\]\:stroke-transparent .recharts-dot[stroke="#fff"]{stroke:#0000}.\[\&_\.recharts-layer\]\:outline-none .recharts-layer{--tw-outline-style:none;outline-style:none}.\[\&_\.recharts-polar-grid_\[stroke\=\'\#ccc\'\]\]\:stroke-border .recharts-polar-grid [stroke="#ccc"]{stroke:var(--syncular-color-border)}.\[\&_\.recharts-radial-bar-background-sector\]\:fill-panel .recharts-radial-bar-background-sector,.\[\&_\.recharts-rectangle\.recharts-tooltip-cursor\]\:fill-panel .recharts-rectangle.recharts-tooltip-cursor{fill:var(--syncular-color-panel)}.\[\&_\.recharts-reference-line_\[stroke\=\'\#ccc\'\]\]\:stroke-border .recharts-reference-line [stroke="#ccc"]{stroke:var(--syncular-color-border)}.\[\&_\.recharts-sector\]\:outline-none .recharts-sector{--tw-outline-style:none;outline-style:none}.\[\&_\.recharts-sector\[stroke\=\'\#fff\'\]\]\:stroke-transparent .recharts-sector[stroke="#fff"]{stroke:#0000}.\[\&_\.recharts-surface\]\:outline-none .recharts-surface{--tw-outline-style:none;outline-style:none}.\[\&\:\:-webkit-scrollbar\]\:hidden::-webkit-scrollbar{display:none}.\[\&\>svg\]\:h-2\.5>svg{height:calc(var(--spacing) * 2.5)}.\[\&\>svg\]\:w-2\.5>svg{width:calc(var(--spacing) * 2.5)}.\[\&\>svg\]\:text-neutral-500>svg{color:var(--color-neutral-500)}}:where(.syncular-console-root,.syncular-ui-root){--syncular-color-background:#0c0c0c;--syncular-color-surface:#0c0c0c;--syncular-color-panel:#111;--syncular-color-panel-alt:#0e0e0e;--syncular-color-border:#1e1e1e;--syncular-color-border-bright:#2a2a2a;--syncular-color-healthy:#22c55e;--syncular-color-syncing:#f59e0b;--syncular-color-offline:#ef4444;--syncular-color-flow:#3b82f6;--syncular-color-relay:#8b5cf6;--syncular-color-encrypt:#f472b6;--syncular-color-foreground:#e5e5e5;--syncular-color-foreground-muted:#737373;--syncular-color-primary:#22c55e;--syncular-color-destructive:#ef4444;--syncular-color-secondary:#1e1e1e;--syncular-font-display:"Inter Tight", system-ui, sans-serif;--syncular-font-mono:"JetBrains Mono", ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--syncular-radius-sm:4px;--syncular-radius-md:6px;--syncular-radius-lg:8px;--syncular-radius-xl:10px;--syncular-glow-healthy:0 0 6px #22c55e;--syncular-glow-syncing:0 0 6px #f59e0b;--syncular-glow-offline:0 0 6px #ef4444;--syncular-glow-flow:0 0 6px #3b82f6;--syncular-glow-relay:0 0 6px #8b5cf6;--syncular-glow-encrypt:0 0 6px #f472b6;--color-background:var(--syncular-color-background);--color-surface:var(--syncular-color-surface);--color-panel:var(--syncular-color-panel);--color-panel-alt:var(--syncular-color-panel-alt);--color-border:var(--syncular-color-border);--color-border-bright:var(--syncular-color-border-bright);--color-healthy:var(--syncular-color-healthy);--color-syncing:var(--syncular-color-syncing);--color-offline:var(--syncular-color-offline);--color-flow:var(--syncular-color-flow);--color-relay:var(--syncular-color-relay);--color-encrypt:var(--syncular-color-encrypt);--color-foreground:var(--syncular-color-foreground);--color-foreground-muted:var(--syncular-color-foreground-muted);--color-primary:var(--syncular-color-primary);--color-destructive:var(--syncular-color-destructive);--color-secondary:var(--syncular-color-secondary);--font-display:var(--syncular-font-display);--font-mono:var(--syncular-font-mono);--radius-sm:var(--syncular-radius-sm);--radius-md:var(--syncular-radius-md);--radius-lg:var(--syncular-radius-lg);--radius-xl:var(--syncular-radius-xl);--sync-glow-healthy:var(--syncular-glow-healthy);--sync-glow-syncing:var(--syncular-glow-syncing);--sync-glow-offline:var(--syncular-glow-offline);--sync-glow-flow:var(--syncular-glow-flow);--sync-glow-relay:var(--syncular-glow-relay);--sync-glow-encrypt:var(--syncular-glow-encrypt);background:var(--syncular-color-background);color:var(--syncular-color-foreground);font-family:var(--syncular-font-display),ui-sans-serif,system-ui,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;margin:0}:where(.syncular-console-root,.syncular-ui-root) .font-mono{font-family:var(--syncular-font-mono),ui-monospace,monospace}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar{width:3px;height:3px}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar-track{background:0 0}:where(.syncular-console-root,.syncular-ui-root) ::-webkit-scrollbar-thumb{background:#333;border-radius:2px}@keyframes dotPulse{0%,to{opacity:1}50%{opacity:.4}}@keyframes dashFlow{0%{stroke-dashoffset:20px}to{stroke-dashoffset:0}}@keyframes streamSlide{0%{opacity:0;transform:translateY(-6px)}15%{opacity:1;transform:translateY(0)}}@keyframes pageIn{0%{opacity:0;transform:translateY(6px)}to{opacity:1;transform:translateY(0)}}@keyframes scanSweep{0%{opacity:0;top:0}10%{opacity:.6}90%{opacity:.6}to{opacity:0;top:100%}}@keyframes heartbeat{0%,40%,to{transform:scaleY(1)}10%{transform:scaleY(2.5)}25%{transform:scaleY(1.5)}}@keyframes fadeUp{0%{opacity:0;transform:translateY(12px)}to{opacity:1;transform:translateY(0)}}:where(.syncular-console-root,.syncular-ui-root) .line-active{stroke-dasharray:4 2;animation:1s linear infinite dashFlow}:where(.syncular-console-root,.syncular-ui-root) .stream-entry{animation:.35s ease-out streamSlide}:where(.syncular-console-root,.syncular-ui-root) .dot-pulse{animation:1.4s ease-in-out infinite dotPulse}@keyframes flowPulse{0%{opacity:.15}50%{opacity:.6}to{opacity:.15}}:where(.syncular-console-root,.syncular-ui-root) .flow-arrow{animation:2s ease-in-out infinite flowPulse}@keyframes mediaLoad{0%{opacity:0;transform:scale(.92)}to{opacity:1;transform:scale(1)}}:where(.syncular-console-root,.syncular-ui-root) .dot-grid{background-image:radial-gradient(circle at 1px 1px,#1a1a1a 1px,#0000 0);background-size:24px 24px}:where(.syncular-console-root,.syncular-ui-root) .scan-line{pointer-events:none;background:linear-gradient(90deg,#0000,#3b82f64d,#0000);height:1px;animation:6s ease-in-out infinite scanSweep;position:absolute;left:0;right:0}:where(.syncular-console-root){color-scheme:dark}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value: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-ordinal{syntax:"*";inherits:false}@property --tw-slashed-zero{syntax:"*";inherits:false}@property --tw-numeric-figure{syntax:"*";inherits:false}@property --tw-numeric-spacing{syntax:"*";inherits:false}@property --tw-numeric-fraction{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-outline-style{syntax:"*";inherits:false;initial-value:solid}@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-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}@property --tw-content{syntax:"*";inherits:false;initial-value:""}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{50%{opacity:.5}}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@syncular/console",
3
- "version": "0.0.4-26",
3
+ "version": "0.0.5-42",
4
4
  "description": "Embeddable Syncular console UI",
5
5
  "license": "MIT",
6
6
  "author": "Benjamin Kniffler",
@@ -68,12 +68,12 @@
68
68
  "release": "bunx syncular-publish"
69
69
  },
70
70
  "dependencies": {
71
- "@syncular/observability-sentry": "0.0.4-26",
72
- "@syncular/transport-http": "0.0.4-26",
73
- "@syncular/ui": "0.0.4-26",
71
+ "@syncular/observability-sentry": "0.0.5-42",
72
+ "@syncular/transport-http": "0.0.5-42",
73
+ "@syncular/ui": "0.0.5-42",
74
74
  "@tanstack/react-query": "^5.90.21",
75
- "@tanstack/react-router": "^1.159.5",
76
- "lucide-react": "^0.563.0"
75
+ "@tanstack/react-router": "^1.161.3",
76
+ "lucide-react": "^0.575.0"
77
77
  },
78
78
  "peerDependencies": {
79
79
  "react": "^19.0.0",
@@ -81,13 +81,13 @@
81
81
  },
82
82
  "devDependencies": {
83
83
  "@syncular/config": "0.0.0",
84
- "@tailwindcss/vite": "^4.1.18",
84
+ "@tailwindcss/vite": "^4.2.0",
85
85
  "@types/react": "^19",
86
86
  "@types/react-dom": "^19",
87
87
  "@vitejs/plugin-react": "^5.1.4",
88
88
  "react": "^19.2.4",
89
89
  "react-dom": "^19.2.4",
90
- "tailwindcss": "^4.1.18",
90
+ "tailwindcss": "^4.2.0",
91
91
  "vite": "^7.3.1"
92
92
  },
93
93
  "files": [
@@ -113,24 +113,14 @@ export function ConnectionProvider({
113
113
 
114
114
  try {
115
115
  const client = createConsoleClient(normalizedConfig);
116
- const ok = await testConnection(client);
117
-
118
- if (ok) {
119
- setState({
120
- isConnected: true,
121
- isConnecting: false,
122
- client,
123
- error: null,
124
- });
125
- return true;
126
- }
116
+ await testConnection(client);
127
117
  setState({
128
- isConnected: false,
118
+ isConnected: true,
129
119
  isConnecting: false,
130
- client: null,
131
- error: 'Failed to connect',
120
+ client,
121
+ error: null,
132
122
  });
133
- return false;
123
+ return true;
134
124
  } catch (err) {
135
125
  setState({
136
126
  isConnected: false,
@@ -7,6 +7,7 @@ import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
7
7
  import type {
8
8
  ConsoleApiKey,
9
9
  ConsoleApiKeyBulkRevokeResponse,
10
+ ConsoleBlobListResponse,
10
11
  ConsoleClient,
11
12
  ConsoleCommitDetail,
12
13
  ConsoleCommitListItem,
@@ -104,6 +105,8 @@ const queryKeys = {
104
105
  expiresWithinDays?: number;
105
106
  instanceId?: string;
106
107
  }) => ['console', 'api-keys', params] as const,
108
+ storage: (params?: Record<string, unknown>) =>
109
+ ['console', 'storage', params] as const,
107
110
  };
108
111
 
109
112
  function resolveRefetchInterval(
@@ -924,3 +927,103 @@ export function useStageRotateApiKeyMutation() {
924
927
  },
925
928
  });
926
929
  }
930
+
931
+ // ---------------------------------------------------------------------------
932
+ // Blob storage hooks
933
+ // ---------------------------------------------------------------------------
934
+
935
+ export function useBlobs(
936
+ options: {
937
+ prefix?: string;
938
+ cursor?: string;
939
+ limit?: number;
940
+ refetchIntervalMs?: number;
941
+ } = {}
942
+ ) {
943
+ const { config: connectionConfig } = useConnection();
944
+ return useQuery<ConsoleBlobListResponse>({
945
+ queryKey: queryKeys.storage({
946
+ prefix: options.prefix,
947
+ cursor: options.cursor,
948
+ limit: options.limit,
949
+ }),
950
+ queryFn: async () => {
951
+ if (!connectionConfig) throw new Error('Not connected');
952
+ const queryString = new URLSearchParams();
953
+ if (options.prefix) queryString.set('prefix', options.prefix);
954
+ if (options.cursor) queryString.set('cursor', options.cursor);
955
+ if (options.limit) queryString.set('limit', String(options.limit));
956
+ const response = await fetch(
957
+ buildConsoleUrl(
958
+ connectionConfig.serverUrl,
959
+ '/console/storage',
960
+ queryString
961
+ ),
962
+ {
963
+ method: 'GET',
964
+ headers: { Authorization: `Bearer ${connectionConfig.token}` },
965
+ }
966
+ );
967
+ if (!response.ok) throw new Error('Failed to list blobs');
968
+ return response.json();
969
+ },
970
+ enabled: !!connectionConfig,
971
+ refetchInterval: resolveRefetchInterval(options.refetchIntervalMs, 30000),
972
+ });
973
+ }
974
+
975
+ export function useDeleteBlobMutation() {
976
+ const { config: connectionConfig } = useConnection();
977
+ const queryClient = useQueryClient();
978
+ return useMutation<{ deleted: boolean }, Error, string>({
979
+ mutationFn: async (key: string) => {
980
+ if (!connectionConfig) throw new Error('Not connected');
981
+ const encodedKey = encodeURIComponent(key);
982
+ const response = await fetch(
983
+ buildConsoleUrl(
984
+ connectionConfig.serverUrl,
985
+ `/console/storage/${encodedKey}`,
986
+ new URLSearchParams()
987
+ ),
988
+ {
989
+ method: 'DELETE',
990
+ headers: { Authorization: `Bearer ${connectionConfig.token}` },
991
+ }
992
+ );
993
+ if (!response.ok) throw new Error('Failed to delete blob');
994
+ return response.json();
995
+ },
996
+ onSuccess: () => {
997
+ queryClient.invalidateQueries({ queryKey: ['console', 'storage'] });
998
+ },
999
+ });
1000
+ }
1001
+
1002
+ export function useBlobDownload() {
1003
+ const { config: connectionConfig } = useConnection();
1004
+ return async (key: string) => {
1005
+ if (!connectionConfig) throw new Error('Not connected');
1006
+ const encodedKey = encodeURIComponent(key);
1007
+ const response = await fetch(
1008
+ buildConsoleUrl(
1009
+ connectionConfig.serverUrl,
1010
+ `/console/storage/${encodedKey}/download`,
1011
+ new URLSearchParams()
1012
+ ),
1013
+ {
1014
+ method: 'GET',
1015
+ headers: { Authorization: `Bearer ${connectionConfig.token}` },
1016
+ }
1017
+ );
1018
+ if (!response.ok) throw new Error('Failed to download blob');
1019
+ const blob = await response.blob();
1020
+ const url = URL.createObjectURL(blob);
1021
+ const a = document.createElement('a');
1022
+ a.href = url;
1023
+ a.download = key.split('/').pop() || key;
1024
+ document.body.appendChild(a);
1025
+ a.click();
1026
+ document.body.removeChild(a);
1027
+ URL.revokeObjectURL(url);
1028
+ };
1029
+ }
@@ -3,7 +3,7 @@
3
3
  */
4
4
 
5
5
  import { useCallback, useEffect, useRef, useState } from 'react';
6
- import type { LiveEvent } from '../lib/types';
6
+ import type { ConsoleRequestEvent, LiveEvent } from '../lib/types';
7
7
  import { useConnection } from './ConnectionContext';
8
8
  import { useInstanceContext } from './useInstanceContext';
9
9
 
@@ -35,6 +35,11 @@ interface UseLiveEventsResult {
35
35
  clearEvents: () => void;
36
36
  }
37
37
 
38
+ function isServiceWorkerServerMode(): boolean {
39
+ if (typeof window === 'undefined') return false;
40
+ return new URLSearchParams(window.location.search).get('swServer') === '1';
41
+ }
42
+
38
43
  export function useLiveEvents(
39
44
  options: UseLiveEventsOptions = {}
40
45
  ): UseLiveEventsResult {
@@ -65,10 +70,12 @@ export function useLiveEvents(
65
70
  const reconnectAttemptsRef = useRef(0);
66
71
  const lastActivityAtRef = useRef(0);
67
72
  const lastEventTimestampRef = useRef<string | null>(null);
73
+ const lastEventIdRef = useRef<number | null>(null);
68
74
 
69
75
  const clearEvents = useCallback(() => {
70
76
  setEvents([]);
71
77
  lastEventTimestampRef.current = null;
78
+ lastEventIdRef.current = null;
72
79
  }, []);
73
80
 
74
81
  useEffect(() => {
@@ -80,6 +87,113 @@ export function useLiveEvents(
80
87
  const normalizedReplayLimit = Number.isFinite(replayLimit)
81
88
  ? Math.max(1, Math.min(500, Math.floor(replayLimit)))
82
89
  : 100;
90
+ const usePollingFallback =
91
+ isServiceWorkerServerMode() || typeof WebSocket === 'undefined';
92
+
93
+ if (usePollingFallback) {
94
+ setConnectionState('connecting');
95
+ setIsConnected(false);
96
+
97
+ let pollTimer: ReturnType<typeof setInterval> | null = null;
98
+ let isPolling = false;
99
+
100
+ const poll = async () => {
101
+ if (isCleanedUp || isPolling) return;
102
+ isPolling = true;
103
+
104
+ try {
105
+ const baseUrl = new URL(config.serverUrl, window.location.origin);
106
+ const normalizedPath = baseUrl.pathname.endsWith('/')
107
+ ? baseUrl.pathname.slice(0, -1)
108
+ : baseUrl.pathname;
109
+ baseUrl.pathname = `${normalizedPath}/console/events`;
110
+ baseUrl.search = '';
111
+ baseUrl.searchParams.set('limit', String(normalizedReplayLimit));
112
+ baseUrl.searchParams.set('offset', '0');
113
+ if (partitionId) {
114
+ baseUrl.searchParams.set('partitionId', partitionId);
115
+ }
116
+
117
+ const response = await fetch(baseUrl.toString(), {
118
+ headers: {
119
+ Authorization: `Bearer ${config.token}`,
120
+ },
121
+ });
122
+
123
+ if (!response.ok) {
124
+ throw new Error(`Live event polling failed (${response.status})`);
125
+ }
126
+
127
+ const payload = (await response.json()) as {
128
+ items?: ConsoleRequestEvent[];
129
+ };
130
+ const rows = Array.isArray(payload.items) ? payload.items : [];
131
+ const filtered = rows
132
+ .filter((row) =>
133
+ effectiveInstanceId
134
+ ? row.instanceId === effectiveInstanceId
135
+ : true
136
+ )
137
+ .sort((a, b) => a.eventId - b.eventId);
138
+
139
+ const previousLastId = lastEventIdRef.current ?? -1;
140
+ const newRows = filtered.filter(
141
+ (row) => row.eventId > previousLastId
142
+ );
143
+ if (newRows.length > 0) {
144
+ const mapped: LiveEvent[] = newRows
145
+ .map((row) => ({
146
+ type: row.eventType,
147
+ timestamp: row.createdAt,
148
+ data: row as unknown as Record<string, unknown>,
149
+ }))
150
+ .reverse();
151
+
152
+ setEvents((prev) => [...mapped, ...prev].slice(0, maxEvents));
153
+ const newest = newRows[newRows.length - 1]!;
154
+ lastEventIdRef.current = newest.eventId;
155
+ lastEventTimestampRef.current = newest.createdAt;
156
+ } else if (filtered.length > 0) {
157
+ const newest = filtered[filtered.length - 1]!;
158
+ lastEventIdRef.current = Math.max(
159
+ lastEventIdRef.current ?? -1,
160
+ newest.eventId
161
+ );
162
+ lastEventTimestampRef.current = newest.createdAt;
163
+ }
164
+
165
+ setError(null);
166
+ setIsConnected(true);
167
+ setConnectionState('connected');
168
+ } catch (err) {
169
+ if (!isCleanedUp) {
170
+ setIsConnected(false);
171
+ setConnectionState('disconnected');
172
+ setError(
173
+ err instanceof Error
174
+ ? err
175
+ : new Error('Live event polling failed')
176
+ );
177
+ }
178
+ } finally {
179
+ isPolling = false;
180
+ }
181
+ };
182
+
183
+ void poll();
184
+ pollTimer = setInterval(() => {
185
+ void poll();
186
+ }, 2_000);
187
+
188
+ return () => {
189
+ isCleanedUp = true;
190
+ if (pollTimer) {
191
+ clearInterval(pollTimer);
192
+ }
193
+ setIsConnected(false);
194
+ setConnectionState('disconnected');
195
+ };
196
+ }
83
197
 
84
198
  const clearReconnectTimeout = () => {
85
199
  if (!reconnectTimeoutRef.current) return;
package/src/layout.tsx CHANGED
@@ -22,7 +22,13 @@ interface ConsoleLayoutProps {
22
22
  basePath?: string;
23
23
  }
24
24
 
25
- type ConsoleNavSuffix = '' | '/stream' | '/fleet' | '/ops' | '/config';
25
+ type ConsoleNavSuffix =
26
+ | ''
27
+ | '/stream'
28
+ | '/fleet'
29
+ | '/ops'
30
+ | '/storage'
31
+ | '/config';
26
32
 
27
33
  interface ConsoleNavItem {
28
34
  suffix: ConsoleNavSuffix;
@@ -34,6 +40,7 @@ const NAV_ITEMS: ConsoleNavItem[] = [
34
40
  { suffix: '/stream', label: 'Stream' },
35
41
  { suffix: '/fleet', label: 'Fleet' },
36
42
  { suffix: '/ops', label: 'Ops' },
43
+ { suffix: '/storage', label: 'Storage' },
37
44
  { suffix: '/config', label: 'Config' },
38
45
  ];
39
46
 
package/src/lib/api.ts CHANGED
@@ -16,11 +16,44 @@ export function createConsoleClient(config: ConnectionConfig): SyncClient {
16
16
  });
17
17
  }
18
18
 
19
- export async function testConnection(client: SyncClient): Promise<boolean> {
19
+ export async function testConnection(client: SyncClient): Promise<void> {
20
20
  try {
21
- const { error } = await client.GET('/console/stats');
22
- return !error;
23
- } catch {
24
- return false;
21
+ const { error, response } = await client.GET('/console/stats');
22
+ if (!error) return;
23
+
24
+ const statusCode = response.status;
25
+ let detail: string | null = null;
26
+ if (typeof error === 'string') {
27
+ detail = error;
28
+ } else if (error && typeof error === 'object') {
29
+ const errorRecord = error as Record<string, unknown>;
30
+ const nestedError = errorRecord.error;
31
+ const nestedMessage = errorRecord.message;
32
+ if (typeof nestedError === 'string' && nestedError.length > 0) {
33
+ detail = nestedError;
34
+ } else if (
35
+ typeof nestedMessage === 'string' &&
36
+ nestedMessage.length > 0
37
+ ) {
38
+ detail = nestedMessage;
39
+ } else {
40
+ try {
41
+ detail = JSON.stringify(errorRecord);
42
+ } catch {
43
+ detail = null;
44
+ }
45
+ }
46
+ }
47
+
48
+ throw new Error(
49
+ detail && detail.length > 0
50
+ ? `Console API /console/stats returned ${statusCode}: ${detail}`
51
+ : `Console API /console/stats returned ${statusCode}`
52
+ );
53
+ } catch (error) {
54
+ if (error instanceof Error) {
55
+ throw error;
56
+ }
57
+ throw new Error('Failed to connect to console API');
25
58
  }
26
59
  }
package/src/lib/types.ts CHANGED
@@ -226,3 +226,20 @@ export interface LiveEvent {
226
226
  timestamp: string;
227
227
  data: Record<string, unknown>;
228
228
  }
229
+
230
+ // ---------------------------------------------------------------------------
231
+ // Blob storage
232
+ // ---------------------------------------------------------------------------
233
+
234
+ export interface ConsoleBlob {
235
+ key: string;
236
+ size: number;
237
+ uploaded: string;
238
+ httpMetadata?: { contentType?: string };
239
+ }
240
+
241
+ export interface ConsoleBlobListResponse {
242
+ items: ConsoleBlob[];
243
+ truncated: boolean;
244
+ cursor: string | null;
245
+ }
@@ -48,12 +48,13 @@ import type {
48
48
  ConsoleApiKeyBulkRevokeResponse,
49
49
  } from '../lib/types';
50
50
 
51
- export function Config() {
51
+ export function Config({ children }: { children?: import('react').ReactNode }) {
52
52
  return (
53
53
  <div className="space-y-4 px-5 py-5">
54
54
  <ConnectionTab />
55
55
  <ApiKeysTab />
56
56
  <PreferencesTab />
57
+ {children}
57
58
  </div>
58
59
  );
59
60
  }