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,301 +0,0 @@
1
- import { spawnSync } from 'node:child_process'
2
- import { mkdtemp, rm, writeFile } from 'node:fs/promises'
3
- import { tmpdir } from 'node:os'
4
- import { join } from 'node:path'
5
- import pRetry, { AbortError } from 'p-retry'
6
- import { Agent, setGlobalDispatcher } from 'undici'
7
- import type { ArkValidator } from './schema/index.js'
8
- import { ApiRoutes, parseArk } from './schema/index.js'
9
-
10
- const REQUEST_TIMEOUT_MS = 15_000
11
- const REQUEST_TIMEOUT_SECONDS = Math.ceil(REQUEST_TIMEOUT_MS / 1000)
12
- const isBun = typeof process !== 'undefined' && Boolean(process.versions?.bun)
13
-
14
- if (typeof process !== 'undefined' && process.versions?.node) {
15
- try {
16
- setGlobalDispatcher(
17
- new Agent({
18
- allowH2: true,
19
- connect: { timeout: REQUEST_TIMEOUT_MS },
20
- }),
21
- )
22
- } catch {
23
- // ignore dispatcher setup failures in non-node runtimes
24
- }
25
- }
26
-
27
- type RequestArgs =
28
- | { method: 'GET' | 'POST' | 'DELETE'; path: string; token?: string; body?: unknown }
29
- | { method: 'GET' | 'POST' | 'DELETE'; url: string; token?: string; body?: unknown }
30
-
31
- export async function apiRequest<T>(registry: string, args: RequestArgs): Promise<T>
32
- export async function apiRequest<T>(
33
- registry: string,
34
- args: RequestArgs,
35
- schema: ArkValidator<T>,
36
- ): Promise<T>
37
- export async function apiRequest<T>(
38
- registry: string,
39
- args: RequestArgs,
40
- schema?: ArkValidator<T>,
41
- ): Promise<T> {
42
- const url = 'url' in args ? args.url : new URL(args.path, registry).toString()
43
- const json = await pRetry(
44
- async () => {
45
- if (isBun) {
46
- return await fetchJsonViaCurl(url, args)
47
- }
48
-
49
- const headers: Record<string, string> = { Accept: 'application/json' }
50
- if (args.token) headers.Authorization = `Bearer ${args.token}`
51
- let body: string | undefined
52
- if (args.method === 'POST') {
53
- headers['Content-Type'] = 'application/json'
54
- body = JSON.stringify(args.body ?? {})
55
- }
56
- const controller = new AbortController()
57
- const timeout = setTimeout(() => controller.abort('Timeout'), REQUEST_TIMEOUT_MS)
58
- const response = await fetch(url, {
59
- method: args.method,
60
- headers,
61
- body,
62
- signal: controller.signal,
63
- })
64
- clearTimeout(timeout)
65
- if (!response.ok) {
66
- const text = await response.text().catch(() => '')
67
- const message = text || `HTTP ${response.status}`
68
- if (response.status === 429 || response.status >= 500) {
69
- throw new Error(message)
70
- }
71
- throw new AbortError(message)
72
- }
73
- return (await response.json()) as unknown
74
- },
75
- { retries: 2 },
76
- )
77
- if (schema) return parseArk(schema, json, 'API response')
78
- return json as T
79
- }
80
-
81
- type FormRequestArgs =
82
- | { method: 'POST'; path: string; token?: string; form: FormData }
83
- | { method: 'POST'; url: string; token?: string; form: FormData }
84
-
85
- export async function apiRequestForm<T>(registry: string, args: FormRequestArgs): Promise<T>
86
- export async function apiRequestForm<T>(
87
- registry: string,
88
- args: FormRequestArgs,
89
- schema: ArkValidator<T>,
90
- ): Promise<T>
91
- export async function apiRequestForm<T>(
92
- registry: string,
93
- args: FormRequestArgs,
94
- schema?: ArkValidator<T>,
95
- ): Promise<T> {
96
- const url = 'url' in args ? args.url : new URL(args.path, registry).toString()
97
- const json = await pRetry(
98
- async () => {
99
- if (isBun) {
100
- return await fetchJsonFormViaCurl(url, args)
101
- }
102
-
103
- const headers: Record<string, string> = { Accept: 'application/json' }
104
- if (args.token) headers.Authorization = `Bearer ${args.token}`
105
- const controller = new AbortController()
106
- const timeout = setTimeout(() => controller.abort('Timeout'), REQUEST_TIMEOUT_MS)
107
- const response = await fetch(url, {
108
- method: args.method,
109
- headers,
110
- body: args.form,
111
- signal: controller.signal,
112
- })
113
- clearTimeout(timeout)
114
- if (!response.ok) {
115
- const text = await response.text().catch(() => '')
116
- const message = text || `HTTP ${response.status}`
117
- if (response.status === 429 || response.status >= 500) {
118
- throw new Error(message)
119
- }
120
- throw new AbortError(message)
121
- }
122
- return (await response.json()) as unknown
123
- },
124
- { retries: 2 },
125
- )
126
- if (schema) return parseArk(schema, json, 'API response')
127
- return json as T
128
- }
129
-
130
- export async function downloadZip(registry: string, args: { slug: string; version?: string }) {
131
- const url = new URL(ApiRoutes.download, registry)
132
- url.searchParams.set('slug', args.slug)
133
- if (args.version) url.searchParams.set('version', args.version)
134
- return pRetry(
135
- async () => {
136
- if (isBun) {
137
- return await fetchBinaryViaCurl(url.toString())
138
- }
139
-
140
- const controller = new AbortController()
141
- const timeout = setTimeout(() => controller.abort('Timeout'), REQUEST_TIMEOUT_MS)
142
- const response = await fetch(url.toString(), { method: 'GET', signal: controller.signal })
143
- clearTimeout(timeout)
144
- if (!response.ok) {
145
- const message = (await response.text().catch(() => '')) || `HTTP ${response.status}`
146
- if (response.status === 429 || response.status >= 500) {
147
- throw new Error(message)
148
- }
149
- throw new AbortError(message)
150
- }
151
- return new Uint8Array(await response.arrayBuffer())
152
- },
153
- { retries: 2 },
154
- )
155
- }
156
-
157
- async function fetchJsonViaCurl(url: string, args: RequestArgs) {
158
- const headers = ['-H', 'Accept: application/json']
159
- if (args.token) {
160
- headers.push('-H', `Authorization: Bearer ${args.token}`)
161
- }
162
- const curlArgs = [
163
- '--silent',
164
- '--show-error',
165
- '--location',
166
- '--max-time',
167
- String(REQUEST_TIMEOUT_SECONDS),
168
- '--write-out',
169
- '\n%{http_code}',
170
- '-X',
171
- args.method,
172
- ...headers,
173
- url,
174
- ]
175
- if (args.method === 'POST') {
176
- curlArgs.push('-H', 'Content-Type: application/json')
177
- curlArgs.push('--data-binary', JSON.stringify(args.body ?? {}))
178
- }
179
-
180
- const result = spawnSync('curl', curlArgs, { encoding: 'utf8' })
181
- if (result.status !== 0) {
182
- throw new Error(result.stderr || 'curl failed')
183
- }
184
- const output = result.stdout ?? ''
185
- const splitAt = output.lastIndexOf('\n')
186
- if (splitAt === -1) throw new Error('curl response missing status')
187
- const body = output.slice(0, splitAt)
188
- const status = Number(output.slice(splitAt + 1).trim())
189
- if (!Number.isFinite(status)) throw new Error('curl response missing status')
190
- if (status < 200 || status >= 300) {
191
- if (status === 429 || status >= 500) {
192
- throw new Error(body || `HTTP ${status}`)
193
- }
194
- throw new AbortError(body || `HTTP ${status}`)
195
- }
196
- return JSON.parse(body || 'null') as unknown
197
- }
198
-
199
- async function fetchJsonFormViaCurl(url: string, args: FormRequestArgs) {
200
- const headers = ['-H', 'Accept: application/json']
201
- if (args.token) {
202
- headers.push('-H', `Authorization: Bearer ${args.token}`)
203
- }
204
-
205
- const tempDir = await mkdtemp(join(tmpdir(), 'pilothub-upload-'))
206
- try {
207
- const formArgs: string[] = []
208
- for (const [key, value] of args.form.entries()) {
209
- if (value instanceof Blob) {
210
- const filename = typeof (value as File).name === 'string' ? (value as File).name : 'file'
211
- const filePath = join(tempDir, filename)
212
- const bytes = new Uint8Array(await value.arrayBuffer())
213
- await writeFile(filePath, bytes)
214
- formArgs.push('-F', `${key}=@${filePath};filename=${filename}`)
215
- } else {
216
- formArgs.push('-F', `${key}=${String(value)}`)
217
- }
218
- }
219
-
220
- const curlArgs = [
221
- '--silent',
222
- '--show-error',
223
- '--location',
224
- '--max-time',
225
- String(REQUEST_TIMEOUT_SECONDS),
226
- '--write-out',
227
- '\n%{http_code}',
228
- '-X',
229
- args.method,
230
- ...headers,
231
- ...formArgs,
232
- url,
233
- ]
234
-
235
- const result = spawnSync('curl', curlArgs, { encoding: 'utf8' })
236
- if (result.status !== 0) {
237
- throw new Error(result.stderr || 'curl failed')
238
- }
239
- const output = result.stdout ?? ''
240
- const splitAt = output.lastIndexOf('\n')
241
- if (splitAt === -1) throw new Error('curl response missing status')
242
- const body = output.slice(0, splitAt)
243
- const status = Number(output.slice(splitAt + 1).trim())
244
- if (!Number.isFinite(status)) throw new Error('curl response missing status')
245
- if (status < 200 || status >= 300) {
246
- if (status === 429 || status >= 500) {
247
- throw new Error(body || `HTTP ${status}`)
248
- }
249
- throw new AbortError(body || `HTTP ${status}`)
250
- }
251
- return JSON.parse(body || 'null') as unknown
252
- } finally {
253
- await rm(tempDir, { recursive: true, force: true })
254
- }
255
- }
256
-
257
- async function fetchBinaryViaCurl(url: string) {
258
- const tempDir = await mkdtemp(join(tmpdir(), 'pilothub-download-'))
259
- const filePath = join(tempDir, 'payload.bin')
260
- try {
261
- const curlArgs = [
262
- '--silent',
263
- '--show-error',
264
- '--location',
265
- '--max-time',
266
- String(REQUEST_TIMEOUT_SECONDS),
267
- '-o',
268
- filePath,
269
- '--write-out',
270
- '%{http_code}',
271
- url,
272
- ]
273
- const result = spawnSync('curl', curlArgs, { encoding: 'utf8' })
274
- if (result.status !== 0) {
275
- throw new Error(result.stderr || 'curl failed')
276
- }
277
- const status = Number((result.stdout ?? '').trim())
278
- if (!Number.isFinite(status)) throw new Error('curl response missing status')
279
- if (status < 200 || status >= 300) {
280
- const body = await readFileSafe(filePath)
281
- const message = body ? new TextDecoder().decode(body) : `HTTP ${status}`
282
- if (status === 429 || status >= 500) {
283
- throw new Error(message)
284
- }
285
- throw new AbortError(message)
286
- }
287
- const bytes = await readFileSafe(filePath)
288
- return bytes ? new Uint8Array(bytes) : new Uint8Array()
289
- } finally {
290
- await rm(tempDir, { recursive: true, force: true })
291
- }
292
- }
293
-
294
- async function readFileSafe(path: string) {
295
- try {
296
- const { readFile } = await import('node:fs/promises')
297
- return await readFile(path)
298
- } catch {
299
- return null
300
- }
301
- }
@@ -1,29 +0,0 @@
1
- import { ArkErrors } from 'arktype'
2
-
3
- export type ArkValidator<T> = (data: unknown) => T | ArkErrors
4
-
5
- export function parseArk<T>(schema: ArkValidator<T>, data: unknown, label: string) {
6
- const result = schema(data)
7
- if (result instanceof ArkErrors) {
8
- throw new Error(`${label}: ${formatArkErrors(result)}`)
9
- }
10
- return result
11
- }
12
-
13
- export function formatArkErrors(errors: ArkErrors) {
14
- const parts: string[] = []
15
- for (const error of errors) {
16
- if (parts.length >= 3) break
17
- const path = Array.isArray(error.path) ? error.path.join('.') : ''
18
- const location = path ? `${path}: ` : ''
19
- const description =
20
- typeof (error as { description?: unknown }).description === 'string'
21
- ? ((error as { description: string }).description as string)
22
- : 'invalid value'
23
- parts.push(`${location}${description}`)
24
- }
25
- if (errors.count > parts.length) {
26
- parts.push(`+${errors.count - parts.length} more`)
27
- }
28
- return parts.join('; ')
29
- }
@@ -1,5 +0,0 @@
1
- export type { ArkValidator } from './ark.js'
2
- export { formatArkErrors, parseArk } from './ark.js'
3
- export { ApiRoutes, LegacyApiRoutes } from './routes.js'
4
- export * from './schemas.js'
5
- export * from './textFiles.js'
@@ -1,22 +0,0 @@
1
- export const LegacyApiRoutes = {
2
- download: '/api/download',
3
- search: '/api/search',
4
- skill: '/api/skill',
5
- skillResolve: '/api/skill/resolve',
6
- cliWhoami: '/api/cli/whoami',
7
- cliUploadUrl: '/api/cli/upload-url',
8
- cliPublish: '/api/cli/publish',
9
- cliTelemetrySync: '/api/cli/telemetry/sync',
10
- cliSkillDelete: '/api/cli/skill/delete',
11
- cliSkillUndelete: '/api/cli/skill/undelete',
12
- } as const
13
-
14
- export const ApiRoutes = {
15
- search: '/api/v1/search',
16
- resolve: '/api/v1/resolve',
17
- download: '/api/v1/download',
18
- skills: '/api/v1/skills',
19
- stars: '/api/v1/stars',
20
- souls: '/api/v1/souls',
21
- whoami: '/api/v1/whoami',
22
- } as const
@@ -1,260 +0,0 @@
1
- import { type inferred, type } from 'arktype'
2
-
3
- export const GlobalConfigSchema = type({
4
- registry: 'string',
5
- token: 'string?',
6
- })
7
- export type GlobalConfig = (typeof GlobalConfigSchema)[inferred]
8
-
9
- export const WellKnownConfigSchema = type({
10
- apiBase: 'string',
11
- authBase: 'string?',
12
- minCliVersion: 'string?',
13
- }).or({
14
- registry: 'string',
15
- authBase: 'string?',
16
- minCliVersion: 'string?',
17
- })
18
- export type WellKnownConfig = (typeof WellKnownConfigSchema)[inferred]
19
-
20
- export const LockfileSchema = type({
21
- version: '1',
22
- skills: {
23
- '[string]': {
24
- version: 'string|null',
25
- installedAt: 'number',
26
- },
27
- },
28
- })
29
- export type Lockfile = (typeof LockfileSchema)[inferred]
30
-
31
- export const ApiCliWhoamiResponseSchema = type({
32
- user: {
33
- handle: 'string|null',
34
- },
35
- })
36
-
37
- export const ApiSearchResponseSchema = type({
38
- results: type({
39
- slug: 'string?',
40
- displayName: 'string?',
41
- version: 'string|null?',
42
- score: 'number',
43
- }).array(),
44
- })
45
-
46
- export const ApiSkillMetaResponseSchema = type({
47
- latestVersion: type({
48
- version: 'string',
49
- }).optional(),
50
- skill: 'unknown|null?',
51
- })
52
-
53
- export const ApiCliUploadUrlResponseSchema = type({
54
- uploadUrl: 'string',
55
- })
56
-
57
- export const ApiUploadFileResponseSchema = type({
58
- storageId: 'string',
59
- })
60
-
61
- export const CliPublishFileSchema = type({
62
- path: 'string',
63
- size: 'number',
64
- storageId: 'string',
65
- sha256: 'string',
66
- contentType: 'string?',
67
- })
68
- export type CliPublishFile = (typeof CliPublishFileSchema)[inferred]
69
-
70
- export const CliPublishRequestSchema = type({
71
- slug: 'string',
72
- displayName: 'string',
73
- version: 'string',
74
- changelog: 'string',
75
- tags: 'string[]?',
76
- forkOf: type({
77
- slug: 'string',
78
- version: 'string?',
79
- }).optional(),
80
- files: CliPublishFileSchema.array(),
81
- })
82
- export type CliPublishRequest = (typeof CliPublishRequestSchema)[inferred]
83
-
84
- export const ApiCliPublishResponseSchema = type({
85
- ok: 'true',
86
- skillId: 'string',
87
- versionId: 'string',
88
- })
89
-
90
- export const CliSkillDeleteRequestSchema = type({
91
- slug: 'string',
92
- })
93
- export type CliSkillDeleteRequest = (typeof CliSkillDeleteRequestSchema)[inferred]
94
-
95
- export const ApiCliSkillDeleteResponseSchema = type({
96
- ok: 'true',
97
- })
98
-
99
- export const ApiSkillResolveResponseSchema = type({
100
- match: type({ version: 'string' }).or('null'),
101
- latestVersion: type({ version: 'string' }).or('null'),
102
- })
103
-
104
- export const CliTelemetrySyncRequestSchema = type({
105
- roots: type({
106
- rootId: 'string',
107
- label: 'string',
108
- skills: type({
109
- slug: 'string',
110
- version: 'string|null?',
111
- }).array(),
112
- }).array(),
113
- })
114
- export type CliTelemetrySyncRequest = (typeof CliTelemetrySyncRequestSchema)[inferred]
115
-
116
- export const ApiCliTelemetrySyncResponseSchema = type({
117
- ok: 'true',
118
- })
119
-
120
- export const ApiV1WhoamiResponseSchema = type({
121
- user: {
122
- handle: 'string|null',
123
- displayName: 'string|null?',
124
- image: 'string|null?',
125
- },
126
- })
127
-
128
- export const ApiV1SearchResponseSchema = type({
129
- results: type({
130
- slug: 'string?',
131
- displayName: 'string?',
132
- summary: 'string|null?',
133
- version: 'string|null?',
134
- score: 'number',
135
- updatedAt: 'number?',
136
- }).array(),
137
- })
138
-
139
- export const ApiV1SkillListResponseSchema = type({
140
- items: type({
141
- slug: 'string',
142
- displayName: 'string',
143
- summary: 'string|null?',
144
- tags: 'unknown',
145
- stats: 'unknown',
146
- createdAt: 'number',
147
- updatedAt: 'number',
148
- latestVersion: type({
149
- version: 'string',
150
- createdAt: 'number',
151
- changelog: 'string',
152
- }).optional(),
153
- }).array(),
154
- nextCursor: 'string|null',
155
- })
156
-
157
- export const ApiV1SkillResponseSchema = type({
158
- skill: type({
159
- slug: 'string',
160
- displayName: 'string',
161
- summary: 'string|null?',
162
- tags: 'unknown',
163
- stats: 'unknown',
164
- createdAt: 'number',
165
- updatedAt: 'number',
166
- }).or('null'),
167
- latestVersion: type({
168
- version: 'string',
169
- createdAt: 'number',
170
- changelog: 'string',
171
- }).or('null'),
172
- owner: type({
173
- handle: 'string|null',
174
- displayName: 'string|null?',
175
- image: 'string|null?',
176
- }).or('null'),
177
- })
178
-
179
- export const ApiV1SkillVersionListResponseSchema = type({
180
- items: type({
181
- version: 'string',
182
- createdAt: 'number',
183
- changelog: 'string',
184
- changelogSource: '"auto"|"user"|null?',
185
- }).array(),
186
- nextCursor: 'string|null',
187
- })
188
-
189
- export const ApiV1SkillVersionResponseSchema = type({
190
- version: type({
191
- version: 'string',
192
- createdAt: 'number',
193
- changelog: 'string',
194
- changelogSource: '"auto"|"user"|null?',
195
- files: 'unknown?',
196
- }).or('null'),
197
- skill: type({
198
- slug: 'string',
199
- displayName: 'string',
200
- }).or('null'),
201
- })
202
-
203
- export const ApiV1SkillResolveResponseSchema = type({
204
- match: type({ version: 'string' }).or('null'),
205
- latestVersion: type({ version: 'string' }).or('null'),
206
- })
207
-
208
- export const ApiV1PublishResponseSchema = type({
209
- ok: 'true',
210
- skillId: 'string',
211
- versionId: 'string',
212
- })
213
-
214
- export const ApiV1DeleteResponseSchema = type({
215
- ok: 'true',
216
- })
217
-
218
- export const ApiV1StarResponseSchema = type({
219
- ok: 'true',
220
- starred: 'boolean',
221
- alreadyStarred: 'boolean',
222
- })
223
-
224
- export const ApiV1UnstarResponseSchema = type({
225
- ok: 'true',
226
- unstarred: 'boolean',
227
- alreadyUnstarred: 'boolean',
228
- })
229
-
230
- export const SkillInstallSpecSchema = type({
231
- id: 'string?',
232
- kind: '"brew"|"node"|"go"|"uv"',
233
- label: 'string?',
234
- bins: 'string[]?',
235
- formula: 'string?',
236
- tap: 'string?',
237
- package: 'string?',
238
- module: 'string?',
239
- })
240
- export type SkillInstallSpec = (typeof SkillInstallSpecSchema)[inferred]
241
-
242
- export const PilotbotRequiresSchema = type({
243
- bins: 'string[]?',
244
- anyBins: 'string[]?',
245
- env: 'string[]?',
246
- config: 'string[]?',
247
- })
248
- export type PilotbotRequires = (typeof PilotbotRequiresSchema)[inferred]
249
-
250
- export const PilotbotSkillMetadataSchema = type({
251
- always: 'boolean?',
252
- skillKey: 'string?',
253
- primaryEnv: 'string?',
254
- emoji: 'string?',
255
- homepage: 'string?',
256
- os: 'string[]?',
257
- requires: PilotbotRequiresSchema.optional(),
258
- install: SkillInstallSpecSchema.array().optional(),
259
- })
260
- export type PilotbotSkillMetadata = (typeof PilotbotSkillMetadataSchema)[inferred]
@@ -1,23 +0,0 @@
1
- /* @vitest-environment node */
2
-
3
- import { describe, expect, it } from 'vitest'
4
- import * as schema from '.'
5
- import { isTextContentType, TEXT_FILE_EXTENSION_SET } from './textFiles'
6
-
7
- describe('packages/pilothub schema textFiles', () => {
8
- it('exports text-file extension set', () => {
9
- expect(TEXT_FILE_EXTENSION_SET.has('md')).toBe(true)
10
- expect(TEXT_FILE_EXTENSION_SET.has('exe')).toBe(false)
11
- })
12
-
13
- it('detects text content types with parameters', () => {
14
- expect(isTextContentType('text/plain; charset=utf-8')).toBe(true)
15
- expect(isTextContentType('application/json; charset=utf-8')).toBe(true)
16
- expect(isTextContentType('application/octet-stream')).toBe(false)
17
- })
18
-
19
- it('re-exports helpers from index', () => {
20
- expect(typeof schema.isTextContentType).toBe('function')
21
- expect(schema.isTextContentType('application/markdown')).toBe(true)
22
- })
23
- })