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,311 +0,0 @@
1
- #!/usr/bin/env node
2
- import { stat } from 'node:fs/promises'
3
- import { join, resolve } from 'node:path'
4
- import { Command } from 'commander'
5
- import { getCliBuildLabel, getCliVersion } from './cli/buildInfo.js'
6
- import { cmdLoginFlow, cmdLogout, cmdWhoami } from './cli/commands/auth.js'
7
- import { cmdDeleteSkill, cmdUndeleteSkill } from './cli/commands/delete.js'
8
- import { cmdPublish } from './cli/commands/publish.js'
9
- import { cmdExplore, cmdInstall, cmdList, cmdSearch, cmdUpdate } from './cli/commands/skills.js'
10
- import { cmdStarSkill } from './cli/commands/star.js'
11
- import { cmdSync } from './cli/commands/sync.js'
12
- import { cmdUnstarSkill } from './cli/commands/unstar.js'
13
- import { configureCommanderHelp, styleEnvBlock, styleTitle } from './cli/helpStyle.js'
14
- import { resolvePilotbotDefaultWorkspace } from './cli/pilotbotConfig.js'
15
- import { DEFAULT_REGISTRY, DEFAULT_SITE } from './cli/registry.js'
16
- import type { GlobalOpts } from './cli/types.js'
17
- import { fail } from './cli/ui.js'
18
- import { readGlobalConfig } from './config.js'
19
-
20
- const program = new Command()
21
- .name('pilothub')
22
- .description(
23
- `${styleTitle(`PilotHub CLI ${getCliBuildLabel()}`)}\n${styleEnvBlock(
24
- 'install, update, search, and publish agent skills.',
25
- )}`,
26
- )
27
- .version(getCliVersion(), '-V, --cli-version', 'Show CLI version')
28
- .option('--workdir <dir>', 'Working directory (default: cwd)')
29
- .option('--dir <dir>', 'Skills directory (relative to workdir, default: skills)')
30
- .option('--site <url>', 'Site base URL (for browser login)')
31
- .option('--registry <url>', 'Registry API base URL')
32
- .option('--no-input', 'Disable prompts')
33
- .showHelpAfterError()
34
- .showSuggestionAfterError()
35
- .addHelpText(
36
- 'after',
37
- styleEnvBlock('\nEnv:\n PILOTHUB_SITE\n PILOTHUB_REGISTRY\n PILOTHUB_WORKDIR\n'),
38
- )
39
-
40
- configureCommanderHelp(program)
41
-
42
- async function resolveGlobalOpts(): Promise<GlobalOpts> {
43
- const raw = program.opts<{ workdir?: string; dir?: string; site?: string; registry?: string }>()
44
- const workdir = await resolveWorkdir(raw.workdir)
45
- const dir = resolve(workdir, raw.dir ?? 'skills')
46
- const site = raw.site ?? process.env.PILOTHUB_SITE ?? DEFAULT_SITE
47
- const registrySource = raw.registry ? 'cli' : process.env.PILOTHUB_REGISTRY ? 'env' : 'default'
48
- const registry = raw.registry ?? process.env.PILOTHUB_REGISTRY ?? DEFAULT_REGISTRY
49
- return { workdir, dir, site, registry, registrySource }
50
- }
51
-
52
- function isInputAllowed() {
53
- const globalFlags = program.opts<{ input?: boolean }>()
54
- return globalFlags.input !== false
55
- }
56
-
57
- async function resolveWorkdir(explicit?: string) {
58
- if (explicit?.trim()) return resolve(explicit.trim())
59
- const envWorkdir = process.env.PILOTHUB_WORKDIR?.trim()
60
- if (envWorkdir) return resolve(envWorkdir)
61
-
62
- const cwd = resolve(process.cwd())
63
- const hasMarker = await hasPilothubMarker(cwd)
64
- if (hasMarker) return cwd
65
-
66
- const pilotbotWorkspace = await resolvePilotbotDefaultWorkspace()
67
- return pilotbotWorkspace ? resolve(pilotbotWorkspace) : cwd
68
- }
69
-
70
- async function hasPilothubMarker(workdir: string) {
71
- const lockfile = join(workdir, '.pilothub', 'lock.json')
72
- if (await pathExists(lockfile)) return true
73
- const markerDir = join(workdir, '.pilothub')
74
- return pathExists(markerDir)
75
- }
76
-
77
- async function pathExists(path: string) {
78
- try {
79
- await stat(path)
80
- return true
81
- } catch {
82
- return false
83
- }
84
- }
85
-
86
- program
87
- .command('login')
88
- .description('Log in (opens browser or stores token)')
89
- .option('--token <token>', 'API token')
90
- .option('--label <label>', 'Token label (browser flow only)', 'CLI token')
91
- .option('--no-browser', 'Do not open browser (requires --token)')
92
- .action(async (options) => {
93
- const opts = await resolveGlobalOpts()
94
- await cmdLoginFlow(opts, options, isInputAllowed())
95
- })
96
-
97
- program
98
- .command('logout')
99
- .description('Remove stored token')
100
- .action(async () => {
101
- const opts = await resolveGlobalOpts()
102
- await cmdLogout(opts)
103
- })
104
-
105
- program
106
- .command('whoami')
107
- .description('Validate token')
108
- .action(async () => {
109
- const opts = await resolveGlobalOpts()
110
- await cmdWhoami(opts)
111
- })
112
-
113
- const auth = program
114
- .command('auth')
115
- .description('Authentication commands')
116
- .showHelpAfterError()
117
- .showSuggestionAfterError()
118
-
119
- auth
120
- .command('login')
121
- .description('Log in (opens browser or stores token)')
122
- .option('--token <token>', 'API token')
123
- .option('--label <label>', 'Token label (browser flow only)', 'CLI token')
124
- .option('--no-browser', 'Do not open browser (requires --token)')
125
- .action(async (options) => {
126
- const opts = await resolveGlobalOpts()
127
- await cmdLoginFlow(opts, options, isInputAllowed())
128
- })
129
-
130
- auth
131
- .command('logout')
132
- .description('Remove stored token')
133
- .action(async () => {
134
- const opts = await resolveGlobalOpts()
135
- await cmdLogout(opts)
136
- })
137
-
138
- auth
139
- .command('whoami')
140
- .description('Validate token')
141
- .action(async () => {
142
- const opts = await resolveGlobalOpts()
143
- await cmdWhoami(opts)
144
- })
145
-
146
- program
147
- .command('search')
148
- .description('Vector search skills')
149
- .argument('<query...>', 'Query string')
150
- .option('--limit <n>', 'Max results', (value) => Number.parseInt(value, 10))
151
- .action(async (queryParts, options) => {
152
- const opts = await resolveGlobalOpts()
153
- const query = queryParts.join(' ').trim()
154
- await cmdSearch(opts, query, options.limit)
155
- })
156
-
157
- program
158
- .command('install')
159
- .description('Install into <dir>/<slug>')
160
- .argument('<slug>', 'Skill slug')
161
- .option('--version <version>', 'Version to install')
162
- .option('--force', 'Overwrite existing folder')
163
- .action(async (slug, options) => {
164
- const opts = await resolveGlobalOpts()
165
- await cmdInstall(opts, slug, options.version, options.force)
166
- })
167
-
168
- program
169
- .command('update')
170
- .description('Update installed skills')
171
- .argument('[slug]', 'Skill slug')
172
- .option('--all', 'Update all installed skills')
173
- .option('--version <version>', 'Update to specific version (single slug only)')
174
- .option('--force', 'Overwrite when local files do not match any version')
175
- .action(async (slug, options) => {
176
- const opts = await resolveGlobalOpts()
177
- await cmdUpdate(opts, slug, options, isInputAllowed())
178
- })
179
-
180
- program
181
- .command('list')
182
- .description('List installed skills (from lockfile)')
183
- .action(async () => {
184
- const opts = await resolveGlobalOpts()
185
- await cmdList(opts)
186
- })
187
-
188
- program
189
- .command('explore')
190
- .description('Browse latest updated skills from the registry')
191
- .option(
192
- '--limit <n>',
193
- 'Number of skills to show (max 200)',
194
- (value) => Number.parseInt(value, 10),
195
- 25,
196
- )
197
- .option(
198
- '--sort <order>',
199
- 'Sort by newest, downloads, rating, installs, installsAllTime, or trending',
200
- 'newest',
201
- )
202
- .option('--json', 'Output JSON')
203
- .action(async (options) => {
204
- const opts = await resolveGlobalOpts()
205
- const limit =
206
- typeof options.limit === 'number' && Number.isFinite(options.limit) ? options.limit : 25
207
- await cmdExplore(opts, { limit, sort: options.sort, json: options.json })
208
- })
209
-
210
- program
211
- .command('publish')
212
- .description('Publish skill from folder')
213
- .argument('<path>', 'Skill folder path')
214
- .option('--slug <slug>', 'Skill slug')
215
- .option('--name <name>', 'Display name')
216
- .option('--version <version>', 'Version (semver)')
217
- .option('--fork-of <slug[@version]>', 'Mark as a fork of an existing skill')
218
- .option('--changelog <text>', 'Changelog text')
219
- .option('--tags <tags>', 'Comma-separated tags', 'latest')
220
- .action(async (folder, options) => {
221
- const opts = await resolveGlobalOpts()
222
- await cmdPublish(opts, folder, options)
223
- })
224
-
225
- program
226
- .command('delete')
227
- .description('Soft-delete a skill (owner/admin only)')
228
- .argument('<slug>', 'Skill slug')
229
- .option('--yes', 'Skip confirmation')
230
- .action(async (slug, options) => {
231
- const opts = await resolveGlobalOpts()
232
- await cmdDeleteSkill(opts, slug, options, isInputAllowed())
233
- })
234
-
235
- program
236
- .command('undelete')
237
- .description('Restore a soft-deleted skill (owner/admin only)')
238
- .argument('<slug>', 'Skill slug')
239
- .option('--yes', 'Skip confirmation')
240
- .action(async (slug, options) => {
241
- const opts = await resolveGlobalOpts()
242
- await cmdUndeleteSkill(opts, slug, options, isInputAllowed())
243
- })
244
-
245
- program
246
- .command('star')
247
- .description('Add a skill to your highlights')
248
- .argument('<slug>', 'Skill slug')
249
- .option('--yes', 'Skip confirmation')
250
- .action(async (slug, options) => {
251
- const opts = await resolveGlobalOpts()
252
- await cmdStarSkill(opts, slug, options, isInputAllowed())
253
- })
254
-
255
- program
256
- .command('unstar')
257
- .description('Remove a skill from your highlights')
258
- .argument('<slug>', 'Skill slug')
259
- .option('--yes', 'Skip confirmation')
260
- .action(async (slug, options) => {
261
- const opts = await resolveGlobalOpts()
262
- await cmdUnstarSkill(opts, slug, options, isInputAllowed())
263
- })
264
-
265
- program
266
- .command('sync')
267
- .description('Scan local skills and publish new/updated ones')
268
- .option('--root <dir...>', 'Extra scan roots (one or more)')
269
- .option('--all', 'Upload all new/updated skills without prompting')
270
- .option('--dry-run', 'Show what would be uploaded')
271
- .option('--bump <type>', 'Version bump for updates (patch|minor|major)', 'patch')
272
- .option('--changelog <text>', 'Changelog to use for updates (non-interactive)')
273
- .option('--tags <tags>', 'Comma-separated tags', 'latest')
274
- .option('--concurrency <n>', 'Concurrent registry checks (default: 4)', '4')
275
- .action(async (options) => {
276
- const opts = await resolveGlobalOpts()
277
- const bump = String(options.bump ?? 'patch') as 'patch' | 'minor' | 'major'
278
- if (!['patch', 'minor', 'major'].includes(bump)) fail('--bump must be patch|minor|major')
279
- const concurrencyRaw = Number(options.concurrency ?? 4)
280
- const concurrency = Number.isFinite(concurrencyRaw) ? Math.round(concurrencyRaw) : 4
281
- if (concurrency < 1 || concurrency > 32) fail('--concurrency must be between 1 and 32')
282
- await cmdSync(
283
- opts,
284
- {
285
- root: options.root,
286
- all: options.all,
287
- dryRun: options.dryRun,
288
- bump,
289
- changelog: options.changelog,
290
- tags: options.tags,
291
- concurrency,
292
- },
293
- isInputAllowed(),
294
- )
295
- })
296
-
297
- program.action(async () => {
298
- const opts = await resolveGlobalOpts()
299
- const cfg = await readGlobalConfig()
300
- if (cfg?.token) {
301
- await cmdSync(opts, {}, isInputAllowed())
302
- return
303
- }
304
- program.outputHelp()
305
- process.exitCode = 0
306
- })
307
-
308
- void program.parseAsync(process.argv).catch((error) => {
309
- const message = error instanceof Error ? error.message : String(error)
310
- fail(message)
311
- })
@@ -1,36 +0,0 @@
1
- import { mkdir, readFile, writeFile } from 'node:fs/promises'
2
- import { homedir } from 'node:os'
3
- import { dirname, join, resolve } from 'node:path'
4
- import { type GlobalConfig, GlobalConfigSchema, parseArk } from './schema/index.js'
5
-
6
- export function getGlobalConfigPath() {
7
- const override = process.env.PILOTHUB_CONFIG_PATH?.trim()
8
- if (override) return resolve(override)
9
- const home = homedir()
10
- if (process.platform === 'darwin') {
11
- return join(home, 'Library', 'Application Support', 'pilothub', 'config.json')
12
- }
13
- const xdg = process.env.XDG_CONFIG_HOME
14
- if (xdg) return join(xdg, 'pilothub', 'config.json')
15
- if (process.platform === 'win32') {
16
- const appData = process.env.APPDATA
17
- if (appData) return join(appData, 'pilothub', 'config.json')
18
- }
19
- return join(home, '.config', 'pilothub', 'config.json')
20
- }
21
-
22
- export async function readGlobalConfig(): Promise<GlobalConfig | null> {
23
- try {
24
- const raw = await readFile(getGlobalConfigPath(), 'utf8')
25
- const parsed = JSON.parse(raw) as unknown
26
- return parseArk(GlobalConfigSchema, parsed, 'Global config')
27
- } catch {
28
- return null
29
- }
30
- }
31
-
32
- export async function writeGlobalConfig(config: GlobalConfig) {
33
- const path = getGlobalConfigPath()
34
- await mkdir(dirname(path), { recursive: true })
35
- await writeFile(path, `${JSON.stringify(config, null, 2)}\n`, 'utf8')
36
- }
@@ -1,75 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { afterEach, describe, expect, it, vi } from 'vitest'
4
- import { discoverRegistryFromSite } from './discovery'
5
-
6
- describe('discovery', () => {
7
- afterEach(() => {
8
- vi.unstubAllGlobals()
9
- })
10
-
11
- it('returns null on non-ok response', async () => {
12
- vi.stubGlobal(
13
- 'fetch',
14
- vi.fn(async () => new Response('nope', { status: 404 })) as unknown as typeof fetch,
15
- )
16
- await expect(discoverRegistryFromSite('https://example.com')).resolves.toBeNull()
17
- })
18
-
19
- it('parses registry config', async () => {
20
- vi.stubGlobal(
21
- 'fetch',
22
- vi.fn(
23
- async () =>
24
- new Response(JSON.stringify({ registry: 'https://example.convex.site' }), {
25
- status: 200,
26
- headers: { 'Content-Type': 'application/json' },
27
- }),
28
- ) as unknown as typeof fetch,
29
- )
30
- await expect(discoverRegistryFromSite('https://example.com')).resolves.toEqual({
31
- apiBase: 'https://example.convex.site',
32
- authBase: undefined,
33
- minCliVersion: undefined,
34
- })
35
- })
36
-
37
- it('parses apiBase config', async () => {
38
- vi.stubGlobal(
39
- 'fetch',
40
- vi.fn(
41
- async () =>
42
- new Response(
43
- JSON.stringify({
44
- apiBase: 'https://api.example.com',
45
- authBase: 'https://auth.example.com',
46
- minCliVersion: '1.2.3',
47
- }),
48
- {
49
- status: 200,
50
- headers: { 'Content-Type': 'application/json' },
51
- },
52
- ),
53
- ) as unknown as typeof fetch,
54
- )
55
- await expect(discoverRegistryFromSite('https://example.com')).resolves.toEqual({
56
- apiBase: 'https://api.example.com',
57
- authBase: 'https://auth.example.com',
58
- minCliVersion: '1.2.3',
59
- })
60
- })
61
-
62
- it('returns null when apiBase is empty', async () => {
63
- vi.stubGlobal(
64
- 'fetch',
65
- vi.fn(
66
- async () =>
67
- new Response(JSON.stringify({ apiBase: '' }), {
68
- status: 200,
69
- headers: { 'Content-Type': 'application/json' },
70
- }),
71
- ) as unknown as typeof fetch,
72
- )
73
- await expect(discoverRegistryFromSite('https://example.com')).resolves.toBeNull()
74
- })
75
- })
@@ -1,19 +0,0 @@
1
- import { parseArk, WellKnownConfigSchema } from './schema/index.js'
2
-
3
- export async function discoverRegistryFromSite(siteUrl: string) {
4
- const url = new URL('/.well-known/pilothub.json', siteUrl)
5
- const response = await fetch(url.toString(), {
6
- method: 'GET',
7
- headers: { Accept: 'application/json' },
8
- })
9
- if (!response.ok) return null
10
- const raw = (await response.json()) as unknown
11
- const parsed = parseArk(WellKnownConfigSchema, raw, 'WellKnown config')
12
- const apiBase = 'apiBase' in parsed ? parsed.apiBase : parsed.registry
13
- if (!apiBase) return null
14
- return {
15
- apiBase,
16
- authBase: parsed.authBase,
17
- minCliVersion: parsed.minCliVersion,
18
- }
19
- }
@@ -1,156 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { describe, expect, it, vi } from 'vitest'
4
- import { apiRequest, apiRequestForm, downloadZip } from './http.js'
5
- import { ApiV1WhoamiResponseSchema } from './schema/index.js'
6
-
7
- describe('apiRequest', () => {
8
- it('adds bearer token and parses json', async () => {
9
- const fetchMock = vi.fn().mockResolvedValue({
10
- ok: true,
11
- json: async () => ({ user: { handle: null } }),
12
- })
13
- vi.stubGlobal('fetch', fetchMock)
14
- const result = await apiRequest(
15
- 'https://example.com',
16
- { method: 'GET', path: '/x', token: 'clh_token' },
17
- ApiV1WhoamiResponseSchema,
18
- )
19
- expect(result.user.handle).toBeNull()
20
- expect(fetchMock).toHaveBeenCalledTimes(1)
21
- const [, init] = fetchMock.mock.calls[0] as [string, RequestInit]
22
- expect((init.headers as Record<string, string>).Authorization).toBe('Bearer clh_token')
23
- vi.unstubAllGlobals()
24
- })
25
-
26
- it('posts json body', async () => {
27
- const fetchMock = vi.fn().mockResolvedValue({
28
- ok: true,
29
- json: async () => ({ ok: true }),
30
- })
31
- vi.stubGlobal('fetch', fetchMock)
32
- await apiRequest('https://example.com', {
33
- method: 'POST',
34
- path: '/x',
35
- body: { a: 1 },
36
- })
37
- const [url, init] = fetchMock.mock.calls[0] as [string, RequestInit]
38
- expect(url).toBe('https://example.com/x')
39
- expect(init.body).toBe(JSON.stringify({ a: 1 }))
40
- expect((init.headers as Record<string, string>)['Content-Type']).toBe('application/json')
41
- vi.unstubAllGlobals()
42
- })
43
-
44
- it('throws text body on non-200', async () => {
45
- const fetchMock = vi.fn().mockResolvedValue({
46
- ok: false,
47
- status: 400,
48
- text: async () => 'bad',
49
- })
50
- vi.stubGlobal('fetch', fetchMock)
51
- await expect(apiRequest('https://example.com', { method: 'GET', path: '/x' })).rejects.toThrow(
52
- 'bad',
53
- )
54
- vi.unstubAllGlobals()
55
- })
56
-
57
- it('falls back to HTTP status when body is empty', async () => {
58
- const fetchMock = vi.fn().mockResolvedValue({
59
- ok: false,
60
- status: 500,
61
- text: async () => '',
62
- })
63
- vi.stubGlobal('fetch', fetchMock)
64
- await expect(
65
- apiRequest('https://example.com', { method: 'GET', url: 'https://example.com/x' }),
66
- ).rejects.toThrow('HTTP 500')
67
- vi.unstubAllGlobals()
68
- })
69
-
70
- it('downloads zip bytes', async () => {
71
- const fetchMock = vi.fn().mockResolvedValue({
72
- ok: true,
73
- arrayBuffer: async () => new Uint8Array([1, 2, 3]).buffer,
74
- })
75
- vi.stubGlobal('fetch', fetchMock)
76
- const bytes = await downloadZip('https://example.com', { slug: 'demo', version: '1.0.0' })
77
- expect(Array.from(bytes)).toEqual([1, 2, 3])
78
- const [url] = fetchMock.mock.calls[0] as [string]
79
- expect(url).toContain('slug=demo')
80
- expect(url).toContain('version=1.0.0')
81
- vi.unstubAllGlobals()
82
- })
83
-
84
- it('does not retry on non-retryable errors', async () => {
85
- const fetchMock = vi.fn().mockResolvedValue({
86
- ok: false,
87
- status: 404,
88
- text: async () => 'nope',
89
- })
90
- vi.stubGlobal('fetch', fetchMock)
91
- await expect(downloadZip('https://example.com', { slug: 'demo' })).rejects.toThrow('nope')
92
- expect(fetchMock).toHaveBeenCalledTimes(1)
93
- vi.unstubAllGlobals()
94
- })
95
- })
96
-
97
- describe('apiRequestForm', () => {
98
- it('posts form data and returns json', async () => {
99
- const fetchMock = vi.fn().mockResolvedValue({
100
- ok: true,
101
- json: async () => ({ ok: true }),
102
- })
103
- vi.stubGlobal('fetch', fetchMock)
104
- const form = new FormData()
105
- form.append('x', '1')
106
- const result = await apiRequestForm('https://example.com', {
107
- method: 'POST',
108
- path: '/upload',
109
- token: 'clh_token',
110
- form,
111
- })
112
- expect(result).toEqual({ ok: true })
113
- const [, init] = fetchMock.mock.calls[0] as [string, RequestInit]
114
- expect(init.body).toBe(form)
115
- expect((init.headers as Record<string, string>).Authorization).toBe('Bearer clh_token')
116
- vi.unstubAllGlobals()
117
- })
118
-
119
- it('retries on 429', async () => {
120
- const fetchMock = vi.fn().mockResolvedValue({
121
- ok: false,
122
- status: 429,
123
- text: async () => 'rate limited',
124
- })
125
- vi.stubGlobal('fetch', fetchMock)
126
- await expect(
127
- apiRequestForm('https://example.com', {
128
- method: 'POST',
129
- path: '/upload',
130
- form: new FormData(),
131
- }),
132
- ).rejects.toThrow('rate limited')
133
- expect(fetchMock).toHaveBeenCalledTimes(3)
134
- vi.unstubAllGlobals()
135
- })
136
-
137
- it('falls back to HTTP status when body cannot be read', async () => {
138
- const fetchMock = vi.fn().mockResolvedValue({
139
- ok: false,
140
- status: 400,
141
- text: async () => {
142
- throw new Error('boom')
143
- },
144
- })
145
- vi.stubGlobal('fetch', fetchMock)
146
- await expect(
147
- apiRequestForm('https://example.com', {
148
- method: 'POST',
149
- path: '/upload',
150
- form: new FormData(),
151
- }),
152
- ).rejects.toThrow('HTTP 400')
153
- expect(fetchMock).toHaveBeenCalledTimes(1)
154
- vi.unstubAllGlobals()
155
- })
156
- })