khal-os 1.260324.2

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 (408) hide show
  1. package/.env.example +23 -0
  2. package/.genie/mailbox/cli-sent.jsonl +3 -0
  3. package/.genie/mailbox/ds1-wave2-engineer-1.json +15 -0
  4. package/.genie/mailbox/ds1-wave2-engineer-2.json +15 -0
  5. package/.genie/mailbox/ds1-wave2-engineer-3.json +15 -0
  6. package/.genie/state/os-observability.json +39 -0
  7. package/.genie/state/tmux-control-mode-terminal.json +28 -0
  8. package/.genie/wishes/genieos-one-theme/WISH.md +417 -0
  9. package/.genie/wishes/workos-prod-rbac/WISH.md +345 -0
  10. package/.github/workflows/ci.yml +39 -0
  11. package/.github/workflows/release.yml +78 -0
  12. package/.github/workflows/version.yml +122 -0
  13. package/.husky/pre-commit +1 -0
  14. package/.pnpm-approve-builds.json +1 -0
  15. package/CLAUDE.md +117 -0
  16. package/LICENSE +21 -0
  17. package/README.md +38 -0
  18. package/biome.json +124 -0
  19. package/bun.lock +1249 -0
  20. package/docs/workos-setup.md +116 -0
  21. package/ecosystem.config.cjs +26 -0
  22. package/instrumentation.ts +8 -0
  23. package/knip.json +35 -0
  24. package/nats.conf +7 -0
  25. package/next.config.ts +25 -0
  26. package/package.json +78 -0
  27. package/packages/dev3000-app/components.ts +12 -0
  28. package/packages/dev3000-app/manifest.ts +19 -0
  29. package/packages/dev3000-app/package.json +23 -0
  30. package/packages/dev3000-app/views/dev3000/Dev3000App.tsx +758 -0
  31. package/packages/dev3000-app/views/dev3000/ErrorsPanel.tsx +160 -0
  32. package/packages/dev3000-app/views/dev3000/dev3000-context.tsx +21 -0
  33. package/packages/dev3000-app/views/dev3000/index.ts +4 -0
  34. package/packages/dev3000-app/views/dev3000/schema.ts +55 -0
  35. package/packages/dev3000-app/views/dev3000/service/index.ts +358 -0
  36. package/packages/dev3000-app/views/dev3000/service/runtime +1 -0
  37. package/packages/dev3000-app/views/dev3000/subjects.ts +9 -0
  38. package/packages/dev3000-app/views/dev3000/types.ts +77 -0
  39. package/packages/files-app/components.ts +12 -0
  40. package/packages/files-app/manifest.ts +19 -0
  41. package/packages/files-app/package.json +23 -0
  42. package/packages/files-app/views/files/ContextMenu.tsx +151 -0
  43. package/packages/files-app/views/files/DeleteConfirmDialog.tsx +39 -0
  44. package/packages/files-app/views/files/FileItem.tsx +128 -0
  45. package/packages/files-app/views/files/FilesApp.tsx +509 -0
  46. package/packages/files-app/views/files/FilesListView.tsx +201 -0
  47. package/packages/files-app/views/files/FilesToolbar.tsx +117 -0
  48. package/packages/files-app/views/files/GridView.tsx +90 -0
  49. package/packages/files-app/views/files/InlineInput.tsx +131 -0
  50. package/packages/files-app/views/files/UploadOverlay.tsx +61 -0
  51. package/packages/files-app/views/files/schema.ts +49 -0
  52. package/packages/files-app/views/files/service/index.ts +184 -0
  53. package/packages/files-app/views/files/service/runtime +1 -0
  54. package/packages/files-app/views/files/use-files.ts +201 -0
  55. package/packages/files-app/views/files/use-upload.ts +105 -0
  56. package/packages/genie-app/components.ts +12 -0
  57. package/packages/genie-app/lib/subjects.ts +87 -0
  58. package/packages/genie-app/manifest.ts +19 -0
  59. package/packages/genie-app/package.json +29 -0
  60. package/packages/genie-app/views/genie/service/agent-lifecycle.ts +136 -0
  61. package/packages/genie-app/views/genie/service/cli.ts +114 -0
  62. package/packages/genie-app/views/genie/service/comms.ts +141 -0
  63. package/packages/genie-app/views/genie/service/directory.ts +167 -0
  64. package/packages/genie-app/views/genie/service/index.ts +219 -0
  65. package/packages/genie-app/views/genie/service/system.ts +123 -0
  66. package/packages/genie-app/views/genie/service/teams.ts +191 -0
  67. package/packages/genie-app/views/genie/service/terminal-proxy.ts +184 -0
  68. package/packages/genie-app/views/genie/service/tmux-control.ts +318 -0
  69. package/packages/genie-app/views/genie/service/wishes.ts +270 -0
  70. package/packages/genie-app/views/genie/ui/GenieApp.tsx +5 -0
  71. package/packages/genie-app/views/genie/ui/PaneCard.tsx +307 -0
  72. package/packages/genie-app/views/genie/ui/Sidebar.tsx +212 -0
  73. package/packages/genie-app/views/genie/ui/TabBar.tsx +70 -0
  74. package/packages/genie-app/views/genie/ui/WorkspaceCanvas.tsx +343 -0
  75. package/packages/genie-app/views/genie/ui/XTermPane.tsx +306 -0
  76. package/packages/genie-app/views/genie/ui/hooks/useNatsAction.ts +54 -0
  77. package/packages/genie-app/views/genie/ui/hooks/useNatsRequest.ts +68 -0
  78. package/packages/genie-app/views/genie/ui/panels/AgentsPanel.tsx +399 -0
  79. package/packages/genie-app/views/genie/ui/panels/ChatPanel.tsx +351 -0
  80. package/packages/genie-app/views/genie/ui/panels/SystemPanel.tsx +195 -0
  81. package/packages/genie-app/views/genie/ui/panels/TeamsPanel.tsx +560 -0
  82. package/packages/genie-app/views/genie/ui/panels/WishesPanel.tsx +424 -0
  83. package/packages/nats-viewer-app/components.ts +12 -0
  84. package/packages/nats-viewer-app/manifest.ts +18 -0
  85. package/packages/nats-viewer-app/package.json +14 -0
  86. package/packages/nats-viewer-app/views/nats-viewer/ActiveSubs.tsx +34 -0
  87. package/packages/nats-viewer-app/views/nats-viewer/MessageLog.tsx +247 -0
  88. package/packages/nats-viewer-app/views/nats-viewer/NatsViewer.tsx +209 -0
  89. package/packages/nats-viewer-app/views/nats-viewer/PublishPanel.tsx +111 -0
  90. package/packages/nats-viewer-app/views/nats-viewer/RequestPanel.tsx +165 -0
  91. package/packages/nats-viewer-app/views/nats-viewer/Sidebar.tsx +59 -0
  92. package/packages/nats-viewer-app/views/nats-viewer/SubjectCatalog.tsx +63 -0
  93. package/packages/nats-viewer-app/views/nats-viewer/SubscribeInput.tsx +59 -0
  94. package/packages/nats-viewer-app/views/nats-viewer/index.ts +5 -0
  95. package/packages/nats-viewer-app/views/nats-viewer/nats-viewer-context.tsx +31 -0
  96. package/packages/nats-viewer-app/views/nats-viewer/types.ts +7 -0
  97. package/packages/nats-viewer-app/views/nats-viewer/use-message-buffer.ts +55 -0
  98. package/packages/os-cli/package.json +18 -0
  99. package/packages/os-cli/src/commands/events.ts +176 -0
  100. package/packages/os-cli/src/commands/logs.ts +96 -0
  101. package/packages/os-cli/src/commands/status.ts +53 -0
  102. package/packages/os-cli/src/commands/traces.ts +115 -0
  103. package/packages/os-cli/src/index.ts +15 -0
  104. package/packages/os-cli/src/lib/formatter.ts +123 -0
  105. package/packages/os-cli/src/lib/nats.ts +16 -0
  106. package/packages/os-cli/src/lib/trace-tree.ts +144 -0
  107. package/packages/os-cli/tsconfig.json +12 -0
  108. package/packages/os-sdk/package.json +27 -0
  109. package/packages/os-sdk/src/api/handler.ts +67 -0
  110. package/packages/os-sdk/src/config.ts +68 -0
  111. package/packages/os-sdk/src/db/factory.test.ts +42 -0
  112. package/packages/os-sdk/src/db/factory.ts +72 -0
  113. package/packages/os-sdk/src/db/migrate.ts +140 -0
  114. package/packages/os-sdk/src/db/provision.ts +44 -0
  115. package/packages/os-sdk/src/index.ts +36 -0
  116. package/packages/os-sdk/src/service/console-intercept.ts +60 -0
  117. package/packages/os-sdk/src/service/logger.ts +88 -0
  118. package/packages/os-sdk/src/service/o11y-streams.ts +88 -0
  119. package/packages/os-sdk/src/service/runtime.ts +259 -0
  120. package/packages/os-sdk/src/service/trace.ts +71 -0
  121. package/packages/os-sdk/tsconfig.json +16 -0
  122. package/packages/os-ui/package.json +13 -0
  123. package/packages/os-ui/src/index.ts +29 -0
  124. package/packages/os-ui/src/server.ts +4 -0
  125. package/packages/os-ui/tsconfig.json +19 -0
  126. package/packages/settings-app/components.ts +12 -0
  127. package/packages/settings-app/manifest.ts +18 -0
  128. package/packages/settings-app/package.json +14 -0
  129. package/packages/settings-app/views/settings/Settings.tsx +492 -0
  130. package/packages/terminal-app/components.ts +12 -0
  131. package/packages/terminal-app/manifest.ts +20 -0
  132. package/packages/terminal-app/package.json +23 -0
  133. package/packages/terminal-app/views/terminal/schema.ts +82 -0
  134. package/packages/terminal-app/views/terminal/service/index.ts +133 -0
  135. package/packages/terminal-app/views/terminal/service/runtime +1 -0
  136. package/packages/terminal-app/views/terminal/service/session.ts +290 -0
  137. package/packages/terminal-app/views/terminal/service/shell-hooks/bashrc-hook.sh +21 -0
  138. package/packages/terminal-app/views/terminal/types.ts +26 -0
  139. package/packages/terminal-app/views/terminal/ui/MultiTerminalApp.tsx +615 -0
  140. package/packages/terminal-app/views/terminal/ui/SplitDragHandle.tsx +91 -0
  141. package/packages/terminal-app/views/terminal/ui/SplitPaneRenderer.tsx +112 -0
  142. package/packages/terminal-app/views/terminal/ui/TerminalPane.tsx +478 -0
  143. package/packages/terminal-app/views/terminal/ui/TerminalTabBar.tsx +131 -0
  144. package/pnpm-workspace.yaml +9 -0
  145. package/postcss.config.mjs +7 -0
  146. package/public/file.svg +1 -0
  147. package/public/globe.svg +1 -0
  148. package/public/icons/code-server.svg +6 -0
  149. package/public/icons/default.svg +5 -0
  150. package/public/icons/dusk/1password.svg +1 -0
  151. package/public/icons/dusk/activity_monitor.svg +1 -0
  152. package/public/icons/dusk/app_store.svg +1 -0
  153. package/public/icons/dusk/atom.svg +1 -0
  154. package/public/icons/dusk/brave.svg +1 -0
  155. package/public/icons/dusk/calculator.svg +1 -0
  156. package/public/icons/dusk/calendar.svg +1 -0
  157. package/public/icons/dusk/chrome.svg +1 -0
  158. package/public/icons/dusk/chrome2.svg +1 -0
  159. package/public/icons/dusk/dashboard.svg +13 -0
  160. package/public/icons/dusk/discord.svg +1 -0
  161. package/public/icons/dusk/dropbox.svg +1 -0
  162. package/public/icons/dusk/electron.svg +1 -0
  163. package/public/icons/dusk/figma.svg +1 -0
  164. package/public/icons/dusk/finder.svg +1 -0
  165. package/public/icons/dusk/finder2.svg +1 -0
  166. package/public/icons/dusk/finder3.svg +1 -0
  167. package/public/icons/dusk/firefox.svg +1 -0
  168. package/public/icons/dusk/framer.svg +1 -0
  169. package/public/icons/dusk/gimp.svg +1 -0
  170. package/public/icons/dusk/github_desktop.svg +1 -0
  171. package/public/icons/dusk/hyper.svg +1 -0
  172. package/public/icons/dusk/hyper3.svg +1 -0
  173. package/public/icons/dusk/intellij.svg +1 -0
  174. package/public/icons/dusk/iterm2.svg +1 -0
  175. package/public/icons/dusk/itunes.svg +1 -0
  176. package/public/icons/dusk/mail.svg +1 -0
  177. package/public/icons/dusk/messenger.svg +1 -0
  178. package/public/icons/dusk/mongodb.svg +1 -0
  179. package/public/icons/dusk/notes.svg +1 -0
  180. package/public/icons/dusk/notion.svg +1 -0
  181. package/public/icons/dusk/obs.svg +1 -0
  182. package/public/icons/dusk/pages.svg +1 -0
  183. package/public/icons/dusk/photos.svg +1 -0
  184. package/public/icons/dusk/postman.svg +1 -0
  185. package/public/icons/dusk/preview.svg +1 -0
  186. package/public/icons/dusk/reminders.svg +1 -0
  187. package/public/icons/dusk/safari.svg +1 -0
  188. package/public/icons/dusk/sequel_pro.svg +1 -0
  189. package/public/icons/dusk/sketch.svg +1 -0
  190. package/public/icons/dusk/skype.svg +1 -0
  191. package/public/icons/dusk/slack.svg +1 -0
  192. package/public/icons/dusk/slack2.svg +1 -0
  193. package/public/icons/dusk/spotify.svg +1 -0
  194. package/public/icons/dusk/steam.svg +1 -0
  195. package/public/icons/dusk/system_preferences.svg +1 -0
  196. package/public/icons/dusk/tableplus.svg +1 -0
  197. package/public/icons/dusk/teams.svg +1 -0
  198. package/public/icons/dusk/telegram.svg +1 -0
  199. package/public/icons/dusk/terminal.svg +1 -0
  200. package/public/icons/dusk/todoist.svg +1 -0
  201. package/public/icons/dusk/trash.svg +1 -0
  202. package/public/icons/dusk/trello.svg +1 -0
  203. package/public/icons/dusk/vivaldi.svg +1 -0
  204. package/public/icons/dusk/vlc.svg +1 -0
  205. package/public/icons/dusk/vscode.svg +1 -0
  206. package/public/icons/dusk/whatsapp.svg +1 -0
  207. package/public/icons/dusk/xeyes.svg +1 -0
  208. package/public/icons/dusk/zoom.svg +1 -0
  209. package/public/icons/files.svg +5 -0
  210. package/public/icons/pwa/icon-192.png +0 -0
  211. package/public/icons/pwa/icon-512.png +0 -0
  212. package/public/icons/settings.svg +14 -0
  213. package/public/icons/terminal.svg +5 -0
  214. package/public/icons/text-editor.svg +7 -0
  215. package/public/manifest.json +38 -0
  216. package/public/next.svg +1 -0
  217. package/public/sw.js +41 -0
  218. package/public/vercel.svg +1 -0
  219. package/public/wallpapers/default.svg +10 -0
  220. package/public/window.svg +1 -0
  221. package/scripts/generate-pwa-icons.mjs +33 -0
  222. package/scripts/install-nats.sh +37 -0
  223. package/sentry.client.config.ts +21 -0
  224. package/sentry.edge.config.ts +12 -0
  225. package/sentry.server.config.ts +12 -0
  226. package/src/app/api/files/download/route.ts +81 -0
  227. package/src/app/api/files/download-zip/route.ts +102 -0
  228. package/src/app/api/files/upload/route.ts +58 -0
  229. package/src/app/api/webhooks/workos/route.ts +98 -0
  230. package/src/app/auth/callback/route.ts +16 -0
  231. package/src/app/auth/logout/route.ts +15 -0
  232. package/src/app/desktop/desktop-shell.tsx +110 -0
  233. package/src/app/desktop/layout.tsx +8 -0
  234. package/src/app/desktop/page.tsx +24 -0
  235. package/src/app/favicon.ico +0 -0
  236. package/src/app/globals.css +7 -0
  237. package/src/app/layout.tsx +64 -0
  238. package/src/app/offline/page.tsx +83 -0
  239. package/src/app/page.tsx +5 -0
  240. package/src/app/standalone/[appId]/page.tsx +28 -0
  241. package/src/app/standalone/layout.tsx +10 -0
  242. package/src/components/app-icon.tsx +55 -0
  243. package/src/components/apps/_echo/schema.ts +14 -0
  244. package/src/components/apps/_echo/service/index.ts +42 -0
  245. package/src/components/apps/app-manifest.ts +97 -0
  246. package/src/components/apps/app-registry.ts +55 -0
  247. package/src/components/apps/dev3000/Dev3000App.tsx +224 -0
  248. package/src/components/apps/dev3000/ErrorsPanel.tsx +160 -0
  249. package/src/components/apps/dev3000/Sidebar.tsx +41 -0
  250. package/src/components/apps/dev3000/TimelineLog.tsx +173 -0
  251. package/src/components/apps/dev3000/dev3000-context.tsx +29 -0
  252. package/src/components/apps/dev3000/index.ts +4 -0
  253. package/src/components/apps/dev3000/schema.ts +48 -0
  254. package/src/components/apps/dev3000/service/index.ts +520 -0
  255. package/src/components/apps/dev3000/service/runtime +1 -0
  256. package/src/components/apps/dev3000/types.ts +15 -0
  257. package/src/components/apps/dev3000/use-message-buffer.ts +46 -0
  258. package/src/components/apps/files/ContextMenu.tsx +151 -0
  259. package/src/components/apps/files/DeleteConfirmDialog.tsx +78 -0
  260. package/src/components/apps/files/FileItem.tsx +128 -0
  261. package/src/components/apps/files/FilesApp.tsx +509 -0
  262. package/src/components/apps/files/FilesListView.tsx +201 -0
  263. package/src/components/apps/files/FilesToolbar.tsx +117 -0
  264. package/src/components/apps/files/GridView.tsx +90 -0
  265. package/src/components/apps/files/InlineInput.tsx +131 -0
  266. package/src/components/apps/files/UploadOverlay.tsx +61 -0
  267. package/src/components/apps/files/schema.ts +49 -0
  268. package/src/components/apps/files/service/index.ts +227 -0
  269. package/src/components/apps/files/service/runtime +1 -0
  270. package/src/components/apps/files/use-files.ts +201 -0
  271. package/src/components/apps/files/use-upload.ts +105 -0
  272. package/src/components/apps/nats-viewer/ActiveSubs.tsx +34 -0
  273. package/src/components/apps/nats-viewer/MessageLog.tsx +247 -0
  274. package/src/components/apps/nats-viewer/NatsViewer.tsx +209 -0
  275. package/src/components/apps/nats-viewer/PublishPanel.tsx +113 -0
  276. package/src/components/apps/nats-viewer/RequestPanel.tsx +167 -0
  277. package/src/components/apps/nats-viewer/Sidebar.tsx +62 -0
  278. package/src/components/apps/nats-viewer/SubjectCatalog.tsx +64 -0
  279. package/src/components/apps/nats-viewer/SubscribeInput.tsx +59 -0
  280. package/src/components/apps/nats-viewer/index.ts +5 -0
  281. package/src/components/apps/nats-viewer/nats-viewer-context.tsx +31 -0
  282. package/src/components/apps/nats-viewer/types.ts +7 -0
  283. package/src/components/apps/nats-viewer/use-message-buffer.ts +55 -0
  284. package/src/components/apps/settings/Settings.tsx +492 -0
  285. package/src/components/apps/terminal/schema.ts +82 -0
  286. package/src/components/apps/terminal/service/index.ts +189 -0
  287. package/src/components/apps/terminal/service/runtime +1 -0
  288. package/src/components/apps/terminal/service/session.ts +296 -0
  289. package/src/components/apps/terminal/service/shell-hooks/bashrc-hook.sh +21 -0
  290. package/src/components/apps/terminal/types.ts +26 -0
  291. package/src/components/apps/terminal/ui/MultiTerminalApp.tsx +617 -0
  292. package/src/components/apps/terminal/ui/SplitDragHandle.tsx +91 -0
  293. package/src/components/apps/terminal/ui/SplitPaneRenderer.tsx +112 -0
  294. package/src/components/apps/terminal/ui/TerminalPane.tsx +476 -0
  295. package/src/components/apps/terminal/ui/TerminalTabBar.tsx +131 -0
  296. package/src/components/desktop/AnimatedBackground.tsx +69 -0
  297. package/src/components/desktop/Desktop.tsx +79 -0
  298. package/src/components/desktop/DesktopBackground.tsx +16 -0
  299. package/src/components/desktop/DesktopIcon.tsx +49 -0
  300. package/src/components/desktop/ShortcutViewer.tsx +136 -0
  301. package/src/components/desktop/WindowRenderer.tsx +34 -0
  302. package/src/components/desktop/WindowSwitcher.tsx +42 -0
  303. package/src/components/notifications/NotificationCenter.tsx +153 -0
  304. package/src/components/notifications/NotificationToasts.tsx +66 -0
  305. package/src/components/notifications/OrphanSessionToast.tsx +293 -0
  306. package/src/components/os-primitives/collapsible-sidebar.tsx +226 -0
  307. package/src/components/os-primitives/dialog.tsx +76 -0
  308. package/src/components/os-primitives/empty-state.tsx +43 -0
  309. package/src/components/os-primitives/index.ts +21 -0
  310. package/src/components/os-primitives/list-view.tsx +155 -0
  311. package/src/components/os-primitives/property-panel.tsx +108 -0
  312. package/src/components/os-primitives/section-header.tsx +19 -0
  313. package/src/components/os-primitives/sidebar-nav.tsx +110 -0
  314. package/src/components/os-primitives/split-pane.tsx +146 -0
  315. package/src/components/os-primitives/status-badge.tsx +10 -0
  316. package/src/components/os-primitives/status-bar.tsx +100 -0
  317. package/src/components/os-primitives/toolbar.tsx +152 -0
  318. package/src/components/taskbar/AppLauncher.tsx +114 -0
  319. package/src/components/taskbar/RunningApps.tsx +71 -0
  320. package/src/components/taskbar/SystemTray.tsx +134 -0
  321. package/src/components/taskbar/Taskbar.tsx +45 -0
  322. package/src/components/taskbar/UserMenu.tsx +138 -0
  323. package/src/components/taskbar/WorkspaceSwitcher.tsx +9 -0
  324. package/src/components/ui/ContextMenu.tsx +130 -0
  325. package/src/components/ui/badge.tsx +39 -0
  326. package/src/components/ui/button.tsx +102 -0
  327. package/src/components/ui/command.tsx +165 -0
  328. package/src/components/ui/dropdown-menu.tsx +233 -0
  329. package/src/components/ui/input.tsx +48 -0
  330. package/src/components/ui/note.tsx +55 -0
  331. package/src/components/ui/separator.tsx +25 -0
  332. package/src/components/ui/spinner.tsx +42 -0
  333. package/src/components/ui/switch.tsx +36 -0
  334. package/src/components/ui/theme-provider.tsx +24 -0
  335. package/src/components/ui/theme-switcher.tsx +51 -0
  336. package/src/components/ui/tooltip.tsx +62 -0
  337. package/src/components/window/MobileWindowStack.tsx +218 -0
  338. package/src/components/window/SnapPreview.tsx +37 -0
  339. package/src/components/window/StandaloneFrame.tsx +170 -0
  340. package/src/components/window/Window.tsx +423 -0
  341. package/src/components/window/WindowContent.tsx +14 -0
  342. package/src/components/window/WindowControlsOverlay.tsx +89 -0
  343. package/src/components/window/WindowFrame.tsx +124 -0
  344. package/src/lib/auth/index.ts +27 -0
  345. package/src/lib/auth/roles.ts +50 -0
  346. package/src/lib/auth/types.ts +32 -0
  347. package/src/lib/auth/use-auth.ts +53 -0
  348. package/src/lib/auth/webhook-handler.ts +87 -0
  349. package/src/lib/auth/workos.ts +67 -0
  350. package/src/lib/constants.ts +1 -0
  351. package/src/lib/desktop/dedup.ts +57 -0
  352. package/src/lib/desktop/schema.ts +55 -0
  353. package/src/lib/files/filename-validation.ts +41 -0
  354. package/src/lib/files/safe-path.ts +49 -0
  355. package/src/lib/hooks/use-desktop-nats.ts +438 -0
  356. package/src/lib/hooks/use-is-mobile.ts +23 -0
  357. package/src/lib/hooks/use-launch-app.ts +79 -0
  358. package/src/lib/hooks/use-nats-notifications.ts +84 -0
  359. package/src/lib/hooks/use-nats.ts +60 -0
  360. package/src/lib/hooks/use-visual-viewport.ts +72 -0
  361. package/src/lib/icons/resolve-window-icon.ts +10 -0
  362. package/src/lib/keyboard/defaults.ts +146 -0
  363. package/src/lib/keyboard/types.ts +52 -0
  364. package/src/lib/keyboard/use-global-keybinds.ts +231 -0
  365. package/src/lib/nats-client.ts +255 -0
  366. package/src/lib/nats.ts +35 -0
  367. package/src/lib/notifications/schema.ts +12 -0
  368. package/src/lib/service-loader.ts +171 -0
  369. package/src/lib/subjects.ts +64 -0
  370. package/src/lib/utils.ts +6 -0
  371. package/src/lib/ws-bridge.ts +288 -0
  372. package/src/lib/ws-protocol.ts +53 -0
  373. package/src/lib/ws-server.ts +167 -0
  374. package/src/middleware.ts +57 -0
  375. package/src/stores/desktop-store.ts +112 -0
  376. package/src/stores/keybind-store.ts +66 -0
  377. package/src/stores/notification-store.ts +271 -0
  378. package/src/stores/theme-store.ts +25 -0
  379. package/src/stores/window-store.ts +294 -0
  380. package/src/theme/animations.css +68 -0
  381. package/src/theme/base.css +123 -0
  382. package/src/theme/controls.css +35 -0
  383. package/src/theme/design-tokens.css +276 -0
  384. package/src/theme/index.css +23 -0
  385. package/src/theme/menus.css +45 -0
  386. package/src/theme/status.css +41 -0
  387. package/src/theme/surfaces.css +94 -0
  388. package/src/theme/tailwind-map.css +138 -0
  389. package/src/theme/taskbar.css +25 -0
  390. package/src/theme/terminal.css +55 -0
  391. package/src/theme/typography.css +26 -0
  392. package/src/theme/utilities.css +156 -0
  393. package/src/theme/window.css +103 -0
  394. package/src/types/desktop-entry.ts +12 -0
  395. package/src/types/use-descendants.d.ts +13 -0
  396. package/src/types/window.ts +28 -0
  397. package/src/types.d.ts +9 -0
  398. package/tauri/Cargo.lock +5464 -0
  399. package/tauri/Cargo.toml +19 -0
  400. package/tauri/build.rs +3 -0
  401. package/tauri/capabilities/default.json +36 -0
  402. package/tauri/icons/128x128.png +0 -0
  403. package/tauri/icons/128x128@2x.png +0 -0
  404. package/tauri/icons/32x32.png +0 -0
  405. package/tauri/icons/icon.png +0 -0
  406. package/tauri/src/main.rs +396 -0
  407. package/tauri/tauri.conf.json +23 -0
  408. package/tsconfig.json +43 -0
@@ -0,0 +1,255 @@
1
+ 'use client';
2
+
3
+ type MessageCallback = (data: unknown, subject: string) => void;
4
+ type StatusCallback = (connected: boolean) => void;
5
+
6
+ interface PendingRequest {
7
+ resolve: (data: unknown) => void;
8
+ reject: (error: Error) => void;
9
+ timer: ReturnType<typeof setTimeout>;
10
+ }
11
+
12
+ const DEFAULT_REQUEST_TIMEOUT_MS = 5000;
13
+ const MAX_RECONNECT_DELAY_MS = 30_000;
14
+
15
+ /**
16
+ * Match a concrete NATS subject against a subscription pattern.
17
+ * - `*` matches exactly one token
18
+ * - `>` (must be last token) matches one or more tokens
19
+ * - everything else is a literal match
20
+ */
21
+ function subjectMatches(pattern: string, subject: string): boolean {
22
+ if (pattern === subject) return true;
23
+
24
+ const patParts = pattern.split('.');
25
+ const subParts = subject.split('.');
26
+
27
+ for (let i = 0; i < patParts.length; i++) {
28
+ const p = patParts[i];
29
+ if (p === '>') return i === patParts.length - 1 && subParts.length > i;
30
+ if (i >= subParts.length) return false;
31
+ if (p !== '*' && p !== subParts[i]) return false;
32
+ }
33
+
34
+ return patParts.length === subParts.length;
35
+ }
36
+
37
+ class NatsClient {
38
+ private ws: WebSocket | null = null;
39
+ private _connected = false;
40
+
41
+ // subject -> Set of callbacks (refcounted)
42
+ private subscribers = new Map<string, Set<MessageCallback>>();
43
+
44
+ // correlation id -> pending request
45
+ private pendingRequests = new Map<string, PendingRequest>();
46
+
47
+ // status change listeners
48
+ private statusListeners = new Set<StatusCallback>();
49
+
50
+ // reconnect state
51
+ private reconnectAttempt = 0;
52
+ private reconnectTimer: ReturnType<typeof setTimeout> | null = null;
53
+ private intentionallyClosed = false;
54
+
55
+ // Identity scoping
56
+ private _orgId = '';
57
+ private _userId = '';
58
+
59
+ constructor() {
60
+ this.connect();
61
+ }
62
+
63
+ // --- Public API ---
64
+
65
+ get connected(): boolean {
66
+ return this._connected;
67
+ }
68
+
69
+ get orgId(): string {
70
+ return this._orgId;
71
+ }
72
+
73
+ get userId(): string {
74
+ return this._userId;
75
+ }
76
+
77
+ setOrgId(orgId: string): void {
78
+ this._orgId = orgId;
79
+ }
80
+
81
+ setUserId(userId: string): void {
82
+ this._userId = userId;
83
+ }
84
+
85
+ onStatusChange(callback: StatusCallback): () => void {
86
+ this.statusListeners.add(callback);
87
+ return () => {
88
+ this.statusListeners.delete(callback);
89
+ };
90
+ }
91
+
92
+ subscribe(subject: string, callback: MessageCallback): () => void {
93
+ let callbacks = this.subscribers.get(subject);
94
+ if (!callbacks) {
95
+ callbacks = new Set();
96
+ this.subscribers.set(subject, callbacks);
97
+ // First subscriber for this subject: send sub frame
98
+ this.send({ op: 'sub', subject });
99
+ }
100
+ callbacks.add(callback);
101
+
102
+ // Return unsubscribe function
103
+ return () => {
104
+ const cbs = this.subscribers.get(subject);
105
+ if (!cbs) return;
106
+ cbs.delete(callback);
107
+ if (cbs.size === 0) {
108
+ this.subscribers.delete(subject);
109
+ // Last subscriber removed: send unsub frame
110
+ this.send({ op: 'unsub', subject });
111
+ }
112
+ };
113
+ }
114
+
115
+ publish(subject: string, data?: unknown): void {
116
+ this.send({ op: 'pub', subject, data });
117
+ }
118
+
119
+ request(subject: string, data?: unknown, timeoutMs?: number): Promise<unknown> {
120
+ const id = crypto.randomUUID();
121
+ const timeout = timeoutMs ?? DEFAULT_REQUEST_TIMEOUT_MS;
122
+
123
+ return new Promise<unknown>((resolve, reject) => {
124
+ const timer = setTimeout(() => {
125
+ this.pendingRequests.delete(id);
126
+ reject(new Error(`request to "${subject}" timed out after ${timeout}ms`));
127
+ }, timeout);
128
+
129
+ this.pendingRequests.set(id, { resolve, reject, timer });
130
+ this.send({ op: 'req', id, subject, data });
131
+ });
132
+ }
133
+
134
+ // --- Connection management ---
135
+
136
+ private connect(): void {
137
+ if (typeof window === 'undefined') return;
138
+
139
+ const wsProtocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
140
+ const url = `${wsProtocol}//${window.location.host}/ws/nats`;
141
+
142
+ const ws = new WebSocket(url);
143
+ this.ws = ws;
144
+
145
+ ws.addEventListener('open', () => {
146
+ this.reconnectAttempt = 0;
147
+ this.setConnected(true);
148
+ // Re-subscribe all active subjects
149
+ for (const subject of this.subscribers.keys()) {
150
+ this.send({ op: 'sub', subject });
151
+ }
152
+ });
153
+
154
+ ws.addEventListener('message', (event) => {
155
+ this.handleMessage(event.data);
156
+ });
157
+
158
+ ws.addEventListener('close', () => {
159
+ this.setConnected(false);
160
+ this.ws = null;
161
+ if (!this.intentionallyClosed) {
162
+ this.scheduleReconnect();
163
+ }
164
+ });
165
+
166
+ ws.addEventListener('error', () => {
167
+ // The close event will fire after error, so reconnect is handled there.
168
+ });
169
+ }
170
+
171
+ private handleMessage(raw: string | ArrayBuffer | Blob): void {
172
+ if (typeof raw !== 'string') return;
173
+
174
+ let frame: Record<string, unknown>;
175
+ try {
176
+ frame = JSON.parse(raw);
177
+ } catch {
178
+ return;
179
+ }
180
+
181
+ if ('error' in frame) {
182
+ this.handleErrorFrame(frame);
183
+ } else if ('id' in frame && typeof frame.id === 'string') {
184
+ this.handleReplyFrame(frame.id, frame.data);
185
+ } else if ('subject' in frame && typeof frame.subject === 'string') {
186
+ this.handleMsgFrame(frame.subject, frame.data);
187
+ }
188
+ }
189
+
190
+ private handleErrorFrame(frame: Record<string, unknown>): void {
191
+ // biome-ignore lint/suspicious/noConsole: client-side debug logging for NATS errors
192
+ console.warn('[nats-client] server error:', frame.error);
193
+ }
194
+
195
+ private handleReplyFrame(id: string, data: unknown): void {
196
+ const pending = this.pendingRequests.get(id);
197
+ if (!pending) return;
198
+ clearTimeout(pending.timer);
199
+ this.pendingRequests.delete(id);
200
+ pending.resolve(data);
201
+ }
202
+
203
+ private handleMsgFrame(subject: string, data: unknown): void {
204
+ for (const [pattern, callbacks] of this.subscribers) {
205
+ if (!subjectMatches(pattern, subject)) continue;
206
+ for (const cb of callbacks) {
207
+ try {
208
+ cb(data, subject);
209
+ } catch {
210
+ // Subscriber callback threw; ignore to protect other subscribers.
211
+ }
212
+ }
213
+ }
214
+ }
215
+
216
+ private send(frame: Record<string, unknown>): void {
217
+ if (this.ws && this.ws.readyState === WebSocket.OPEN) {
218
+ this.ws.send(JSON.stringify(frame));
219
+ }
220
+ }
221
+
222
+ private setConnected(value: boolean): void {
223
+ if (this._connected === value) return;
224
+ this._connected = value;
225
+ for (const listener of this.statusListeners) {
226
+ try {
227
+ listener(value);
228
+ } catch {
229
+ // Protect against listener errors.
230
+ }
231
+ }
232
+ }
233
+
234
+ private scheduleReconnect(): void {
235
+ if (this.reconnectTimer) return;
236
+
237
+ const delay = Math.min(1000 * 2 ** this.reconnectAttempt, MAX_RECONNECT_DELAY_MS);
238
+ this.reconnectAttempt++;
239
+
240
+ this.reconnectTimer = setTimeout(() => {
241
+ this.reconnectTimer = null;
242
+ this.connect();
243
+ }, delay);
244
+ }
245
+ }
246
+
247
+ // Singleton
248
+ let instance: NatsClient | null = null;
249
+
250
+ export function getNatsClient(): NatsClient {
251
+ if (!instance) {
252
+ instance = new NatsClient();
253
+ }
254
+ return instance;
255
+ }
@@ -0,0 +1,35 @@
1
+ import { connect, type NatsConnection } from '@nats-io/transport-node';
2
+
3
+ let connection: NatsConnection | null = null;
4
+ let connecting: Promise<NatsConnection> | null = null;
5
+
6
+ export async function getNatsConnection(): Promise<NatsConnection> {
7
+ if (connection && !connection.isClosed()) {
8
+ return connection;
9
+ }
10
+
11
+ if (connecting) {
12
+ return connecting;
13
+ }
14
+
15
+ connecting = (async () => {
16
+ const nc = await connect({
17
+ servers: process.env.NATS_URL || 'nats://localhost:4222',
18
+ maxReconnectAttempts: -1,
19
+ reconnectTimeWait: 2000,
20
+ });
21
+
22
+ nc.closed().then((err) => {
23
+ connection = null;
24
+ if (err) {
25
+ console.error('[nats] connection closed with error:', err.message);
26
+ }
27
+ });
28
+
29
+ connection = nc;
30
+ connecting = null;
31
+ return nc;
32
+ })();
33
+
34
+ return connecting;
35
+ }
@@ -0,0 +1,12 @@
1
+ import { type Static, Type } from '@sinclair/typebox';
2
+
3
+ export const NatsNotification = Type.Object({
4
+ summary: Type.String(),
5
+ body: Type.Optional(Type.String()),
6
+ appName: Type.Optional(Type.String()),
7
+ icon: Type.Optional(Type.String()),
8
+ urgency: Type.Optional(Type.Union([Type.Literal('low'), Type.Literal('normal'), Type.Literal('critical')])),
9
+ category: Type.Optional(Type.String()),
10
+ transient: Type.Optional(Type.Boolean()),
11
+ });
12
+ export type NatsNotification = Static<typeof NatsNotification>;
@@ -0,0 +1,171 @@
1
+ // @ts-nocheck
2
+ import { existsSync, readdirSync, readFileSync } from 'node:fs';
3
+ import { relative, resolve } from 'node:path';
4
+ import { ensureO11yStreams } from '@genie-os/sdk/service/o11y-streams';
5
+ import { connect } from '@nats-io/transport-node';
6
+
7
+ const ROOT = resolve(import.meta.dir, '../components/apps');
8
+ const MAX_RETRIES = 5;
9
+ const RETRY_DELAY_MS = 2000;
10
+ const HEALTH_INTERVAL_MS = 30_000;
11
+
12
+ interface ManagedService {
13
+ name: string;
14
+ path: string;
15
+ runtime: 'bun' | 'node';
16
+ process: ReturnType<typeof Bun.spawn> | null;
17
+ retries: number;
18
+ running: boolean;
19
+ }
20
+
21
+ function discoverServices(): ManagedService[] {
22
+ const services: ManagedService[] = [];
23
+
24
+ if (!existsSync(ROOT)) {
25
+ console.warn('[service-loader] apps directory not found:', ROOT);
26
+ return services;
27
+ }
28
+
29
+ const entries = readdirSync(ROOT, { withFileTypes: true });
30
+ for (const entry of entries) {
31
+ if (!entry.isDirectory()) continue;
32
+ const servicePath = resolve(ROOT, entry.name, 'service', 'index.ts');
33
+ if (existsSync(servicePath)) {
34
+ // Check for runtime marker (e.g. service/runtime containing "node")
35
+ const runtimeFile = resolve(ROOT, entry.name, 'service', 'runtime');
36
+ let runtime: 'bun' | 'node' = 'bun';
37
+ if (existsSync(runtimeFile)) {
38
+ const content = readFileSync(runtimeFile, 'utf8').trim();
39
+ if (content === 'node') runtime = 'node';
40
+ }
41
+
42
+ services.push({
43
+ name: entry.name,
44
+ path: servicePath,
45
+ runtime,
46
+ process: null,
47
+ retries: 0,
48
+ running: false,
49
+ });
50
+ }
51
+ }
52
+
53
+ return services;
54
+ }
55
+
56
+ function spawnService(service: ManagedService): void {
57
+ const relPath = relative(process.cwd(), service.path);
58
+ const runtimeLabel = service.runtime === 'node' ? 'node/tsx' : 'bun';
59
+ console.log(`[service-loader] starting ${service.name} (${relPath}) [${runtimeLabel}]`);
60
+
61
+ // node-pty and other native addons may not work under Bun;
62
+ // services can opt into Node via a `service/runtime` file containing "node"
63
+ const cmd = service.runtime === 'node' ? ['npx', 'tsx', service.path] : ['bun', 'run', service.path];
64
+
65
+ const proc = Bun.spawn(cmd, {
66
+ stdout: 'inherit',
67
+ stderr: 'inherit',
68
+ env: { ...process.env },
69
+ });
70
+
71
+ service.process = proc;
72
+ service.running = true;
73
+
74
+ proc.exited.then((code) => {
75
+ service.running = false;
76
+ service.process = null;
77
+
78
+ if (code !== 0 && code !== null) {
79
+ console.warn(`[service-loader] ${service.name} exited with code ${code}`);
80
+ } else {
81
+ console.log(`[service-loader] ${service.name} exited`);
82
+ }
83
+
84
+ if (service.retries < MAX_RETRIES) {
85
+ service.retries++;
86
+ console.log(
87
+ `[service-loader] restarting ${service.name} (attempt ${service.retries}/${MAX_RETRIES}) in ${RETRY_DELAY_MS}ms`
88
+ );
89
+ setTimeout(() => spawnService(service), RETRY_DELAY_MS);
90
+ } else {
91
+ console.error(`[service-loader] ${service.name} exceeded max retries (${MAX_RETRIES}), giving up`);
92
+ }
93
+ });
94
+ }
95
+
96
+ async function publishHealth(services: ManagedService[]): Promise<void> {
97
+ try {
98
+ const nc = await connect({
99
+ servers: process.env.NATS_URL || 'nats://localhost:4222',
100
+ });
101
+
102
+ const status = services.map((s) => ({
103
+ name: s.name,
104
+ running: s.running,
105
+ retries: s.retries,
106
+ pid: s.process?.pid ?? null,
107
+ }));
108
+
109
+ const payload = JSON.stringify({
110
+ ts: new Date().toISOString(),
111
+ loader: 'service-loader',
112
+ services: status,
113
+ });
114
+
115
+ nc.publish('khal._internal.system.health', payload);
116
+ await nc.flush();
117
+ await nc.close();
118
+ } catch (err) {
119
+ console.error('[service-loader] failed to publish health:', err);
120
+ }
121
+ }
122
+
123
+ async function main(): Promise<void> {
124
+ const services = discoverServices();
125
+
126
+ if (services.length === 0) {
127
+ console.warn('[service-loader] no services discovered');
128
+ return;
129
+ }
130
+
131
+ console.log(`[service-loader] discovered ${services.length} service(s):`, services.map((s) => s.name).join(', '));
132
+
133
+ // Provision observability JetStream streams before spawning services
134
+ try {
135
+ const nc = await connect({ servers: process.env.NATS_URL || 'nats://localhost:4222' });
136
+ await ensureO11yStreams(nc);
137
+ await nc.close();
138
+ } catch (err) {
139
+ console.error('[service-loader] failed to provision o11y streams:', err);
140
+ }
141
+
142
+ for (const service of services) {
143
+ spawnService(service);
144
+ }
145
+
146
+ // Initial health publish after a short delay to let services start
147
+ setTimeout(() => publishHealth(services), 2000);
148
+
149
+ // Periodic health publishing
150
+ setInterval(() => publishHealth(services), HEALTH_INTERVAL_MS);
151
+
152
+ // Handle graceful shutdown
153
+ const shutdown = () => {
154
+ console.log('[service-loader] shutting down...');
155
+ for (const service of services) {
156
+ if (service.process) {
157
+ service.retries = MAX_RETRIES; // prevent restart
158
+ service.process.kill();
159
+ }
160
+ }
161
+ process.exit(0);
162
+ };
163
+
164
+ process.on('SIGINT', shutdown);
165
+ process.on('SIGTERM', shutdown);
166
+ }
167
+
168
+ main().catch((err) => {
169
+ console.error('[service-loader] fatal:', err);
170
+ process.exit(1);
171
+ });
@@ -0,0 +1,64 @@
1
+ export const SUBJECTS = {
2
+ echo: (orgId: string) => `khal.${orgId}.echo`,
3
+ system: {
4
+ health: (orgId: string) => `khal.${orgId}.system.health`,
5
+ },
6
+ pty: {
7
+ create: (orgId: string) => `khal.${orgId}.pty.create`,
8
+ destroy: (orgId: string) => `khal.${orgId}.pty.destroy`,
9
+ list: (orgId: string) => `khal.${orgId}.pty.list`,
10
+ data: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.data`,
11
+ input: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.input`,
12
+ resize: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.resize`,
13
+ exit: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.exit`,
14
+ replay: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.replay`,
15
+ buffer: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.buffer`,
16
+ bufferEnd: (orgId: string, sessionId: string) => `khal.${orgId}.pty.${sessionId}.buffer-end`,
17
+ },
18
+ fs: {
19
+ list: (orgId: string) => `khal.${orgId}.fs.list`,
20
+ read: (orgId: string) => `khal.${orgId}.fs.read`,
21
+ write: (orgId: string) => `khal.${orgId}.fs.write`,
22
+ search: (orgId: string) => `khal.${orgId}.fs.search`,
23
+ watch: (orgId: string, pathHash: string) => `khal.${orgId}.fs.watch.${pathHash}`,
24
+ },
25
+ notify: {
26
+ broadcast: (orgId: string) => `khal.${orgId}.notify.broadcast`,
27
+ user: (orgId: string, userId: string) => `khal.${orgId}.notify.user.${userId}`,
28
+ },
29
+ d3k: {
30
+ log: (orgId: string) => `khal.${orgId}.d3k.log`,
31
+ cmd: (orgId: string) => `khal.${orgId}.d3k.cmd`,
32
+ status: (orgId: string) => `khal.${orgId}.d3k.status`,
33
+ },
34
+ auth: {
35
+ roleChanged: 'os.auth.role-changed',
36
+ membershipRevoked: 'os.auth.membership-revoked',
37
+ },
38
+ desktop: {
39
+ cmd: {
40
+ open: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.open`,
41
+ close: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.close`,
42
+ focus: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.focus`,
43
+ minimize: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.minimize`,
44
+ maximize: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.maximize`,
45
+ restore: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.restore`,
46
+ notify: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.notify`,
47
+ sync: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.sync`,
48
+ all: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.cmd.>`,
49
+ },
50
+ event: {
51
+ opened: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.opened`,
52
+ closed: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.closed`,
53
+ focused: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.focused`,
54
+ minimized: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.minimized`,
55
+ maximized: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.maximized`,
56
+ restored: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.restored`,
57
+ state: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.state`,
58
+ metaUpdated: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.meta-updated`,
59
+ moved: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.moved`,
60
+ resized: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.resized`,
61
+ all: (orgId: string, userId: string) => `khal.${orgId}.desktop.${userId}.event.>`,
62
+ },
63
+ },
64
+ } as const;
@@ -0,0 +1,6 @@
1
+ import { type ClassValue, clsx } from 'clsx';
2
+ import { twMerge } from 'tailwind-merge';
3
+
4
+ export function cn(...inputs: ClassValue[]) {
5
+ return twMerge(clsx(inputs));
6
+ }