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,147 +0,0 @@
1
- import { readFile } from 'node:fs/promises'
2
- import { homedir } from 'node:os'
3
- import { basename, join, resolve } from 'node:path'
4
- import JSON5 from 'json5'
5
-
6
- type PilotbotConfig = {
7
- agent?: { workspace?: string }
8
- agents?: {
9
- defaults?: { workspace?: string }
10
- list?: Array<{
11
- id?: string
12
- name?: string
13
- workspace?: string
14
- default?: boolean
15
- }>
16
- }
17
- routing?: {
18
- agents?: Record<
19
- string,
20
- {
21
- name?: string
22
- workspace?: string
23
- }
24
- >
25
- }
26
- skills?: {
27
- load?: {
28
- extraDirs?: string[]
29
- }
30
- }
31
- }
32
-
33
- export type PilotbotSkillRoots = {
34
- roots: string[]
35
- labels: Record<string, string>
36
- }
37
-
38
- export async function resolvePilotbotSkillRoots(): Promise<PilotbotSkillRoots> {
39
- const roots: string[] = []
40
- const labels: Record<string, string> = {}
41
-
42
- const stateDir = resolvePilotbotStateDir()
43
- const sharedSkills = resolveUserPath(join(stateDir, 'skills'))
44
- pushRoot(roots, labels, sharedSkills, 'Shared skills')
45
-
46
- const config = await readPilotbotConfig()
47
- if (!config) return { roots, labels }
48
-
49
- const mainWorkspace = resolveUserPath(
50
- config.agents?.defaults?.workspace ?? config.agent?.workspace ?? '',
51
- )
52
- if (mainWorkspace) {
53
- pushRoot(roots, labels, join(mainWorkspace, 'skills'), 'Agent: main')
54
- }
55
-
56
- const listedAgents = config.agents?.list ?? []
57
- for (const entry of listedAgents) {
58
- const workspace = resolveUserPath(entry?.workspace ?? '')
59
- if (!workspace) continue
60
- const name = entry?.name?.trim() || entry?.id?.trim() || 'agent'
61
- pushRoot(roots, labels, join(workspace, 'skills'), `Agent: ${name}`)
62
- }
63
-
64
- const agents = config.routing?.agents ?? {}
65
- for (const [agentId, entry] of Object.entries(agents)) {
66
- const workspace = resolveUserPath(entry?.workspace ?? '')
67
- if (!workspace) continue
68
- const name = entry?.name?.trim() || agentId
69
- pushRoot(roots, labels, join(workspace, 'skills'), `Agent: ${name}`)
70
- }
71
-
72
- const extraDirs = config.skills?.load?.extraDirs ?? []
73
- for (const dir of extraDirs) {
74
- const resolved = resolveUserPath(String(dir))
75
- if (!resolved) continue
76
- const label = `Extra: ${basename(resolved) || resolved}`
77
- pushRoot(roots, labels, resolved, label)
78
- }
79
-
80
- return { roots, labels }
81
- }
82
-
83
- export async function resolvePilotbotDefaultWorkspace(): Promise<string | null> {
84
- const config = await readPilotbotConfig()
85
- if (!config) return null
86
-
87
- const defaultsWorkspace = resolveUserPath(
88
- config.agents?.defaults?.workspace ?? config.agent?.workspace ?? '',
89
- )
90
- if (defaultsWorkspace) return defaultsWorkspace
91
-
92
- const listedAgents = config.agents?.list ?? []
93
- const defaultAgent =
94
- listedAgents.find((entry) => entry.default) ?? listedAgents.find((entry) => entry.id === 'main')
95
- const listWorkspace = resolveUserPath(defaultAgent?.workspace ?? '')
96
- return listWorkspace || null
97
- }
98
-
99
- function resolvePilotbotStateDir() {
100
- const override = process.env.PILOTBOT_STATE_DIR?.trim()
101
- if (override) return resolveUserPath(override)
102
- return join(homedir(), '.pilotbot')
103
- }
104
-
105
- function resolvePilotbotConfigPath() {
106
- const override = process.env.PILOTBOT_CONFIG_PATH?.trim()
107
- if (override) return resolveUserPath(override)
108
- return join(resolvePilotbotStateDir(), 'pilotbot.json')
109
- }
110
-
111
- function resolveUserPath(input: string) {
112
- const trimmed = input.trim()
113
- if (!trimmed) return ''
114
- if (trimmed.startsWith('~')) {
115
- return resolve(trimmed.replace(/^~(?=$|[\\/])/, homedir()))
116
- }
117
- return resolve(trimmed)
118
- }
119
-
120
- async function readPilotbotConfig(): Promise<PilotbotConfig | null> {
121
- try {
122
- const raw = await readFile(resolvePilotbotConfigPath(), 'utf8')
123
- const parsed = JSON5.parse(raw)
124
- if (!parsed || typeof parsed !== 'object') return null
125
- return parsed as PilotbotConfig
126
- } catch {
127
- return null
128
- }
129
- }
130
-
131
- function pushRoot(roots: string[], labels: Record<string, string>, root: string, label?: string) {
132
- const resolved = resolveUserPath(root)
133
- if (!resolved) return
134
- if (!roots.includes(resolved)) roots.push(resolved)
135
- if (!label) return
136
- const existing = labels[resolved]
137
- if (!existing) {
138
- labels[resolved] = label
139
- return
140
- }
141
- const parts = existing
142
- .split(', ')
143
- .map((part) => part.trim())
144
- .filter(Boolean)
145
- if (parts.includes(label)) return
146
- labels[resolved] = `${existing}, ${label}`
147
- }
@@ -1,63 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { beforeEach, describe, expect, it, vi } from 'vitest'
4
- import type { GlobalOpts } from './types'
5
-
6
- const readGlobalConfig = vi.fn()
7
- const writeGlobalConfig = vi.fn()
8
- const discoverRegistryFromSite = vi.fn()
9
-
10
- vi.mock('../config.js', () => ({
11
- readGlobalConfig: (...args: unknown[]) => readGlobalConfig(...args),
12
- writeGlobalConfig: (...args: unknown[]) => writeGlobalConfig(...args),
13
- }))
14
-
15
- vi.mock('../discovery.js', () => ({
16
- discoverRegistryFromSite: (...args: unknown[]) => discoverRegistryFromSite(...args),
17
- }))
18
-
19
- const { DEFAULT_REGISTRY, getRegistry, resolveRegistry } = await import('./registry')
20
-
21
- function makeOpts(overrides: Partial<GlobalOpts> = {}): GlobalOpts {
22
- return {
23
- workdir: '/work',
24
- dir: '/work/skills',
25
- site: 'https://pilothub.com',
26
- registry: DEFAULT_REGISTRY,
27
- registrySource: 'default',
28
- ...overrides,
29
- }
30
- }
31
-
32
- beforeEach(() => {
33
- readGlobalConfig.mockReset()
34
- writeGlobalConfig.mockReset()
35
- discoverRegistryFromSite.mockReset()
36
- })
37
-
38
- describe('registry resolution', () => {
39
- it('prefers explicit registry over discovery/cache', async () => {
40
- readGlobalConfig.mockResolvedValue({ registry: 'https://auth.pilothub.com' })
41
- discoverRegistryFromSite.mockResolvedValue({ apiBase: 'https://pilothub.com' })
42
-
43
- const registry = await resolveRegistry(
44
- makeOpts({ registry: 'https://custom.example', registrySource: 'cli' }),
45
- )
46
-
47
- expect(registry).toBe('https://custom.example')
48
- expect(discoverRegistryFromSite).not.toHaveBeenCalled()
49
- })
50
-
51
- it('ignores legacy registry and updates cache from discovery', async () => {
52
- readGlobalConfig.mockResolvedValue({ registry: 'https://auth.pilothub.com', token: 'tkn' })
53
- discoverRegistryFromSite.mockResolvedValue({ apiBase: 'https://pilothub.com' })
54
-
55
- const registry = await getRegistry(makeOpts(), { cache: true })
56
-
57
- expect(registry).toBe('https://pilothub.com')
58
- expect(writeGlobalConfig).toHaveBeenCalledWith({
59
- registry: 'https://pilothub.com',
60
- token: 'tkn',
61
- })
62
- })
63
- })
@@ -1,43 +0,0 @@
1
- import { readGlobalConfig, writeGlobalConfig } from '../config.js'
2
- import { discoverRegistryFromSite } from '../discovery.js'
3
- import type { GlobalOpts } from './types.js'
4
-
5
- export const DEFAULT_SITE = 'https://pilothub.com'
6
- export const DEFAULT_REGISTRY = 'https://pilothub.com'
7
- const LEGACY_REGISTRY_HOSTS = new Set(['auth.pilothub.com'])
8
-
9
- export async function resolveRegistry(opts: GlobalOpts) {
10
- const explicit = opts.registrySource !== 'default' ? opts.registry.trim() : ''
11
- if (explicit) return explicit
12
-
13
- const discovery = await discoverRegistryFromSite(opts.site).catch(() => null)
14
- const discovered = discovery?.apiBase?.trim()
15
- if (discovered) return discovered
16
-
17
- const cfg = await readGlobalConfig()
18
- const cached = cfg?.registry?.trim()
19
- if (cached && !isLegacyRegistry(cached)) return cached
20
- return DEFAULT_REGISTRY
21
- }
22
-
23
- export async function getRegistry(opts: GlobalOpts, params?: { cache?: boolean }) {
24
- const cache = params?.cache !== false
25
- const registry = await resolveRegistry(opts)
26
- if (!cache) return registry
27
- const cfg = await readGlobalConfig()
28
- const cached = cfg?.registry?.trim()
29
- const shouldUpdate =
30
- !cached ||
31
- isLegacyRegistry(cached) ||
32
- (cached === DEFAULT_REGISTRY && registry !== DEFAULT_REGISTRY)
33
- if (shouldUpdate) await writeGlobalConfig({ registry, token: cfg?.token })
34
- return registry
35
- }
36
-
37
- function isLegacyRegistry(registry: string) {
38
- try {
39
- return LEGACY_REGISTRY_HOSTS.has(new URL(registry).hostname)
40
- } catch {
41
- return false
42
- }
43
- }
@@ -1,64 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { mkdir, mkdtemp, rm, writeFile } from 'node:fs/promises'
4
- import { tmpdir } from 'node:os'
5
- import { join, resolve } from 'node:path'
6
- import { describe, expect, it } from 'vitest'
7
- import { findSkillFolders, getFallbackSkillRoots } from './scanSkills'
8
-
9
- async function makeTmpDir() {
10
- return mkdtemp(join(tmpdir(), 'pilothub-scan-'))
11
- }
12
-
13
- describe('scanSkills', () => {
14
- it('detects a single skill folder (root contains SKILL.md)', async () => {
15
- const root = await makeTmpDir()
16
- try {
17
- await writeFile(join(root, 'SKILL.md'), '# Skill\n', 'utf8')
18
- const found = await findSkillFolders(root)
19
- expect(found).toHaveLength(1)
20
- expect(found[0]?.folder).toBe(resolve(root))
21
- expect(found[0]?.slug).toBeTruthy()
22
- } finally {
23
- await rm(root, { recursive: true, force: true })
24
- }
25
- })
26
-
27
- it('detects skills in a skills directory (subfolders)', async () => {
28
- const root = await makeTmpDir()
29
- try {
30
- const skillsDir = join(root, 'skills')
31
- const folder = join(skillsDir, 'cool-skill')
32
- await mkdir(folder, { recursive: true })
33
- await writeFile(join(folder, 'SKILL.md'), '# Skill\n', 'utf8')
34
-
35
- const found = await findSkillFolders(skillsDir)
36
- expect(found).toHaveLength(1)
37
- expect(found[0]?.slug).toBe('cool-skill')
38
- expect(found[0]?.folder).toBe(resolve(folder))
39
- } finally {
40
- await rm(root, { recursive: true, force: true })
41
- }
42
- })
43
-
44
- it('ignores plural skills.md marker files', async () => {
45
- const root = await makeTmpDir()
46
- try {
47
- const folder = join(root, 'docs')
48
- await mkdir(folder, { recursive: true })
49
- await writeFile(join(folder, 'skills.md'), '# Docs\n', 'utf8')
50
-
51
- const found = await findSkillFolders(root)
52
- expect(found).toHaveLength(0)
53
- } finally {
54
- await rm(root, { recursive: true, force: true })
55
- }
56
- })
57
-
58
- it('includes known legacy roots', () => {
59
- const roots = getFallbackSkillRoots('/tmp/anywhere')
60
- expect(roots.some((p) => p.endsWith('/pilotbot/skills'))).toBe(true)
61
- expect(roots.some((p) => p.endsWith('/pilot/skills'))).toBe(true)
62
- expect(roots.some((p) => p.endsWith('/pilotbot/skills'))).toBe(true)
63
- })
64
- })
@@ -1,84 +0,0 @@
1
- import { readdir, stat } from 'node:fs/promises'
2
- import { homedir } from 'node:os'
3
- import { basename, join, resolve } from 'node:path'
4
- import { sanitizeSlug, titleCase } from './slug.js'
5
-
6
- export type SkillFolder = {
7
- folder: string
8
- slug: string
9
- displayName: string
10
- }
11
-
12
- export async function findSkillFolders(root: string): Promise<SkillFolder[]> {
13
- const absRoot = resolve(root)
14
- const rootStat = await stat(absRoot).catch(() => null)
15
- if (!rootStat || !rootStat.isDirectory()) return []
16
-
17
- const direct = await isSkillFolder(absRoot)
18
- if (direct) return [direct]
19
-
20
- const entries = await readdir(absRoot, { withFileTypes: true }).catch(() => [])
21
- const folders = entries
22
- .filter((entry) => entry.isDirectory())
23
- .map((entry) => join(absRoot, entry.name))
24
- const results: SkillFolder[] = []
25
- for (const folder of folders) {
26
- const found = await isSkillFolder(folder)
27
- if (found) results.push(found)
28
- }
29
- return results.sort((a, b) => a.slug.localeCompare(b.slug))
30
- }
31
-
32
- export function getFallbackSkillRoots(workdir: string) {
33
- const home = homedir()
34
- const roots = [
35
- // adjacent repo installs
36
- resolve(workdir, '..', 'pilotbot', 'skills'),
37
- resolve(workdir, '..', 'pilotbot', 'Skills'),
38
- resolve(workdir, '..', 'pilotbot', 'skills'),
39
- resolve(workdir, '..', 'pilotbot', 'Skills'),
40
-
41
- // legacy locations
42
- resolve(home, 'pilot', 'skills'),
43
- resolve(home, 'pilot', 'Skills'),
44
- resolve(home, '.pilot', 'skills'),
45
- resolve(home, '.pilot', 'Skills'),
46
-
47
- resolve(home, 'pilotbot', 'skills'),
48
- resolve(home, 'pilotbot', 'Skills'),
49
- resolve(home, '.pilotbot', 'skills'),
50
- resolve(home, '.pilotbot', 'Skills'),
51
-
52
- resolve(home, 'pilotbot', 'skills'),
53
- resolve(home, 'pilotbot', 'Skills'),
54
- resolve(home, '.pilotbot', 'skills'),
55
- resolve(home, '.pilotbot', 'Skills'),
56
-
57
- // macOS App Support legacy
58
- resolve(home, 'Library', 'Application Support', 'pilotbot', 'skills'),
59
- resolve(home, 'Library', 'Application Support', 'pilotbot', 'Skills'),
60
- resolve(home, 'Library', 'Application Support', 'pilotbot', 'skills'),
61
- resolve(home, 'Library', 'Application Support', 'pilotbot', 'Skills'),
62
- ]
63
- return Array.from(new Set(roots))
64
- }
65
-
66
- async function isSkillFolder(folder: string): Promise<SkillFolder | null> {
67
- const marker = await findSkillMarker(folder)
68
- if (!marker) return null
69
- const base = basename(folder)
70
- const slug = sanitizeSlug(base)
71
- if (!slug) return null
72
- const displayName = titleCase(base)
73
- return { folder, slug, displayName }
74
- }
75
-
76
- async function findSkillMarker(folder: string) {
77
- const candidates = ['SKILL.md', 'skill.md']
78
- for (const name of candidates) {
79
- const path = join(folder, name)
80
- const st = await stat(path).catch(() => null)
81
- if (st?.isFile()) return path
82
- }
83
- return null
84
- }
@@ -1,16 +0,0 @@
1
- export function sanitizeSlug(value: string) {
2
- const raw = value
3
- .trim()
4
- .toLowerCase()
5
- .replace(/[^a-z0-9-]+/g, '-')
6
- const cleaned = raw.replace(/^-+/, '').replace(/-+$/, '').replace(/--+/g, '-')
7
- return cleaned
8
- }
9
-
10
- export function titleCase(value: string) {
11
- return value
12
- .trim()
13
- .replace(/[-_]+/g, ' ')
14
- .replace(/\s+/g, ' ')
15
- .replace(/\b\w/g, (char) => char.toUpperCase())
16
- }
@@ -1,12 +0,0 @@
1
- export type GlobalOpts = {
2
- workdir: string
3
- dir: string
4
- site: string
5
- registry: string
6
- registrySource: 'cli' | 'env' | 'default'
7
- }
8
-
9
- export type ResolveResult = {
10
- match: { version: string } | null
11
- latestVersion: { version: string } | null
12
- }
@@ -1,75 +0,0 @@
1
- import { spawn } from 'node:child_process'
2
- import { stdin } from 'node:process'
3
- import { confirm, isCancel } from '@clack/prompts'
4
- import ora from 'ora'
5
-
6
- export async function promptHidden(prompt: string) {
7
- if (!stdin.isTTY) return ''
8
- process.stdout.write(prompt)
9
- const chunks: Buffer[] = []
10
- stdin.setRawMode(true)
11
- stdin.resume()
12
- return new Promise<string>((resolvePromise) => {
13
- function onData(data: Buffer) {
14
- const text = data.toString('utf8')
15
- if (text === '\r' || text === '\n') {
16
- stdin.setRawMode(false)
17
- stdin.pause()
18
- stdin.off('data', onData)
19
- process.stdout.write('\n')
20
- resolvePromise(Buffer.concat(chunks).toString('utf8').trim())
21
- return
22
- }
23
- if (text === '\u0003') {
24
- stdin.setRawMode(false)
25
- stdin.pause()
26
- stdin.off('data', onData)
27
- process.stdout.write('\n')
28
- fail('Canceled')
29
- }
30
- if (text === '\u007f') {
31
- chunks.pop()
32
- return
33
- }
34
- chunks.push(data)
35
- }
36
- stdin.on('data', onData)
37
- })
38
- }
39
-
40
- export async function promptConfirm(prompt: string) {
41
- const answer = await confirm({ message: prompt })
42
- if (isCancel(answer)) return false
43
- return Boolean(answer)
44
- }
45
-
46
- export function openInBrowser(url: string) {
47
- const args =
48
- process.platform === 'darwin'
49
- ? ['open', url]
50
- : process.platform === 'win32'
51
- ? ['cmd', '/c', 'start', '', url]
52
- : ['xdg-open', url]
53
- const [command, ...commandArgs] = args
54
- if (!command) return
55
- const child = spawn(command, commandArgs, { stdio: 'ignore', detached: true })
56
- child.unref()
57
- }
58
-
59
- export function isInteractive() {
60
- return Boolean(process.stdout.isTTY && stdin.isTTY)
61
- }
62
-
63
- export function createSpinner(text: string) {
64
- return ora({ text, spinner: 'dots', isEnabled: isInteractive() }).start()
65
- }
66
-
67
- export function formatError(error: unknown) {
68
- if (error instanceof Error) return error.message
69
- return String(error)
70
- }
71
-
72
- export function fail(message: string): never {
73
- console.error(`Error: ${message}`)
74
- process.exit(1)
75
- }