pilothub 0.0.1 → 0.0.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 (388) hide show
  1. package/LICENSE +1 -0
  2. package/README.md +36 -129
  3. package/dist/browserAuth.d.ts +20 -0
  4. package/dist/browserAuth.js +156 -0
  5. package/dist/browserAuth.js.map +1 -0
  6. package/dist/browserAuth.test.d.ts +1 -0
  7. package/dist/browserAuth.test.js +83 -0
  8. package/dist/browserAuth.test.js.map +1 -0
  9. package/dist/cli/buildInfo.d.ts +3 -0
  10. package/dist/cli/buildInfo.js +103 -0
  11. package/dist/cli/buildInfo.js.map +1 -0
  12. package/dist/cli/commands/auth.d.ts +9 -0
  13. package/dist/cli/commands/auth.js +75 -0
  14. package/dist/cli/commands/auth.js.map +1 -0
  15. package/dist/cli/commands/delete.d.ts +11 -0
  16. package/dist/cli/commands/delete.js +67 -0
  17. package/dist/cli/commands/delete.js.map +1 -0
  18. package/dist/cli/commands/delete.test.d.ts +1 -0
  19. package/dist/cli/commands/delete.test.js +52 -0
  20. package/dist/cli/commands/delete.test.js.map +1 -0
  21. package/dist/cli/commands/publish.d.ts +9 -0
  22. package/dist/cli/commands/publish.js +87 -0
  23. package/dist/cli/commands/publish.js.map +1 -0
  24. package/dist/cli/commands/publish.test.d.ts +1 -0
  25. package/dist/cli/commands/publish.test.js +104 -0
  26. package/dist/cli/commands/publish.test.js.map +1 -0
  27. package/dist/cli/commands/skills.d.ts +23 -0
  28. package/dist/cli/commands/skills.js +298 -0
  29. package/dist/cli/commands/skills.js.map +1 -0
  30. package/dist/cli/commands/skills.test.d.ts +1 -0
  31. package/dist/cli/commands/skills.test.js +156 -0
  32. package/dist/cli/commands/skills.test.js.map +1 -0
  33. package/dist/cli/commands/star.d.ts +8 -0
  34. package/dist/cli/commands/star.js +38 -0
  35. package/dist/cli/commands/star.js.map +1 -0
  36. package/dist/cli/commands/sync.d.ts +3 -0
  37. package/dist/cli/commands/sync.js +160 -0
  38. package/dist/cli/commands/sync.js.map +1 -0
  39. package/dist/cli/commands/sync.test.d.ts +1 -0
  40. package/dist/cli/commands/sync.test.js +277 -0
  41. package/dist/cli/commands/sync.test.js.map +1 -0
  42. package/dist/cli/commands/syncHelpers.d.ts +76 -0
  43. package/dist/cli/commands/syncHelpers.js +349 -0
  44. package/dist/cli/commands/syncHelpers.js.map +1 -0
  45. package/dist/cli/commands/syncHelpers.test.d.ts +1 -0
  46. package/dist/cli/commands/syncHelpers.test.js +22 -0
  47. package/dist/cli/commands/syncHelpers.test.js.map +1 -0
  48. package/dist/cli/commands/syncTypes.d.ts +24 -0
  49. package/dist/cli/commands/syncTypes.js +2 -0
  50. package/dist/cli/commands/syncTypes.js.map +1 -0
  51. package/dist/cli/commands/unstar.d.ts +8 -0
  52. package/dist/cli/commands/unstar.js +38 -0
  53. package/dist/cli/commands/unstar.js.map +1 -0
  54. package/dist/cli/helpStyle.d.ts +13 -0
  55. package/dist/cli/helpStyle.js +38 -0
  56. package/dist/cli/helpStyle.js.map +1 -0
  57. package/dist/cli/pilotbotConfig.d.ts +6 -0
  58. package/dist/cli/pilotbotConfig.js +110 -0
  59. package/dist/cli/pilotbotConfig.js.map +1 -0
  60. package/dist/cli/pilotbotConfig.test.d.ts +1 -0
  61. package/dist/cli/pilotbotConfig.test.js +133 -0
  62. package/dist/cli/pilotbotConfig.test.js.map +1 -0
  63. package/dist/cli/registry.d.ts +7 -0
  64. package/dist/cli/registry.js +42 -0
  65. package/dist/cli/registry.js.map +1 -0
  66. package/dist/cli/registry.test.d.ts +1 -0
  67. package/dist/cli/registry.test.js +48 -0
  68. package/dist/cli/registry.test.js.map +1 -0
  69. package/dist/cli/scanSkills.d.ts +7 -0
  70. package/dist/cli/scanSkills.js +75 -0
  71. package/dist/cli/scanSkills.js.map +1 -0
  72. package/dist/cli/scanSkills.test.d.ts +1 -0
  73. package/dist/cli/scanSkills.test.js +60 -0
  74. package/dist/cli/scanSkills.test.js.map +1 -0
  75. package/dist/cli/slug.d.ts +2 -0
  76. package/dist/cli/slug.js +16 -0
  77. package/dist/cli/slug.js.map +1 -0
  78. package/dist/cli/types.d.ts +15 -0
  79. package/dist/cli/types.js +2 -0
  80. package/dist/cli/types.js.map +1 -0
  81. package/dist/cli/ui.d.ts +7 -0
  82. package/dist/cli/ui.js +72 -0
  83. package/dist/cli/ui.js.map +1 -0
  84. package/dist/cli.d.ts +2 -0
  85. package/dist/cli.js +268 -0
  86. package/dist/cli.js.map +1 -0
  87. package/dist/config.d.ts +4 -0
  88. package/dist/config.js +38 -0
  89. package/dist/config.js.map +1 -0
  90. package/dist/discovery.d.ts +5 -0
  91. package/dist/discovery.js +21 -0
  92. package/dist/discovery.js.map +1 -0
  93. package/dist/discovery.test.d.ts +1 -0
  94. package/dist/discovery.test.js +46 -0
  95. package/dist/discovery.test.js.map +1 -0
  96. package/dist/http.d.ts +32 -0
  97. package/dist/http.js +261 -0
  98. package/dist/http.js.map +1 -0
  99. package/dist/http.test.d.ts +1 -0
  100. package/dist/http.test.js +135 -0
  101. package/dist/http.test.js.map +1 -0
  102. package/dist/schema/ark.js.map +1 -0
  103. package/dist/schema/index.js.map +1 -0
  104. package/dist/schema/routes.js.map +1 -0
  105. package/{packages/schema/dist → dist/schema}/schemas.d.ts +0 -39
  106. package/{packages/schema/dist → dist/schema}/schemas.js +0 -22
  107. package/dist/schema/schemas.js.map +1 -0
  108. package/dist/schema/textFiles.js.map +1 -0
  109. package/dist/schema/textFiles.test.d.ts +1 -0
  110. package/dist/schema/textFiles.test.js +20 -0
  111. package/dist/schema/textFiles.test.js.map +1 -0
  112. package/dist/skills.d.ts +43 -0
  113. package/dist/skills.js +163 -0
  114. package/dist/skills.js.map +1 -0
  115. package/dist/skills.test.d.ts +1 -0
  116. package/dist/skills.test.js +144 -0
  117. package/dist/skills.test.js.map +1 -0
  118. package/dist/types.d.ts +7 -0
  119. package/dist/types.js +2 -0
  120. package/dist/types.js.map +1 -0
  121. package/package.json +27 -70
  122. package/.env.local.example +0 -19
  123. package/.github/workflows/ci.yml +0 -40
  124. package/.oxlintrc.json +0 -3
  125. package/AGENTS.md +0 -45
  126. package/CHANGELOG.md +0 -138
  127. package/DEPRECATIONS.md +0 -7
  128. package/biome.json +0 -41
  129. package/convex/_generated/api.d.ts +0 -153
  130. package/convex/_generated/api.js +0 -23
  131. package/convex/_generated/dataModel.d.ts +0 -60
  132. package/convex/_generated/server.d.ts +0 -143
  133. package/convex/_generated/server.js +0 -93
  134. package/convex/auth.config.ts +0 -8
  135. package/convex/auth.ts +0 -19
  136. package/convex/comments.ts +0 -88
  137. package/convex/crons.ts +0 -34
  138. package/convex/devSeed.ts +0 -459
  139. package/convex/devSeedExtra.ts +0 -541
  140. package/convex/downloads.ts +0 -78
  141. package/convex/githubBackups.ts +0 -170
  142. package/convex/githubBackupsNode.ts +0 -183
  143. package/convex/githubImport.ts +0 -317
  144. package/convex/githubSoulBackups.ts +0 -170
  145. package/convex/githubSoulBackupsNode.ts +0 -186
  146. package/convex/http.ts +0 -194
  147. package/convex/httpApi.handlers.test.ts +0 -488
  148. package/convex/httpApi.test.ts +0 -70
  149. package/convex/httpApi.ts +0 -305
  150. package/convex/httpApiV1.handlers.test.ts +0 -584
  151. package/convex/httpApiV1.ts +0 -1172
  152. package/convex/leaderboards.ts +0 -39
  153. package/convex/lib/access.ts +0 -36
  154. package/convex/lib/apiTokenAuth.ts +0 -36
  155. package/convex/lib/badges.ts +0 -50
  156. package/convex/lib/changelog.test.ts +0 -34
  157. package/convex/lib/changelog.ts +0 -278
  158. package/convex/lib/embeddings.ts +0 -38
  159. package/convex/lib/githubBackup.ts +0 -443
  160. package/convex/lib/githubImport.test.ts +0 -247
  161. package/convex/lib/githubImport.ts +0 -425
  162. package/convex/lib/githubSoulBackup.ts +0 -443
  163. package/convex/lib/leaderboards.ts +0 -103
  164. package/convex/lib/moderation.ts +0 -42
  165. package/convex/lib/public.ts +0 -89
  166. package/convex/lib/searchText.test.ts +0 -46
  167. package/convex/lib/searchText.ts +0 -27
  168. package/convex/lib/skillBackfill.test.ts +0 -34
  169. package/convex/lib/skillBackfill.ts +0 -67
  170. package/convex/lib/skillPublish.test.ts +0 -28
  171. package/convex/lib/skillPublish.ts +0 -284
  172. package/convex/lib/skillStats.ts +0 -80
  173. package/convex/lib/skills.test.ts +0 -197
  174. package/convex/lib/skills.ts +0 -273
  175. package/convex/lib/soulChangelog.ts +0 -273
  176. package/convex/lib/soulPublish.ts +0 -236
  177. package/convex/lib/tokens.test.ts +0 -33
  178. package/convex/lib/tokens.ts +0 -51
  179. package/convex/lib/webhooks.test.ts +0 -91
  180. package/convex/lib/webhooks.ts +0 -112
  181. package/convex/maintenance.test.ts +0 -270
  182. package/convex/maintenance.ts +0 -840
  183. package/convex/rateLimits.ts +0 -50
  184. package/convex/schema.ts +0 -472
  185. package/convex/search.test.ts +0 -12
  186. package/convex/search.ts +0 -254
  187. package/convex/seed.test.ts +0 -37
  188. package/convex/seed.ts +0 -254
  189. package/convex/seedSouls.ts +0 -111
  190. package/convex/skillStatEvents.ts +0 -568
  191. package/convex/skills.ts +0 -1606
  192. package/convex/soulComments.ts +0 -88
  193. package/convex/soulDownloads.ts +0 -14
  194. package/convex/soulStars.ts +0 -71
  195. package/convex/souls.ts +0 -570
  196. package/convex/stars.ts +0 -108
  197. package/convex/statsMaintenance.ts +0 -205
  198. package/convex/telemetry.ts +0 -434
  199. package/convex/tokens.ts +0 -88
  200. package/convex/tsconfig.json +0 -7
  201. package/convex/uploads.ts +0 -20
  202. package/convex/users.ts +0 -122
  203. package/convex/webhooks.ts +0 -50
  204. package/convex.json +0 -3
  205. package/docs/README.md +0 -32
  206. package/docs/api.md +0 -51
  207. package/docs/architecture.md +0 -61
  208. package/docs/auth.md +0 -54
  209. package/docs/cli.md +0 -117
  210. package/docs/deploy.md +0 -78
  211. package/docs/diffing.md +0 -84
  212. package/docs/github-import.md +0 -171
  213. package/docs/http-api.md +0 -187
  214. package/docs/manual-testing.md +0 -64
  215. package/docs/mintlify.md +0 -43
  216. package/docs/quickstart.md +0 -120
  217. package/docs/skill-format.md +0 -58
  218. package/docs/soul-format.md +0 -37
  219. package/docs/spec.md +0 -177
  220. package/docs/telemetry.md +0 -91
  221. package/docs/troubleshooting.md +0 -49
  222. package/docs/webhook.md +0 -51
  223. package/e2e/menu-smoke.pw.test.ts +0 -49
  224. package/e2e/pilothub.e2e.test.ts +0 -494
  225. package/e2e/search-exact.pw.test.ts +0 -97
  226. package/packages/pilothub/LICENSE +0 -22
  227. package/packages/pilothub/README.md +0 -57
  228. package/packages/pilothub/package.json +0 -41
  229. package/packages/pilothub/src/browserAuth.test.ts +0 -96
  230. package/packages/pilothub/src/browserAuth.ts +0 -174
  231. package/packages/pilothub/src/cli/buildInfo.ts +0 -94
  232. package/packages/pilothub/src/cli/commands/auth.ts +0 -97
  233. package/packages/pilothub/src/cli/commands/delete.test.ts +0 -73
  234. package/packages/pilothub/src/cli/commands/delete.ts +0 -83
  235. package/packages/pilothub/src/cli/commands/publish.test.ts +0 -122
  236. package/packages/pilothub/src/cli/commands/publish.ts +0 -108
  237. package/packages/pilothub/src/cli/commands/skills.test.ts +0 -191
  238. package/packages/pilothub/src/cli/commands/skills.ts +0 -380
  239. package/packages/pilothub/src/cli/commands/star.ts +0 -46
  240. package/packages/pilothub/src/cli/commands/sync.test.ts +0 -310
  241. package/packages/pilothub/src/cli/commands/sync.ts +0 -200
  242. package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +0 -26
  243. package/packages/pilothub/src/cli/commands/syncHelpers.ts +0 -427
  244. package/packages/pilothub/src/cli/commands/syncTypes.ts +0 -27
  245. package/packages/pilothub/src/cli/commands/unstar.ts +0 -48
  246. package/packages/pilothub/src/cli/helpStyle.ts +0 -45
  247. package/packages/pilothub/src/cli/pilotbotConfig.test.ts +0 -159
  248. package/packages/pilothub/src/cli/pilotbotConfig.ts +0 -147
  249. package/packages/pilothub/src/cli/registry.test.ts +0 -63
  250. package/packages/pilothub/src/cli/registry.ts +0 -43
  251. package/packages/pilothub/src/cli/scanSkills.test.ts +0 -64
  252. package/packages/pilothub/src/cli/scanSkills.ts +0 -84
  253. package/packages/pilothub/src/cli/slug.ts +0 -16
  254. package/packages/pilothub/src/cli/types.ts +0 -12
  255. package/packages/pilothub/src/cli/ui.ts +0 -75
  256. package/packages/pilothub/src/cli.ts +0 -311
  257. package/packages/pilothub/src/config.ts +0 -36
  258. package/packages/pilothub/src/discovery.test.ts +0 -75
  259. package/packages/pilothub/src/discovery.ts +0 -19
  260. package/packages/pilothub/src/http.test.ts +0 -156
  261. package/packages/pilothub/src/http.ts +0 -301
  262. package/packages/pilothub/src/schema/ark.ts +0 -29
  263. package/packages/pilothub/src/schema/index.ts +0 -5
  264. package/packages/pilothub/src/schema/routes.ts +0 -22
  265. package/packages/pilothub/src/schema/schemas.ts +0 -260
  266. package/packages/pilothub/src/schema/textFiles.test.ts +0 -23
  267. package/packages/pilothub/src/schema/textFiles.ts +0 -66
  268. package/packages/pilothub/src/skills.test.ts +0 -191
  269. package/packages/pilothub/src/skills.ts +0 -172
  270. package/packages/pilothub/src/types.ts +0 -10
  271. package/packages/pilothub/tsconfig.json +0 -14
  272. package/packages/schema/README.md +0 -3
  273. package/packages/schema/dist/ark.js.map +0 -1
  274. package/packages/schema/dist/index.js.map +0 -1
  275. package/packages/schema/dist/routes.js.map +0 -1
  276. package/packages/schema/dist/schemas.js.map +0 -1
  277. package/packages/schema/dist/textFiles.js.map +0 -1
  278. package/packages/schema/package.json +0 -26
  279. package/packages/schema/src/ark.ts +0 -29
  280. package/packages/schema/src/index.ts +0 -5
  281. package/packages/schema/src/routes.ts +0 -22
  282. package/packages/schema/src/schemas.test.ts +0 -123
  283. package/packages/schema/src/schemas.ts +0 -287
  284. package/packages/schema/src/textFiles.test.ts +0 -23
  285. package/packages/schema/src/textFiles.ts +0 -66
  286. package/packages/schema/tsconfig.json +0 -15
  287. package/pilothub +0 -46
  288. package/playwright.config.ts +0 -33
  289. package/public/.well-known/pilothub.json +0 -6
  290. package/public/api/v1/openapi.json +0 -379
  291. package/public/favicon.ico +0 -0
  292. package/public/logo192.png +0 -0
  293. package/public/logo512.png +0 -0
  294. package/public/manifest.json +0 -25
  295. package/public/og.png +0 -0
  296. package/public/og.svg +0 -98
  297. package/public/pilot-logo.png +0 -0
  298. package/public/pilot-mark.png +0 -0
  299. package/public/robots.txt +0 -3
  300. package/public/tanstack-circle-logo.png +0 -0
  301. package/public/tanstack-word-logo-white.svg +0 -1
  302. package/scripts/check-peer-deps.ts +0 -56
  303. package/scripts/docs-list.ts +0 -148
  304. package/scripts/run-playwright-local.sh +0 -14
  305. package/server/og/fetchSkillOgMeta.ts +0 -27
  306. package/server/og/fetchSoulOgMeta.ts +0 -27
  307. package/server/og/ogAssets.ts +0 -80
  308. package/server/og/skillOgSvg.test.ts +0 -59
  309. package/server/og/skillOgSvg.ts +0 -258
  310. package/server/og/soulOgSvg.ts +0 -209
  311. package/server/routes/og/skill.png.ts +0 -103
  312. package/server/routes/og/soul.png.ts +0 -111
  313. package/src/__tests__/skill-detail-page.test.tsx +0 -86
  314. package/src/__tests__/skills-index.test.tsx +0 -145
  315. package/src/__tests__/upload.route.test.tsx +0 -228
  316. package/src/components/AppProviders.tsx +0 -19
  317. package/src/components/ClientOnly.tsx +0 -18
  318. package/src/components/Footer.tsx +0 -29
  319. package/src/components/Header.tsx +0 -295
  320. package/src/components/InstallSwitcher.tsx +0 -53
  321. package/src/components/SkillCard.tsx +0 -36
  322. package/src/components/SkillDetailPage.tsx +0 -817
  323. package/src/components/SkillDiffCard.tsx +0 -485
  324. package/src/components/SoulCard.tsx +0 -19
  325. package/src/components/SoulDetailPage.tsx +0 -263
  326. package/src/components/UserBootstrap.tsx +0 -18
  327. package/src/components/ui/dropdown-menu.tsx +0 -67
  328. package/src/components/ui/toggle-group.tsx +0 -35
  329. package/src/convex/client.ts +0 -3
  330. package/src/lib/badges.ts +0 -29
  331. package/src/lib/diffing.test.ts +0 -163
  332. package/src/lib/diffing.ts +0 -106
  333. package/src/lib/gravatar.test.ts +0 -9
  334. package/src/lib/gravatar.ts +0 -158
  335. package/src/lib/og.test.ts +0 -142
  336. package/src/lib/og.ts +0 -156
  337. package/src/lib/publicUser.ts +0 -39
  338. package/src/lib/roles.ts +0 -19
  339. package/src/lib/site.test.ts +0 -130
  340. package/src/lib/site.ts +0 -84
  341. package/src/lib/theme-transition.test.ts +0 -134
  342. package/src/lib/theme-transition.ts +0 -134
  343. package/src/lib/theme.test.tsx +0 -88
  344. package/src/lib/theme.ts +0 -43
  345. package/src/lib/uploadFiles.jsdom.test.ts +0 -33
  346. package/src/lib/uploadFiles.test.ts +0 -123
  347. package/src/lib/uploadFiles.ts +0 -245
  348. package/src/lib/uploadUtils.test.ts +0 -78
  349. package/src/lib/uploadUtils.ts +0 -93
  350. package/src/lib/useAuthStatus.ts +0 -12
  351. package/src/lib/utils.test.ts +0 -9
  352. package/src/lib/utils.ts +0 -6
  353. package/src/logo.svg +0 -12
  354. package/src/routeTree.gen.ts +0 -345
  355. package/src/router.tsx +0 -17
  356. package/src/routes/$owner/$slug.tsx +0 -55
  357. package/src/routes/__root.tsx +0 -136
  358. package/src/routes/admin.tsx +0 -11
  359. package/src/routes/cli/auth.tsx +0 -168
  360. package/src/routes/dashboard.tsx +0 -97
  361. package/src/routes/import.tsx +0 -415
  362. package/src/routes/index.tsx +0 -252
  363. package/src/routes/management.tsx +0 -529
  364. package/src/routes/settings.tsx +0 -203
  365. package/src/routes/skills/index.tsx +0 -422
  366. package/src/routes/souls/$slug.tsx +0 -55
  367. package/src/routes/souls/index.tsx +0 -243
  368. package/src/routes/stars.tsx +0 -68
  369. package/src/routes/u/$handle.tsx +0 -307
  370. package/src/routes/upload/utils.ts +0 -81
  371. package/src/routes/upload.tsx +0 -499
  372. package/src/styles.css +0 -2718
  373. package/tsconfig.json +0 -24
  374. package/tsconfig.oxlint.json +0 -16
  375. package/vercel.json +0 -8
  376. package/vite.config.ts +0 -48
  377. package/vitest.config.ts +0 -47
  378. package/vitest.e2e.config.ts +0 -11
  379. package/vitest.setup.ts +0 -1
  380. /package/{packages/pilothub/bin → bin}/pilothub.js +0 -0
  381. /package/{packages/schema/dist → dist/schema}/ark.d.ts +0 -0
  382. /package/{packages/schema/dist → dist/schema}/ark.js +0 -0
  383. /package/{packages/schema/dist → dist/schema}/index.d.ts +0 -0
  384. /package/{packages/schema/dist → dist/schema}/index.js +0 -0
  385. /package/{packages/schema/dist → dist/schema}/routes.d.ts +0 -0
  386. /package/{packages/schema/dist → dist/schema}/routes.js +0 -0
  387. /package/{packages/schema/dist → dist/schema}/textFiles.d.ts +0 -0
  388. /package/{packages/schema/dist → dist/schema}/textFiles.js +0 -0
@@ -1,41 +0,0 @@
1
- {
2
- "name": "pilothub",
3
- "version": "0.3.0",
4
- "description": "PilotHub CLI \\u2014 install, update, search, and publish agent skills.",
5
- "license": "MIT",
6
- "type": "module",
7
- "bin": {
8
- "pilothub": "bin/pilothub.js"
9
- },
10
- "files": [
11
- "bin",
12
- "dist",
13
- "README.md",
14
- "LICENSE"
15
- ],
16
- "scripts": {
17
- "build": "tsc -p tsconfig.json",
18
- "dev": "node --enable-source-maps dist/cli.js",
19
- "prepublishOnly": "npm run build"
20
- },
21
- "dependencies": {
22
- "@clack/prompts": "^0.11.0",
23
- "arktype": "^2.1.29",
24
- "commander": "^14.0.2",
25
- "fflate": "^0.8.2",
26
- "ignore": "^7.0.5",
27
- "json5": "^2.2.3",
28
- "mime": "^4.1.0",
29
- "ora": "^9.0.0",
30
- "p-retry": "^7.1.1",
31
- "semver": "^7.7.3",
32
- "undici": "^7.16.0"
33
- },
34
- "devDependencies": {
35
- "@types/node": "^25.0.9",
36
- "typescript": "^5.9.3"
37
- },
38
- "engines": {
39
- "node": ">=20"
40
- }
41
- }
@@ -1,96 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { describe, expect, it } from 'vitest'
4
- import {
5
- buildCliAuthUrl,
6
- isAllowedLoopbackRedirectUri,
7
- startLoopbackAuthServer,
8
- } from './browserAuth'
9
-
10
- describe('browserAuth', () => {
11
- it('builds auth url', () => {
12
- const url = buildCliAuthUrl({
13
- siteUrl: 'https://example.com',
14
- redirectUri: 'http://127.0.0.1:1234/callback',
15
- label: 'CLI token',
16
- state: 'state123',
17
- })
18
- expect(url).toContain('https://example.com/cli/auth?')
19
- expect(url).toContain('redirect_uri=')
20
- expect(url).toContain('label_b64=')
21
- expect(url).toContain('state=')
22
- })
23
-
24
- it('builds auth url without label', () => {
25
- const url = buildCliAuthUrl({
26
- siteUrl: 'https://example.com',
27
- redirectUri: 'http://127.0.0.1:1234/callback',
28
- state: 'state123',
29
- })
30
- expect(url).toContain('https://example.com/cli/auth?')
31
- expect(url).not.toContain('label_b64=')
32
- })
33
-
34
- it('accepts only loopback http redirect uris', () => {
35
- expect(isAllowedLoopbackRedirectUri('http://127.0.0.1:1234/callback')).toBe(true)
36
- expect(isAllowedLoopbackRedirectUri('http://localhost:1234/callback')).toBe(true)
37
- expect(isAllowedLoopbackRedirectUri('http://[::1]:1234/callback')).toBe(true)
38
- expect(isAllowedLoopbackRedirectUri('https://127.0.0.1:1234/callback')).toBe(false)
39
- expect(isAllowedLoopbackRedirectUri('http://evil.com/callback')).toBe(false)
40
- expect(isAllowedLoopbackRedirectUri('not a url')).toBe(false)
41
- })
42
-
43
- it('receives token via loopback server', async () => {
44
- const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
45
- const payload = {
46
- token: 'clh_test',
47
- registry: 'https://example.convex.site',
48
- state: server.state,
49
- }
50
- await fetch(server.redirectUri.replace('/callback', '/token'), {
51
- method: 'POST',
52
- headers: { 'Content-Type': 'application/json' },
53
- body: JSON.stringify(payload),
54
- })
55
- await expect(server.waitForResult()).resolves.toEqual(payload)
56
- })
57
-
58
- it('serves callback html', async () => {
59
- const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
60
- const response = await fetch(server.redirectUri)
61
- expect(response.status).toBe(200)
62
- const text = await response.text()
63
- expect(text).toContain('PilotHub CLI Login')
64
- server.close()
65
- })
66
-
67
- it('returns 404 for unknown routes', async () => {
68
- const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
69
- const response = await fetch(server.redirectUri.replace('/callback', '/nope'))
70
- expect(response.status).toBe(404)
71
- server.close()
72
- })
73
-
74
- it('rejects invalid json payloads', async () => {
75
- const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
76
- const tokenUrl = server.redirectUri.replace('/callback', '/token')
77
- const response = await fetch(tokenUrl, { method: 'POST', body: '{' })
78
- expect(response.status).toBe(400)
79
- await expect(server.waitForResult()).rejects.toThrow()
80
- })
81
-
82
- it('rejects state mismatches', async () => {
83
- const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
84
- await fetch(server.redirectUri.replace('/callback', '/token'), {
85
- method: 'POST',
86
- headers: { 'Content-Type': 'application/json' },
87
- body: JSON.stringify({ token: 'clh_test', registry: 'https://example.com', state: 'nope' }),
88
- })
89
- await expect(server.waitForResult()).rejects.toThrow(/state mismatch/i)
90
- })
91
-
92
- it('times out waiting for login', async () => {
93
- const server = await startLoopbackAuthServer({ timeoutMs: 25 })
94
- await expect(server.waitForResult()).rejects.toThrow(/timed out waiting for browser login/i)
95
- })
96
- })
@@ -1,174 +0,0 @@
1
- import { createServer } from 'node:http'
2
- import type { AddressInfo } from 'node:net'
3
-
4
- export type LoopbackAuthResult = {
5
- token: string
6
- registry?: string
7
- state?: string
8
- }
9
-
10
- export function buildCliAuthUrl(params: {
11
- siteUrl: string
12
- redirectUri: string
13
- label?: string
14
- state: string
15
- }) {
16
- const url = new URL('/cli/auth', params.siteUrl)
17
- url.searchParams.set('redirect_uri', params.redirectUri)
18
- if (params.label) url.searchParams.set('label_b64', encodeBase64Url(params.label))
19
- url.searchParams.set('state', params.state)
20
- return url.toString()
21
- }
22
-
23
- export function isAllowedLoopbackRedirectUri(value: string) {
24
- let url: URL
25
- try {
26
- url = new URL(value)
27
- } catch {
28
- return false
29
- }
30
- if (url.protocol !== 'http:') return false
31
- const host = url.hostname.toLowerCase()
32
- if (host !== '127.0.0.1' && host !== 'localhost' && host !== '::1' && host !== '[::1]') {
33
- return false
34
- }
35
- return true
36
- }
37
-
38
- export async function startLoopbackAuthServer(params?: { timeoutMs?: number }) {
39
- const timeoutMs = params?.timeoutMs ?? 5 * 60_000
40
- const expectedState = generateState()
41
-
42
- let resolveToken: ((value: LoopbackAuthResult) => void) | null = null
43
- let rejectToken: ((error: Error) => void) | null = null
44
- const tokenPromise = new Promise<LoopbackAuthResult>((resolve, reject) => {
45
- resolveToken = resolve
46
- rejectToken = reject
47
- })
48
-
49
- const server = createServer((req, res) => {
50
- const method = req.method ?? 'GET'
51
- const url = req.url ?? '/'
52
-
53
- if (method === 'GET' && (url === '/' || url.startsWith('/callback'))) {
54
- res.statusCode = 200
55
- res.setHeader('Content-Type', 'text/html; charset=utf-8')
56
- res.end(CALLBACK_HTML)
57
- return
58
- }
59
-
60
- if (method === 'POST' && url === '/token') {
61
- const chunks: Uint8Array[] = []
62
- req.on('data', (chunk) => chunks.push(chunk as Uint8Array))
63
- req.on('end', () => {
64
- try {
65
- const raw = Buffer.concat(chunks).toString('utf8')
66
- const parsed = JSON.parse(raw) as unknown
67
- if (!parsed || typeof parsed !== 'object') throw new Error('invalid payload')
68
- const token = (parsed as { token?: unknown }).token
69
- const registry = (parsed as { registry?: unknown }).registry
70
- const state = (parsed as { state?: unknown }).state
71
- if (typeof token !== 'string' || !token.trim()) throw new Error('token required')
72
- if (typeof state !== 'string' || state !== expectedState) {
73
- throw new Error('state mismatch')
74
- }
75
- res.statusCode = 200
76
- res.setHeader('Content-Type', 'application/json')
77
- res.end(JSON.stringify({ ok: true }))
78
- resolveToken?.({
79
- token: token.trim(),
80
- registry: typeof registry === 'string' ? registry : undefined,
81
- state,
82
- })
83
- } catch (error) {
84
- res.statusCode = 400
85
- res.setHeader('Content-Type', 'application/json')
86
- res.end(JSON.stringify({ ok: false }))
87
- const message = error instanceof Error ? error.message : 'invalid payload'
88
- rejectToken?.(new Error(message))
89
- } finally {
90
- server.close()
91
- }
92
- })
93
- return
94
- }
95
-
96
- res.statusCode = 404
97
- res.setHeader('Content-Type', 'text/plain; charset=utf-8')
98
- res.end('Not found')
99
- })
100
-
101
- await new Promise<void>((resolve, reject) => {
102
- server.once('error', reject)
103
- server.listen(0, '127.0.0.1', () => resolve())
104
- })
105
- const address = server.address() as AddressInfo | null
106
- if (!address) {
107
- server.close()
108
- throw new Error('Failed to bind loopback server')
109
- }
110
- const redirectUri = `http://127.0.0.1:${address.port}/callback`
111
-
112
- const timeout = setTimeout(() => {
113
- server.close()
114
- rejectToken?.(new Error('Timed out waiting for browser login'))
115
- }, timeoutMs)
116
- tokenPromise.finally(() => clearTimeout(timeout)).catch(() => {})
117
-
118
- return {
119
- redirectUri,
120
- state: expectedState,
121
- waitForResult: () => tokenPromise,
122
- close: () => server.close(),
123
- }
124
- }
125
-
126
- const CALLBACK_HTML = `<!doctype html>
127
- <html lang="en">
128
- <meta charset="utf-8" />
129
- <meta name="viewport" content="width=device-width, initial-scale=1" />
130
- <title>PilotHub CLI Login</title>
131
- <style>
132
- :root { color-scheme: light dark; }
133
- body { font-family: ui-sans-serif, system-ui, -apple-system, Segoe UI, sans-serif; padding: 24px; }
134
- .card { max-width: 560px; margin: 40px auto; padding: 18px 16px; border: 1px solid rgba(127,127,127,.35); border-radius: 12px; }
135
- code { font-family: ui-monospace, SFMono-Regular, Menlo, monospace; }
136
- </style>
137
- <body>
138
- <div class="card">
139
- <h1 style="margin: 0 0 10px; font-size: 18px;">Completing login…</h1>
140
- <p id="status" style="margin: 0; opacity: .8;">Waiting for token.</p>
141
- </div>
142
- <script>
143
- const statusEl = document.getElementById('status')
144
- const params = new URLSearchParams(location.hash.replace(/^#/, ''))
145
- const token = params.get('token')
146
- const registry = params.get('registry')
147
- const state = params.get('state')
148
- if (!token) {
149
- statusEl.textContent = 'Missing token in URL. You can close this tab and try again.'
150
- } else if (!state) {
151
- statusEl.textContent = 'Missing state in URL. You can close this tab and try again.'
152
- } else {
153
- fetch('/token', {
154
- method: 'POST',
155
- headers: { 'Content-Type': 'application/json' },
156
- body: JSON.stringify({ token, registry, state }),
157
- }).then(() => {
158
- statusEl.textContent = 'Logged in. You can close this tab.'
159
- setTimeout(() => window.close(), 250)
160
- }).catch(() => {
161
- statusEl.textContent = 'Failed to send token to CLI. You can close this tab and try again.'
162
- })
163
- }
164
- </script>
165
- </body>
166
- </html>`
167
-
168
- function encodeBase64Url(value: string) {
169
- return Buffer.from(value, 'utf8').toString('base64url')
170
- }
171
-
172
- function generateState() {
173
- return Buffer.from(crypto.getRandomValues(new Uint8Array(16))).toString('hex')
174
- }
@@ -1,94 +0,0 @@
1
- import { existsSync, readFileSync, statSync } from 'node:fs'
2
- import { dirname, join, resolve } from 'node:path'
3
- import { fileURLToPath } from 'node:url'
4
-
5
- type PackageJson = { version?: string }
6
-
7
- function readPackageVersion() {
8
- try {
9
- const path = join(dirname(fileURLToPath(import.meta.url)), '../../package.json')
10
- const raw = readFileSync(path, 'utf8')
11
- const pkg = JSON.parse(raw) as PackageJson
12
- return typeof pkg.version === 'string' ? pkg.version : '0.0.0'
13
- } catch {
14
- return '0.0.0'
15
- }
16
- }
17
-
18
- function shortCommit(value: string) {
19
- const trimmed = value.trim()
20
- if (!trimmed) return null
21
- if (trimmed.length <= 8) return trimmed
22
- return trimmed.slice(0, 8)
23
- }
24
-
25
- export function getCliCommit() {
26
- const candidates = [
27
- process.env.PILOTHUB_COMMIT,
28
- process.env.VERCEL_GIT_COMMIT_SHA,
29
- process.env.GITHUB_SHA,
30
- process.env.COMMIT_SHA,
31
- ]
32
- for (const candidate of candidates) {
33
- if (!candidate) continue
34
- const short = shortCommit(candidate)
35
- if (short) return short
36
- }
37
- return readGitCommitFromCwd()
38
- }
39
-
40
- export function getCliVersion() {
41
- return readPackageVersion()
42
- }
43
-
44
- export function getCliBuildLabel() {
45
- const version = getCliVersion()
46
- const commit = getCliCommit()
47
- return commit ? `v${version} (${commit})` : `v${version}`
48
- }
49
-
50
- function readGitCommitFromCwd() {
51
- try {
52
- const gitDir = findGitDir(process.cwd())
53
- if (!gitDir) return null
54
- const headPath = join(gitDir, 'HEAD')
55
- if (!existsSync(headPath)) return null
56
- const head = readFileSync(headPath, 'utf8').trim()
57
- if (!head) return null
58
- if (!head.startsWith('ref:')) return shortCommit(head)
59
- const ref = head.replace(/^ref:\s*/, '').trim()
60
- if (!ref) return null
61
- const refPath = join(gitDir, ref)
62
- if (!existsSync(refPath)) return null
63
- const sha = readFileSync(refPath, 'utf8').trim()
64
- return shortCommit(sha)
65
- } catch {
66
- return null
67
- }
68
- }
69
-
70
- function findGitDir(start: string) {
71
- let current = resolve(start)
72
- for (;;) {
73
- const dotGit = join(current, '.git')
74
- if (existsSync(dotGit)) {
75
- try {
76
- const stat = statSync(dotGit)
77
- if (stat.isDirectory()) return dotGit
78
- } catch {
79
- // ignore
80
- }
81
- try {
82
- const content = readFileSync(dotGit, 'utf8').trim()
83
- const match = content.match(/^gitdir:\s*(.+)$/)
84
- if (match?.[1]) return resolve(current, match[1])
85
- } catch {
86
- return dotGit
87
- }
88
- return dotGit
89
- }
90
- const parent = resolve(current, '..')
91
- if (parent === current) return null
92
- current = parent
93
- }
94
- }
@@ -1,97 +0,0 @@
1
- import { buildCliAuthUrl, startLoopbackAuthServer } from '../../browserAuth.js'
2
- import { readGlobalConfig, writeGlobalConfig } from '../../config.js'
3
- import { discoverRegistryFromSite } from '../../discovery.js'
4
- import { apiRequest } from '../../http.js'
5
- import { ApiRoutes, ApiV1WhoamiResponseSchema } from '../../schema/index.js'
6
- import { getRegistry } from '../registry.js'
7
- import type { GlobalOpts } from '../types.js'
8
- import { createSpinner, fail, formatError, openInBrowser, promptHidden } from '../ui.js'
9
-
10
- export async function cmdLoginFlow(
11
- opts: GlobalOpts,
12
- options: { token?: string; label?: string; browser?: boolean },
13
- inputAllowed: boolean,
14
- ) {
15
- if (options.token) {
16
- await cmdLogin(opts, options.token, inputAllowed)
17
- return
18
- }
19
-
20
- if (options.browser === false) {
21
- fail('Token required (use --token or remove --no-browser)')
22
- }
23
-
24
- const label = String(options.label ?? 'CLI token').trim() || 'CLI token'
25
- const receiver = await startLoopbackAuthServer()
26
- const discovery = await discoverRegistryFromSite(opts.site).catch(() => null)
27
- const authBase = discovery?.authBase?.trim() || opts.site
28
- const authUrl = buildCliAuthUrl({
29
- siteUrl: authBase,
30
- redirectUri: receiver.redirectUri,
31
- label,
32
- state: receiver.state,
33
- })
34
-
35
- console.log(`Opening browser: ${authUrl}`)
36
- openInBrowser(authUrl)
37
-
38
- const result = await receiver.waitForResult()
39
- const registry = result.registry?.trim() || opts.registry
40
- await cmdLogin({ ...opts, registry }, result.token, inputAllowed)
41
- }
42
-
43
- export async function cmdLogin(
44
- opts: GlobalOpts,
45
- tokenFlag: string | undefined,
46
- inputAllowed: boolean,
47
- ) {
48
- if (!tokenFlag && !inputAllowed) fail('Token required (use --token or remove --no-input)')
49
-
50
- const token = tokenFlag || (await promptHidden('PilotHub token: '))
51
- if (!token) fail('Token required')
52
-
53
- const registry = await getRegistry(opts, { cache: true })
54
- const spinner = createSpinner('Verifying token')
55
- try {
56
- const whoami = await apiRequest(
57
- registry,
58
- { method: 'GET', path: ApiRoutes.whoami, token },
59
- ApiV1WhoamiResponseSchema,
60
- )
61
- if (!whoami.user) fail('Login failed')
62
-
63
- await writeGlobalConfig({ registry, token })
64
- const handle = whoami.user.handle ? `@${whoami.user.handle}` : 'unknown user'
65
- spinner.succeed(`OK. Logged in as ${handle}.`)
66
- } catch (error) {
67
- spinner.fail(formatError(error))
68
- throw error
69
- }
70
- }
71
-
72
- export async function cmdLogout(opts: GlobalOpts) {
73
- const cfg = await readGlobalConfig()
74
- const registry = cfg?.registry || (await getRegistry(opts, { cache: true }))
75
- await writeGlobalConfig({ registry, token: undefined })
76
- console.log('OK. Logged out.')
77
- }
78
-
79
- export async function cmdWhoami(opts: GlobalOpts) {
80
- const cfg = await readGlobalConfig()
81
- const token = cfg?.token
82
- if (!token) fail('Not logged in. Run: pilothub login')
83
- const registry = await getRegistry(opts, { cache: true })
84
-
85
- const spinner = createSpinner('Checking token')
86
- try {
87
- const whoami = await apiRequest(
88
- registry,
89
- { method: 'GET', path: ApiRoutes.whoami, token },
90
- ApiV1WhoamiResponseSchema,
91
- )
92
- spinner.succeed(whoami.user.handle ?? 'unknown')
93
- } catch (error) {
94
- spinner.fail(formatError(error))
95
- throw error
96
- }
97
- }
@@ -1,73 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { afterEach, describe, expect, it, vi } from 'vitest'
4
- import type { GlobalOpts } from '../types'
5
-
6
- vi.mock('../../config.js', () => ({
7
- readGlobalConfig: vi.fn(async () => ({ registry: 'https://pilothub.com', token: 'tkn' })),
8
- }))
9
-
10
- vi.mock('../registry.js', () => ({
11
- getRegistry: vi.fn(async () => 'https://pilothub.com'),
12
- }))
13
-
14
- const mockApiRequest = vi.fn()
15
- vi.mock('../../http.js', () => ({
16
- apiRequest: (registry: unknown, args: unknown, schema?: unknown) =>
17
- mockApiRequest(registry, args, schema),
18
- }))
19
-
20
- const mockFail = vi.fn((message: string) => {
21
- throw new Error(message)
22
- })
23
-
24
- vi.mock('../ui.js', () => ({
25
- createSpinner: vi.fn(() => ({ succeed: vi.fn(), fail: vi.fn() })),
26
- fail: (message: string) => mockFail(message),
27
- formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
28
- isInteractive: () => false,
29
- promptConfirm: vi.fn(async () => true),
30
- }))
31
-
32
- const { cmdDeleteSkill, cmdUndeleteSkill } = await import('./delete')
33
-
34
- function makeOpts(): GlobalOpts {
35
- return {
36
- workdir: '/work',
37
- dir: '/work/skills',
38
- site: 'https://pilothub.com',
39
- registry: 'https://pilothub.com',
40
- registrySource: 'default',
41
- }
42
- }
43
-
44
- afterEach(() => {
45
- vi.clearAllMocks()
46
- })
47
-
48
- describe('delete/undelete', () => {
49
- it('requires --yes when input is disabled', async () => {
50
- await expect(cmdDeleteSkill(makeOpts(), 'demo', {}, false)).rejects.toThrow(/--yes/i)
51
- await expect(cmdUndeleteSkill(makeOpts(), 'demo', {}, false)).rejects.toThrow(/--yes/i)
52
- })
53
-
54
- it('calls delete endpoint with --yes', async () => {
55
- mockApiRequest.mockResolvedValueOnce({ ok: true })
56
- await cmdDeleteSkill(makeOpts(), 'demo', { yes: true }, false)
57
- expect(mockApiRequest).toHaveBeenCalledWith(
58
- expect.anything(),
59
- expect.objectContaining({ method: 'DELETE', path: '/api/v1/skills/demo' }),
60
- expect.anything(),
61
- )
62
- })
63
-
64
- it('calls undelete endpoint with --yes', async () => {
65
- mockApiRequest.mockResolvedValueOnce({ ok: true })
66
- await cmdUndeleteSkill(makeOpts(), 'demo', { yes: true }, false)
67
- expect(mockApiRequest).toHaveBeenCalledWith(
68
- expect.anything(),
69
- expect.objectContaining({ method: 'POST', path: '/api/v1/skills/demo/undelete' }),
70
- expect.anything(),
71
- )
72
- })
73
- })
@@ -1,83 +0,0 @@
1
- import { readGlobalConfig } from '../../config.js'
2
- import { apiRequest } from '../../http.js'
3
- import { ApiRoutes, ApiV1DeleteResponseSchema, parseArk } from '../../schema/index.js'
4
- import { getRegistry } from '../registry.js'
5
- import type { GlobalOpts } from '../types.js'
6
- import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js'
7
-
8
- async function requireToken() {
9
- const cfg = await readGlobalConfig()
10
- const token = cfg?.token
11
- if (!token) fail('Not logged in. Run: pilothub login')
12
- return token
13
- }
14
-
15
- export async function cmdDeleteSkill(
16
- opts: GlobalOpts,
17
- slugArg: string,
18
- options: { yes?: boolean },
19
- inputAllowed: boolean,
20
- ) {
21
- const slug = slugArg.trim().toLowerCase()
22
- if (!slug) fail('Slug required')
23
- const allowPrompt = isInteractive() && inputAllowed !== false
24
-
25
- if (!options.yes) {
26
- if (!allowPrompt) fail('Pass --yes (no input)')
27
- const ok = await promptConfirm(`Delete ${slug}? (soft delete)`)
28
- if (!ok) return
29
- }
30
-
31
- const token = await requireToken()
32
- const registry = await getRegistry(opts, { cache: true })
33
- const spinner = createSpinner(`Deleting ${slug}`)
34
- try {
35
- const result = await apiRequest(
36
- registry,
37
- { method: 'DELETE', path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}`, token },
38
- ApiV1DeleteResponseSchema,
39
- )
40
- spinner.succeed(`OK. Deleted ${slug}`)
41
- return parseArk(ApiV1DeleteResponseSchema, result, 'Delete response')
42
- } catch (error) {
43
- spinner.fail(formatError(error))
44
- throw error
45
- }
46
- }
47
-
48
- export async function cmdUndeleteSkill(
49
- opts: GlobalOpts,
50
- slugArg: string,
51
- options: { yes?: boolean },
52
- inputAllowed: boolean,
53
- ) {
54
- const slug = slugArg.trim().toLowerCase()
55
- if (!slug) fail('Slug required')
56
- const allowPrompt = isInteractive() && inputAllowed !== false
57
-
58
- if (!options.yes) {
59
- if (!allowPrompt) fail('Pass --yes (no input)')
60
- const ok = await promptConfirm(`Undelete ${slug}?`)
61
- if (!ok) return
62
- }
63
-
64
- const token = await requireToken()
65
- const registry = await getRegistry(opts, { cache: true })
66
- const spinner = createSpinner(`Undeleting ${slug}`)
67
- try {
68
- const result = await apiRequest(
69
- registry,
70
- {
71
- method: 'POST',
72
- path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}/undelete`,
73
- token,
74
- },
75
- ApiV1DeleteResponseSchema,
76
- )
77
- spinner.succeed(`OK. Undeleted ${slug}`)
78
- return parseArk(ApiV1DeleteResponseSchema, result, 'Undelete response')
79
- } catch (error) {
80
- spinner.fail(formatError(error))
81
- throw error
82
- }
83
- }