gitspace 0.2.0-rc.2 → 0.2.0-rc.21

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 (316) hide show
  1. package/README.md +68 -38
  2. package/package.json +36 -25
  3. package/.claude/settings.local.json +0 -21
  4. package/.gitspace/bundle.json +0 -50
  5. package/.gitspace/select/01-status.sh +0 -40
  6. package/.gitspace/setup/01-install-deps.sh +0 -12
  7. package/.gitspace/setup/02-typecheck.sh +0 -16
  8. package/AGENTS.md +0 -439
  9. package/CLAUDE.md +0 -1
  10. package/bun.lock +0 -647
  11. package/docs/CONNECTION.md +0 -623
  12. package/docs/GATEWAY-WORKER.md +0 -319
  13. package/docs/GETTING-STARTED.md +0 -448
  14. package/docs/GITSPACE-PLATFORM.md +0 -1819
  15. package/docs/INFRASTRUCTURE.md +0 -1347
  16. package/docs/PROTOCOL.md +0 -619
  17. package/docs/QUICKSTART.md +0 -174
  18. package/docs/RELAY.md +0 -327
  19. package/docs/REMOTE-DESIGN.md +0 -549
  20. package/docs/ROADMAP.md +0 -564
  21. package/docs/SITE_DOCS_FIGMA_MAKE.md +0 -1167
  22. package/docs/STACK-DESIGN.md +0 -588
  23. package/docs/UNIFIED_ARCHITECTURE.md +0 -292
  24. package/experiments/pty-benchmark.ts +0 -148
  25. package/experiments/pty-latency.ts +0 -100
  26. package/experiments/router/client.ts +0 -199
  27. package/experiments/router/protocol.ts +0 -74
  28. package/experiments/router/router.ts +0 -217
  29. package/experiments/router/session.ts +0 -180
  30. package/experiments/router/test.ts +0 -133
  31. package/experiments/socket-bandwidth.ts +0 -77
  32. package/homebrew/gitspace.rb +0 -45
  33. package/landing-page/ATTRIBUTIONS.md +0 -3
  34. package/landing-page/README.md +0 -11
  35. package/landing-page/bun.lock +0 -801
  36. package/landing-page/guidelines/Guidelines.md +0 -61
  37. package/landing-page/index.html +0 -37
  38. package/landing-page/package.json +0 -90
  39. package/landing-page/postcss.config.mjs +0 -15
  40. package/landing-page/public/_redirects +0 -1
  41. package/landing-page/public/favicon.png +0 -0
  42. package/landing-page/src/app/App.tsx +0 -53
  43. package/landing-page/src/app/components/figma/ImageWithFallback.tsx +0 -27
  44. package/landing-page/src/app/components/ui/accordion.tsx +0 -66
  45. package/landing-page/src/app/components/ui/alert-dialog.tsx +0 -157
  46. package/landing-page/src/app/components/ui/alert.tsx +0 -66
  47. package/landing-page/src/app/components/ui/aspect-ratio.tsx +0 -11
  48. package/landing-page/src/app/components/ui/avatar.tsx +0 -53
  49. package/landing-page/src/app/components/ui/badge.tsx +0 -46
  50. package/landing-page/src/app/components/ui/breadcrumb.tsx +0 -109
  51. package/landing-page/src/app/components/ui/button.tsx +0 -57
  52. package/landing-page/src/app/components/ui/calendar.tsx +0 -75
  53. package/landing-page/src/app/components/ui/card.tsx +0 -92
  54. package/landing-page/src/app/components/ui/carousel.tsx +0 -241
  55. package/landing-page/src/app/components/ui/chart.tsx +0 -353
  56. package/landing-page/src/app/components/ui/checkbox.tsx +0 -32
  57. package/landing-page/src/app/components/ui/collapsible.tsx +0 -33
  58. package/landing-page/src/app/components/ui/command.tsx +0 -177
  59. package/landing-page/src/app/components/ui/context-menu.tsx +0 -252
  60. package/landing-page/src/app/components/ui/dialog.tsx +0 -135
  61. package/landing-page/src/app/components/ui/drawer.tsx +0 -132
  62. package/landing-page/src/app/components/ui/dropdown-menu.tsx +0 -257
  63. package/landing-page/src/app/components/ui/form.tsx +0 -168
  64. package/landing-page/src/app/components/ui/hover-card.tsx +0 -44
  65. package/landing-page/src/app/components/ui/input-otp.tsx +0 -77
  66. package/landing-page/src/app/components/ui/input.tsx +0 -21
  67. package/landing-page/src/app/components/ui/label.tsx +0 -24
  68. package/landing-page/src/app/components/ui/menubar.tsx +0 -276
  69. package/landing-page/src/app/components/ui/navigation-menu.tsx +0 -168
  70. package/landing-page/src/app/components/ui/pagination.tsx +0 -127
  71. package/landing-page/src/app/components/ui/popover.tsx +0 -48
  72. package/landing-page/src/app/components/ui/progress.tsx +0 -31
  73. package/landing-page/src/app/components/ui/radio-group.tsx +0 -45
  74. package/landing-page/src/app/components/ui/resizable.tsx +0 -56
  75. package/landing-page/src/app/components/ui/scroll-area.tsx +0 -58
  76. package/landing-page/src/app/components/ui/select.tsx +0 -189
  77. package/landing-page/src/app/components/ui/separator.tsx +0 -28
  78. package/landing-page/src/app/components/ui/sheet.tsx +0 -139
  79. package/landing-page/src/app/components/ui/sidebar.tsx +0 -726
  80. package/landing-page/src/app/components/ui/skeleton.tsx +0 -13
  81. package/landing-page/src/app/components/ui/slider.tsx +0 -63
  82. package/landing-page/src/app/components/ui/sonner.tsx +0 -25
  83. package/landing-page/src/app/components/ui/switch.tsx +0 -31
  84. package/landing-page/src/app/components/ui/table.tsx +0 -116
  85. package/landing-page/src/app/components/ui/tabs.tsx +0 -66
  86. package/landing-page/src/app/components/ui/textarea.tsx +0 -18
  87. package/landing-page/src/app/components/ui/toggle-group.tsx +0 -73
  88. package/landing-page/src/app/components/ui/toggle.tsx +0 -47
  89. package/landing-page/src/app/components/ui/tooltip.tsx +0 -61
  90. package/landing-page/src/app/components/ui/use-mobile.ts +0 -21
  91. package/landing-page/src/app/components/ui/utils.ts +0 -6
  92. package/landing-page/src/components/docs/DocsContent.tsx +0 -718
  93. package/landing-page/src/components/docs/DocsSidebar.tsx +0 -84
  94. package/landing-page/src/components/landing/CTA.tsx +0 -59
  95. package/landing-page/src/components/landing/Comparison.tsx +0 -84
  96. package/landing-page/src/components/landing/FaultyTerminal.tsx +0 -424
  97. package/landing-page/src/components/landing/Features.tsx +0 -201
  98. package/landing-page/src/components/landing/Hero.tsx +0 -142
  99. package/landing-page/src/components/landing/Pricing.tsx +0 -140
  100. package/landing-page/src/components/landing/Roadmap.tsx +0 -86
  101. package/landing-page/src/components/landing/Security.tsx +0 -81
  102. package/landing-page/src/components/landing/TerminalWindow.tsx +0 -27
  103. package/landing-page/src/components/landing/UseCases.tsx +0 -55
  104. package/landing-page/src/components/landing/Workflow.tsx +0 -101
  105. package/landing-page/src/components/layout/DashboardNavbar.tsx +0 -37
  106. package/landing-page/src/components/layout/Footer.tsx +0 -55
  107. package/landing-page/src/components/layout/LandingNavbar.tsx +0 -82
  108. package/landing-page/src/components/ui/badge.tsx +0 -39
  109. package/landing-page/src/components/ui/breadcrumb.tsx +0 -115
  110. package/landing-page/src/components/ui/button.tsx +0 -57
  111. package/landing-page/src/components/ui/card.tsx +0 -79
  112. package/landing-page/src/components/ui/mock-terminal.tsx +0 -68
  113. package/landing-page/src/components/ui/separator.tsx +0 -28
  114. package/landing-page/src/lib/utils.ts +0 -6
  115. package/landing-page/src/main.tsx +0 -10
  116. package/landing-page/src/pages/Dashboard.tsx +0 -133
  117. package/landing-page/src/pages/DocsPage.tsx +0 -79
  118. package/landing-page/src/pages/LandingPage.tsx +0 -31
  119. package/landing-page/src/pages/TerminalView.tsx +0 -106
  120. package/landing-page/src/styles/fonts.css +0 -0
  121. package/landing-page/src/styles/index.css +0 -3
  122. package/landing-page/src/styles/tailwind.css +0 -4
  123. package/landing-page/src/styles/theme.css +0 -181
  124. package/landing-page/vite.config.ts +0 -19
  125. package/npm/darwin-arm64/bin/gssh +0 -0
  126. package/npm/darwin-arm64/package.json +0 -20
  127. package/scripts/build.ts +0 -298
  128. package/scripts/release.ts +0 -140
  129. package/src/__tests__/test-utils.ts +0 -298
  130. package/src/commands/__tests__/serve-messages.test.ts +0 -190
  131. package/src/commands/access.ts +0 -298
  132. package/src/commands/add.ts +0 -452
  133. package/src/commands/auth.ts +0 -364
  134. package/src/commands/connect.ts +0 -287
  135. package/src/commands/directory.ts +0 -16
  136. package/src/commands/host.ts +0 -396
  137. package/src/commands/identity.ts +0 -184
  138. package/src/commands/list.ts +0 -200
  139. package/src/commands/relay.ts +0 -315
  140. package/src/commands/remove.ts +0 -241
  141. package/src/commands/serve.ts +0 -1493
  142. package/src/commands/share.ts +0 -456
  143. package/src/commands/status.ts +0 -125
  144. package/src/commands/switch.ts +0 -353
  145. package/src/commands/tmux.ts +0 -317
  146. package/src/core/__tests__/access.test.ts +0 -240
  147. package/src/core/access.ts +0 -277
  148. package/src/core/bundle.ts +0 -342
  149. package/src/core/config.ts +0 -510
  150. package/src/core/git.ts +0 -317
  151. package/src/core/github.ts +0 -151
  152. package/src/core/identity.ts +0 -631
  153. package/src/core/linear.ts +0 -225
  154. package/src/core/shell.ts +0 -161
  155. package/src/core/trusted-relays.ts +0 -315
  156. package/src/index.ts +0 -810
  157. package/src/lib/remote-session/index.ts +0 -7
  158. package/src/lib/remote-session/protocol.ts +0 -267
  159. package/src/lib/remote-session/session-handler.ts +0 -581
  160. package/src/lib/remote-session/workspace-scanner.ts +0 -167
  161. package/src/lib/tmux-lite/README.md +0 -81
  162. package/src/lib/tmux-lite/cli.ts +0 -796
  163. package/src/lib/tmux-lite/crypto/__tests__/helpers/handshake-runner.ts +0 -349
  164. package/src/lib/tmux-lite/crypto/__tests__/helpers/mock-relay.ts +0 -291
  165. package/src/lib/tmux-lite/crypto/__tests__/helpers/test-identities.ts +0 -142
  166. package/src/lib/tmux-lite/crypto/__tests__/integration/authorization.integration.test.ts +0 -339
  167. package/src/lib/tmux-lite/crypto/__tests__/integration/e2e-communication.integration.test.ts +0 -477
  168. package/src/lib/tmux-lite/crypto/__tests__/integration/error-handling.integration.test.ts +0 -499
  169. package/src/lib/tmux-lite/crypto/__tests__/integration/handshake.integration.test.ts +0 -371
  170. package/src/lib/tmux-lite/crypto/__tests__/integration/security.integration.test.ts +0 -573
  171. package/src/lib/tmux-lite/crypto/access-control.test.ts +0 -512
  172. package/src/lib/tmux-lite/crypto/access-control.ts +0 -320
  173. package/src/lib/tmux-lite/crypto/frames.test.ts +0 -262
  174. package/src/lib/tmux-lite/crypto/frames.ts +0 -141
  175. package/src/lib/tmux-lite/crypto/handshake.ts +0 -894
  176. package/src/lib/tmux-lite/crypto/identity.test.ts +0 -220
  177. package/src/lib/tmux-lite/crypto/identity.ts +0 -286
  178. package/src/lib/tmux-lite/crypto/index.ts +0 -51
  179. package/src/lib/tmux-lite/crypto/invites.test.ts +0 -381
  180. package/src/lib/tmux-lite/crypto/invites.ts +0 -215
  181. package/src/lib/tmux-lite/crypto/keyexchange.ts +0 -435
  182. package/src/lib/tmux-lite/crypto/keys.test.ts +0 -58
  183. package/src/lib/tmux-lite/crypto/keys.ts +0 -47
  184. package/src/lib/tmux-lite/crypto/secretbox.test.ts +0 -169
  185. package/src/lib/tmux-lite/crypto/secretbox.ts +0 -124
  186. package/src/lib/tmux-lite/handshake-handler.ts +0 -451
  187. package/src/lib/tmux-lite/protocol.test.ts +0 -307
  188. package/src/lib/tmux-lite/protocol.ts +0 -266
  189. package/src/lib/tmux-lite/relay-client.ts +0 -506
  190. package/src/lib/tmux-lite/server.ts +0 -1250
  191. package/src/lib/tmux-lite/shell-integration.sh +0 -37
  192. package/src/lib/tmux-lite/terminal-queries.test.ts +0 -54
  193. package/src/lib/tmux-lite/terminal-queries.ts +0 -49
  194. package/src/relay/__tests__/e2e-flow.test.ts +0 -1284
  195. package/src/relay/__tests__/helpers/auth.ts +0 -354
  196. package/src/relay/__tests__/helpers/ports.ts +0 -51
  197. package/src/relay/__tests__/protocol-validation.test.ts +0 -265
  198. package/src/relay/authorization.ts +0 -303
  199. package/src/relay/embedded-assets.generated.d.ts +0 -15
  200. package/src/relay/identity.ts +0 -352
  201. package/src/relay/index.ts +0 -57
  202. package/src/relay/pipes.test.ts +0 -427
  203. package/src/relay/pipes.ts +0 -195
  204. package/src/relay/protocol.ts +0 -804
  205. package/src/relay/registries.test.ts +0 -437
  206. package/src/relay/registries.ts +0 -593
  207. package/src/relay/server.test.ts +0 -1323
  208. package/src/relay/server.ts +0 -1092
  209. package/src/relay/signing.ts +0 -238
  210. package/src/relay/types.ts +0 -69
  211. package/src/serve/client-session-manager.ts +0 -622
  212. package/src/serve/daemon.ts +0 -497
  213. package/src/serve/pty-session.ts +0 -236
  214. package/src/serve/types.ts +0 -169
  215. package/src/shared/components/Flow.tsx +0 -453
  216. package/src/shared/components/Flow.tui.tsx +0 -343
  217. package/src/shared/components/Flow.web.tsx +0 -442
  218. package/src/shared/components/Inbox.tsx +0 -446
  219. package/src/shared/components/Inbox.tui.tsx +0 -262
  220. package/src/shared/components/Inbox.web.tsx +0 -329
  221. package/src/shared/components/MachineList.tsx +0 -187
  222. package/src/shared/components/MachineList.tui.tsx +0 -161
  223. package/src/shared/components/MachineList.web.tsx +0 -210
  224. package/src/shared/components/ProjectList.tsx +0 -176
  225. package/src/shared/components/ProjectList.tui.tsx +0 -109
  226. package/src/shared/components/ProjectList.web.tsx +0 -143
  227. package/src/shared/components/SpacesBrowser.tsx +0 -332
  228. package/src/shared/components/SpacesBrowser.tui.tsx +0 -163
  229. package/src/shared/components/SpacesBrowser.web.tsx +0 -221
  230. package/src/shared/components/index.ts +0 -103
  231. package/src/shared/hooks/index.ts +0 -16
  232. package/src/shared/hooks/useNavigation.ts +0 -226
  233. package/src/shared/index.ts +0 -122
  234. package/src/shared/providers/LocalMachineProvider.ts +0 -425
  235. package/src/shared/providers/MachineProvider.ts +0 -165
  236. package/src/shared/providers/RemoteMachineProvider.ts +0 -444
  237. package/src/shared/providers/index.ts +0 -26
  238. package/src/shared/types.ts +0 -145
  239. package/src/tui/adapters.ts +0 -120
  240. package/src/tui/app.tsx +0 -1816
  241. package/src/tui/components/Terminal.tsx +0 -580
  242. package/src/tui/hooks/index.ts +0 -35
  243. package/src/tui/hooks/useAppState.ts +0 -314
  244. package/src/tui/hooks/useDaemonStatus.ts +0 -174
  245. package/src/tui/hooks/useInboxTUI.ts +0 -113
  246. package/src/tui/hooks/useRemoteMachines.ts +0 -209
  247. package/src/tui/index.ts +0 -24
  248. package/src/tui/state.ts +0 -299
  249. package/src/tui/terminal-bracketed-paste.test.ts +0 -45
  250. package/src/tui/terminal-bracketed-paste.ts +0 -47
  251. package/src/types/bundle.ts +0 -112
  252. package/src/types/config.ts +0 -89
  253. package/src/types/errors.ts +0 -206
  254. package/src/types/identity.ts +0 -284
  255. package/src/types/workspace-fuzzy.ts +0 -49
  256. package/src/types/workspace.ts +0 -151
  257. package/src/utils/bun-socket-writer.ts +0 -80
  258. package/src/utils/deps.ts +0 -127
  259. package/src/utils/fuzzy-match.ts +0 -125
  260. package/src/utils/logger.ts +0 -127
  261. package/src/utils/markdown.ts +0 -254
  262. package/src/utils/onboarding.ts +0 -229
  263. package/src/utils/prompts.ts +0 -114
  264. package/src/utils/run-commands.ts +0 -112
  265. package/src/utils/run-scripts.ts +0 -142
  266. package/src/utils/sanitize.ts +0 -98
  267. package/src/utils/secrets.ts +0 -122
  268. package/src/utils/shell-escape.ts +0 -40
  269. package/src/utils/utf8.ts +0 -79
  270. package/src/utils/workspace-state.ts +0 -47
  271. package/src/web/README.md +0 -73
  272. package/src/web/bun.lock +0 -575
  273. package/src/web/eslint.config.js +0 -23
  274. package/src/web/index.html +0 -16
  275. package/src/web/package.json +0 -37
  276. package/src/web/public/vite.svg +0 -1
  277. package/src/web/src/App.tsx +0 -604
  278. package/src/web/src/assets/react.svg +0 -1
  279. package/src/web/src/components/Terminal.tsx +0 -207
  280. package/src/web/src/hooks/useRelayConnection.ts +0 -224
  281. package/src/web/src/hooks/useTerminal.ts +0 -699
  282. package/src/web/src/index.css +0 -55
  283. package/src/web/src/lib/crypto/__tests__/web-terminal.test.ts +0 -1158
  284. package/src/web/src/lib/crypto/frames.ts +0 -205
  285. package/src/web/src/lib/crypto/handshake.ts +0 -396
  286. package/src/web/src/lib/crypto/identity.ts +0 -128
  287. package/src/web/src/lib/crypto/keyexchange.ts +0 -246
  288. package/src/web/src/lib/crypto/relay-signing.ts +0 -53
  289. package/src/web/src/lib/invite.ts +0 -58
  290. package/src/web/src/lib/storage/identity-store.ts +0 -94
  291. package/src/web/src/main.tsx +0 -10
  292. package/src/web/src/types/identity.ts +0 -45
  293. package/src/web/tsconfig.app.json +0 -28
  294. package/src/web/tsconfig.json +0 -7
  295. package/src/web/tsconfig.node.json +0 -26
  296. package/src/web/vite.config.ts +0 -31
  297. package/todo-security.md +0 -92
  298. package/tsconfig.json +0 -23
  299. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/12b7107e435bf1b9a8713a7f320472a63e543104d633d89a26f8d21f4e4ef182.sqlite +0 -0
  300. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/12b7107e435bf1b9a8713a7f320472a63e543104d633d89a26f8d21f4e4ef182.sqlite-shm +0 -0
  301. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/12b7107e435bf1b9a8713a7f320472a63e543104d633d89a26f8d21f4e4ef182.sqlite-wal +0 -0
  302. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/1a1ac3db1ab86ecf712f90322868a9aabc2c7dc9fe2dfbe94f9b075096276b0f.sqlite +0 -0
  303. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/1a1ac3db1ab86ecf712f90322868a9aabc2c7dc9fe2dfbe94f9b075096276b0f.sqlite-shm +0 -0
  304. package/worker/.wrangler/state/v3/d1/miniflare-D1DatabaseObject/1a1ac3db1ab86ecf712f90322868a9aabc2c7dc9fe2dfbe94f9b075096276b0f.sqlite-wal +0 -0
  305. package/worker/bun.lock +0 -237
  306. package/worker/package.json +0 -22
  307. package/worker/schema.sql +0 -96
  308. package/worker/src/handlers/auth.ts +0 -451
  309. package/worker/src/handlers/subdomains.ts +0 -376
  310. package/worker/src/handlers/user.ts +0 -98
  311. package/worker/src/index.ts +0 -70
  312. package/worker/src/middleware/auth.ts +0 -152
  313. package/worker/src/services/cloudflare.ts +0 -609
  314. package/worker/src/types.ts +0 -96
  315. package/worker/tsconfig.json +0 -15
  316. package/worker/wrangler.toml +0 -26
@@ -1,174 +0,0 @@
1
- # GitSpace Quick Start Guide
2
-
3
- Get up and running with GitSpace in 5 minutes.
4
-
5
- ---
6
-
7
- ## Prerequisites
8
-
9
- Install these tools first:
10
-
11
- - [Bun](https://bun.sh) - JavaScript runtime
12
- - [Git](https://git-scm.com/) - Version control
13
- - [GitHub CLI](https://cli.github.com/) - `gh auth login` before using GitSpace
14
-
15
- ---
16
-
17
- ## Installation
18
-
19
- ```bash
20
- bun install -g https://github.com/inkibra/gitspace.sh
21
-
22
- # Verify
23
- gssh --version
24
- ```
25
-
26
- ---
27
-
28
- ## Part 1: Local Workspace Management
29
-
30
- ### Launch the TUI
31
-
32
- ```bash
33
- gssh
34
- ```
35
-
36
- Use arrow keys to navigate, `Enter` to select, `?` for help, `q` to quit.
37
-
38
- ### Add a Project (CLI)
39
-
40
- ```bash
41
- gssh add project
42
- # Select a GitHub repo from the list
43
- ```
44
-
45
- ### Create a Workspace
46
-
47
- ```bash
48
- # In a project directory
49
- gssh add my-feature
50
- ```
51
-
52
- ### Switch Workspaces
53
-
54
- ```bash
55
- gssh switch my-feature
56
- # Or just: gssh switch (interactive)
57
- ```
58
-
59
- ---
60
-
61
- ## Part 2: Remote Terminal Access
62
-
63
- ### Option A: gitspace.sh (Easiest)
64
-
65
- ```bash
66
- # 1. Sign in with GitHub
67
- gssh auth login
68
-
69
- # 2. Reserve your subdomain
70
- gssh host reserve yourname
71
-
72
- # 3. Start serving
73
- gssh serve
74
-
75
- # 4. Open https://yourname.gitspace.sh in browser
76
- ```
77
-
78
- ### Option B: Self-Hosted
79
-
80
- ```bash
81
- # Terminal 1: Start relay
82
- gssh relay start --port 4480
83
-
84
- # Terminal 2: Setup identity and serve
85
- gssh identity init --label "My Mac"
86
- gssh serve --relay ws://localhost:4480/ws
87
-
88
- # Terminal 3: Create invite
89
- gssh share create
90
- # Share the URL with collaborators
91
- ```
92
-
93
- ### Connect from Another Device
94
-
95
- ```bash
96
- # First time: create identity
97
- gssh identity init --label "Laptop"
98
-
99
- # Connect using invite
100
- gssh connect <invite-token>
101
- ```
102
-
103
- ---
104
-
105
- ## Common Commands
106
-
107
- | Command | Description |
108
- |---------|-------------|
109
- | `gssh` | Launch TUI |
110
- | `gssh add project` | Add GitHub project |
111
- | `gssh add <name>` | Create workspace |
112
- | `gssh switch` | Switch workspace |
113
- | `gssh list` | List workspaces |
114
- | `gssh serve` | Enable remote access |
115
- | `gssh status` | Check daemon status |
116
-
117
- ---
118
-
119
- ## Key Bindings (TUI)
120
-
121
- | Key | Action |
122
- |-----|--------|
123
- | `Enter` | Select/Open |
124
- | `Tab` | Switch panels |
125
- | `n` | New item |
126
- | `d` | Delete |
127
- | `?` | Help |
128
- | `q` | Quit |
129
-
130
- ---
131
-
132
- ## Directory Structure
133
-
134
- ```
135
- ~/gitspace/
136
- ├── .config.json # Global config
137
- ├── <project>/
138
- │ ├── base/ # Base repo clone
139
- │ └── workspaces/ # Your worktrees
140
- │ └── my-feature/
141
- ```
142
-
143
- ---
144
-
145
- ## Next Steps
146
-
147
- - **Remote access in depth**: [docs/GETTING-STARTED.md](GETTING-STARTED.md)
148
- - **Security architecture**: [docs/REMOTE-DESIGN.md](REMOTE-DESIGN.md)
149
- - **Protocol reference**: [docs/PROTOCOL.md](PROTOCOL.md)
150
- - **Full README**: [README.md](../README.md)
151
-
152
- ---
153
-
154
- ## Troubleshooting
155
-
156
- ### "GitHub CLI not authenticated"
157
-
158
- ```bash
159
- gh auth login
160
- ```
161
-
162
- ### "No identity found"
163
-
164
- ```bash
165
- gssh identity init --label "My Device"
166
- ```
167
-
168
- ### "Machine offline"
169
-
170
- Ensure `gssh serve` is running on the target machine.
171
-
172
- ---
173
-
174
- *Last updated: 2025-01*
package/docs/RELAY.md DELETED
@@ -1,327 +0,0 @@
1
- # Relay Server Design
2
-
3
- The relay is the bridge between your machine and remote clients. It routes E2E encrypted terminal sessions between machines and clients over WebSocket.
4
-
5
- ---
6
-
7
- ## The Big Picture
8
-
9
- ```
10
- Cloudflare (optional)
11
- (TLS termination, DDoS protection)
12
-
13
-
14
- ┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐
15
- │ Your Mac │◀═════▶│ Relay Server │◀═════▶│ Browser/CLI │
16
- │ │ wss │ │ wss │ │
17
- │ gssh serve │ │ Routes by: │ │ Terminal view │
18
- │ (PTY sessions) │ │ - machine ID │ │ (xterm.js) │
19
- │ │ │ - connection │ │ │
20
- └─────────────────┘ └─────────────────┘ └─────────────────┘
21
- ```
22
-
23
- Your machine maintains a single persistent WebSocket to the relay. Terminal I/O flows over that connection as encrypted bytes. The relay is just a router - it cannot decrypt any content.
24
-
25
- ---
26
-
27
- ## Design Decisions
28
-
29
- ### Why One Connection from Machine?
30
-
31
- **Single multiplexed WebSocket** rather than connection-per-session because:
32
-
33
- 1. **Simpler firewall/NAT traversal** - One outbound connection, no port opening
34
- 2. **Easier reconnection** - One reconnect restores everything
35
- 3. **Lower overhead** - WebSocket framing once, not per-session
36
- 4. **Single TLS handshake** - Less CPU on both ends
37
-
38
- The machine-to-relay connection is the "trunk line." All client connections are multiplexed over it via `connectionId`.
39
-
40
- ### Why Relay Can't See Terminal Content?
41
-
42
- The relay forwards encrypted bytes for terminal sessions. It never decrypts. This is critical for:
43
-
44
- 1. **Zero-knowledge** - We can't see your commands, secrets, or output
45
- 2. **Security** - Compromised relay can't read terminal traffic
46
- 3. **Trust** - Users can verify encryption client-side
47
-
48
- See [PROTOCOL.md](./PROTOCOL.md) for the encryption protocol details.
49
-
50
- ---
51
-
52
- ## Connection Types
53
-
54
- ### 1. Machine Connection
55
-
56
- When `gssh serve` starts, it opens a WebSocket to `ws://relay:port/ws?role=machine`:
57
-
58
- ```
59
- → Relay sends relay_identity with challenge nonce
60
- → Machine signs nonce with Ed25519 private key
61
- → Machine sends register_machine with signingKey/keyExchangeKey + challengeResponse
62
- → Relay verifies signature and that signingKey is authorized
63
- → Relay sends access_list with authorized clients
64
- → Machine is now registered and ready
65
- ```
66
-
67
- The machine ID is derived from the machine's signing key. This ensures identity consistency across reconnects.
68
-
69
- ### 2. Client Connection
70
-
71
- Browser or CLI connects to `ws://relay:port/ws?role=client`:
72
-
73
- ```
74
- → Sign list/connect messages with Ed25519 identity
75
- → Connect via invite OR directly (if pre-authorized)
76
- → Relay routes to the target machine
77
- → Perform X3DH handshake with machine
78
- → Exchange E2E encrypted terminal data
79
- ```
80
-
81
- The relay verifies signatures, checks authorization, and creates a bidirectional pipe. It doesn't decrypt the content.
82
-
83
- ---
84
-
85
- ## Routing Architecture
86
-
87
- The relay maintains four registries:
88
-
89
- ### Machine Registry
90
- ```typescript
91
- machines: Map<machineId, {
92
- accountId: string,
93
- signingKey: string, // Ed25519 public key
94
- keyExchangeKey: string, // X25519 public key
95
- label: string,
96
- ws: WebSocket | null, // null = offline
97
- lastSeen: Date
98
- }>
99
- ```
100
-
101
- ### Invite Registry
102
- ```typescript
103
- invites: Map<inviteId, {
104
- machineId: string,
105
- expiresAt: number,
106
- maxUses: number,
107
- usedCount: number
108
- }>
109
- ```
110
-
111
- ### Machine Authorization Registry
112
- Per-machine client access:
113
- ```typescript
114
- authorizations: Map<machineId, Map<clientIdentityId, {
115
- signingKey: string,
116
- keyExchangeKey: string,
117
- accessType: 'full' | 'session-invite',
118
- sessionId?: string,
119
- label?: string,
120
- grantedAt: number
121
- }>>
122
- ```
123
-
124
- ### Global Access Registry
125
- Account-level access that applies to all machines:
126
- ```typescript
127
- globalAccess: Map<accountId, Map<clientIdentityId, {
128
- signingKey: string,
129
- keyExchangeKey: string,
130
- accessType: 'full' | 'session-invite',
131
- label?: string,
132
- grantedAt: number,
133
- machineIds?: string[] // If set, only applies to specific machines
134
- }>>
135
- ```
136
-
137
- When a machine connects, it receives the combined access list from both registries.
138
-
139
- ---
140
-
141
- ## Data Flow
142
-
143
- ### Client → Machine
144
-
145
- ```
146
- Client sends: { type: "data", data: "base64-encrypted" }
147
- Relay wraps: { type: "data", connectionId: "xyz", data: "base64-encrypted" }
148
- Machine receives with connectionId to identify the client
149
- ```
150
-
151
- ### Machine → Client
152
-
153
- ```
154
- Machine sends: { type: "data", connectionId: "xyz", data: "base64-encrypted" }
155
- Relay unwraps: { type: "data", data: "base64-encrypted" }
156
- Client receives without connectionId (it's their only connection)
157
- ```
158
-
159
- The `connectionId` is assigned by the relay when a client connects and is used to multiplex multiple clients over the single machine WebSocket.
160
-
161
- ---
162
-
163
- ## Connection Health
164
-
165
- ### Machine Keepalive
166
-
167
- WebSocket protocol ping/pong:
168
- - Machine sends ping every 30 seconds
169
- - Relay responds with pong
170
- - If no traffic for 60 seconds, machine reconnects
171
-
172
- ### Client Keepalive
173
-
174
- Same pattern. Relay pings clients, clients respond.
175
-
176
- ### Reconnection
177
-
178
- When machine reconnects:
179
- 1. Re-authenticates with same account
180
- 2. Re-registers with same signing key (must match)
181
- 3. Relay updates routing tables
182
- 4. Connected clients receive notification or continue streaming
183
-
184
- When machine goes offline:
185
- 1. Relay marks machine's `ws` as `null`
186
- 2. All connected clients receive `connection_failed`
187
- 3. Client connections are closed
188
-
189
- ---
190
-
191
- ## Auth Model
192
-
193
- ### Machine Authentication
194
-
195
- 1. Relay sends `relay_identity` with a random challenge nonce
196
- 2. Machine signs the nonce with its Ed25519 private key
197
- 3. Machine sends `register_machine` with signing keys + challengeResponse
198
- 4. Relay verifies the signature and checks the signing key is authorized
199
-
200
- ### Client Authentication
201
-
202
- Clients sign relay messages with their Ed25519 identity. Then either:
203
-
204
- 1. **Via Invite** - `connect_with_invite` with invite ID
205
- - Relay validates invite exists and isn't expired/exhausted
206
- - Relay decrements use count
207
- - Routes to target machine
208
-
209
- 2. **Direct** - `connect_to_machine` with machine ID + client identity
210
- - Relay checks both per-machine and global authorization registries
211
- - Client must be pre-authorized
212
-
213
- ### Authorization Flow
214
-
215
- 1. Client connects with invite → routed to machine
216
- 2. Machine performs X3DH handshake, validates invite signature
217
- 3. Machine sends `authorize_client` to relay (unless single-use invite)
218
- 4. Client is now authorized for future direct connections
219
-
220
- ### Global Access Flow
221
-
222
- 1. Machine owner sends `add_global_access` via any connected machine
223
- 2. Relay adds to global access registry
224
- 3. Relay broadcasts `access_update` to all connected machines in the account
225
- 4. All machines update their local access lists immediately
226
-
227
- ---
228
-
229
- ## Relay CLI Commands
230
-
231
- The relay server and its management are controlled via the `gssh relay` command group:
232
-
233
- | Command | Description |
234
- |---------|-------------|
235
- | `gssh relay start` | Start the relay server |
236
- | `gssh relay authorize <pubkey>` | Authorize a machine by its public key |
237
- | `gssh relay revoke <pubkey>` | Revoke a machine's authorization |
238
- | `gssh relay machines` | List authorized machines |
239
- | `gssh relay trusted` | List trusted relays (client-side) |
240
- | `gssh relay untrust <url>` | Remove relay trust (client-side) |
241
-
242
- **Note:** There is no separate `gssh machine` command group. Machine authorization is managed through the relay commands above.
243
-
244
- ---
245
-
246
- ## Error States
247
-
248
- | Situation | Response |
249
- |-----------|----------|
250
- | Invalid signature | `{ type: "error", code: "INVALID_SIGNATURE" }` |
251
- | Machine offline | `{ type: "error", code: "OFFLINE" }` |
252
- | Invite not found | `{ type: "error", code: "NOT_FOUND" }` |
253
- | Invite expired | `{ type: "error", code: "INVALID" }` |
254
- | Not authorized | `{ type: "error", code: "FORBIDDEN" }` |
255
- | Machine re-registration conflict | `{ success: false, error: "..." }` |
256
-
257
- ---
258
-
259
- ## Health Check Endpoint
260
-
261
- ```
262
- GET /health
263
- ```
264
-
265
- Returns:
266
- ```json
267
- {
268
- "machineCount": 5,
269
- "onlineMachineCount": 3,
270
- "inviteCount": 12,
271
- "authorizationCount": 8,
272
- "connectedClients": 7
273
- }
274
- ```
275
-
276
- ---
277
-
278
- ## Implementation Notes
279
-
280
- Using Bun APIs:
281
- - `Bun.serve()` with `websocket` handler for all WS connections
282
- - `fetch` handler for health check and static file serving
283
- - WebSocket `data` field holds connection metadata
284
- - All state in memory Maps (no database for MVP)
285
-
286
- The server is essentially:
287
- 1. Parse incoming connection (machine or client?)
288
- 2. Route to appropriate handler
289
- 3. Maintain registries of connections
290
- 4. Forward messages between matched pairs
291
-
292
- ---
293
-
294
- ## Current Features
295
-
296
- ### Cloudflare Hosting (Implemented)
297
-
298
- Users can expose their machine at `yourname.gitspace.sh`:
299
- - `gssh auth login` - Authenticate with GitHub
300
- - `gssh host reserve <name>` - Reserve a subdomain
301
- - `gssh serve` - Connects to gitspace.sh relay + Cloudflare tunnel
302
-
303
- See [GATEWAY-WORKER.md](./GATEWAY-WORKER.md) for the gateway architecture.
304
-
305
- ---
306
-
307
- ## Future Considerations
308
-
309
- ### Scaling
310
-
311
- Single relay works for MVP. For scale:
312
- - Consistent hash machines to specific relay pods
313
- - Redis/Valkey for cross-pod routing info
314
- - Load balancer with sticky sessions
315
-
316
- ### Port Tunnels (Not Yet Implemented)
317
-
318
- Future feature to expose `localhost:3000` at a public URL:
319
- - HTTP request proxying
320
- - WebSocket tunneling for HMR
321
- - Subdomain allocation per service
322
-
323
- See [INFRASTRUCTURE.md](./INFRASTRUCTURE.md) for the full vision.
324
-
325
- ---
326
-
327
- *Last updated: 2025-01*