pilothub 0.0.1

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 (272) hide show
  1. package/.env.local.example +19 -0
  2. package/.github/workflows/ci.yml +40 -0
  3. package/.oxlintrc.json +3 -0
  4. package/AGENTS.md +45 -0
  5. package/CHANGELOG.md +138 -0
  6. package/DEPRECATIONS.md +7 -0
  7. package/LICENSE +21 -0
  8. package/README.md +150 -0
  9. package/biome.json +41 -0
  10. package/convex/_generated/api.d.ts +153 -0
  11. package/convex/_generated/api.js +23 -0
  12. package/convex/_generated/dataModel.d.ts +60 -0
  13. package/convex/_generated/server.d.ts +143 -0
  14. package/convex/_generated/server.js +93 -0
  15. package/convex/auth.config.ts +8 -0
  16. package/convex/auth.ts +19 -0
  17. package/convex/comments.ts +88 -0
  18. package/convex/crons.ts +34 -0
  19. package/convex/devSeed.ts +459 -0
  20. package/convex/devSeedExtra.ts +541 -0
  21. package/convex/downloads.ts +78 -0
  22. package/convex/githubBackups.ts +170 -0
  23. package/convex/githubBackupsNode.ts +183 -0
  24. package/convex/githubImport.ts +317 -0
  25. package/convex/githubSoulBackups.ts +170 -0
  26. package/convex/githubSoulBackupsNode.ts +186 -0
  27. package/convex/http.ts +194 -0
  28. package/convex/httpApi.handlers.test.ts +488 -0
  29. package/convex/httpApi.test.ts +70 -0
  30. package/convex/httpApi.ts +305 -0
  31. package/convex/httpApiV1.handlers.test.ts +584 -0
  32. package/convex/httpApiV1.ts +1172 -0
  33. package/convex/leaderboards.ts +39 -0
  34. package/convex/lib/access.ts +36 -0
  35. package/convex/lib/apiTokenAuth.ts +36 -0
  36. package/convex/lib/badges.ts +50 -0
  37. package/convex/lib/changelog.test.ts +34 -0
  38. package/convex/lib/changelog.ts +278 -0
  39. package/convex/lib/embeddings.ts +38 -0
  40. package/convex/lib/githubBackup.ts +443 -0
  41. package/convex/lib/githubImport.test.ts +247 -0
  42. package/convex/lib/githubImport.ts +425 -0
  43. package/convex/lib/githubSoulBackup.ts +443 -0
  44. package/convex/lib/leaderboards.ts +103 -0
  45. package/convex/lib/moderation.ts +42 -0
  46. package/convex/lib/public.ts +89 -0
  47. package/convex/lib/searchText.test.ts +46 -0
  48. package/convex/lib/searchText.ts +27 -0
  49. package/convex/lib/skillBackfill.test.ts +34 -0
  50. package/convex/lib/skillBackfill.ts +67 -0
  51. package/convex/lib/skillPublish.test.ts +28 -0
  52. package/convex/lib/skillPublish.ts +284 -0
  53. package/convex/lib/skillStats.ts +80 -0
  54. package/convex/lib/skills.test.ts +197 -0
  55. package/convex/lib/skills.ts +273 -0
  56. package/convex/lib/soulChangelog.ts +273 -0
  57. package/convex/lib/soulPublish.ts +236 -0
  58. package/convex/lib/tokens.test.ts +33 -0
  59. package/convex/lib/tokens.ts +51 -0
  60. package/convex/lib/webhooks.test.ts +91 -0
  61. package/convex/lib/webhooks.ts +112 -0
  62. package/convex/maintenance.test.ts +270 -0
  63. package/convex/maintenance.ts +840 -0
  64. package/convex/rateLimits.ts +50 -0
  65. package/convex/schema.ts +472 -0
  66. package/convex/search.test.ts +12 -0
  67. package/convex/search.ts +254 -0
  68. package/convex/seed.test.ts +37 -0
  69. package/convex/seed.ts +254 -0
  70. package/convex/seedSouls.ts +111 -0
  71. package/convex/skillStatEvents.ts +568 -0
  72. package/convex/skills.ts +1606 -0
  73. package/convex/soulComments.ts +88 -0
  74. package/convex/soulDownloads.ts +14 -0
  75. package/convex/soulStars.ts +71 -0
  76. package/convex/souls.ts +570 -0
  77. package/convex/stars.ts +108 -0
  78. package/convex/statsMaintenance.ts +205 -0
  79. package/convex/telemetry.ts +434 -0
  80. package/convex/tokens.ts +88 -0
  81. package/convex/tsconfig.json +7 -0
  82. package/convex/uploads.ts +20 -0
  83. package/convex/users.ts +122 -0
  84. package/convex/webhooks.ts +50 -0
  85. package/convex.json +3 -0
  86. package/docs/README.md +32 -0
  87. package/docs/api.md +51 -0
  88. package/docs/architecture.md +61 -0
  89. package/docs/auth.md +54 -0
  90. package/docs/cli.md +117 -0
  91. package/docs/deploy.md +78 -0
  92. package/docs/diffing.md +84 -0
  93. package/docs/github-import.md +171 -0
  94. package/docs/http-api.md +187 -0
  95. package/docs/manual-testing.md +64 -0
  96. package/docs/mintlify.md +43 -0
  97. package/docs/quickstart.md +120 -0
  98. package/docs/skill-format.md +58 -0
  99. package/docs/soul-format.md +37 -0
  100. package/docs/spec.md +177 -0
  101. package/docs/telemetry.md +91 -0
  102. package/docs/troubleshooting.md +49 -0
  103. package/docs/webhook.md +51 -0
  104. package/e2e/menu-smoke.pw.test.ts +49 -0
  105. package/e2e/pilothub.e2e.test.ts +494 -0
  106. package/e2e/search-exact.pw.test.ts +97 -0
  107. package/package.json +84 -0
  108. package/packages/pilothub/LICENSE +22 -0
  109. package/packages/pilothub/README.md +57 -0
  110. package/packages/pilothub/bin/pilothub.js +2 -0
  111. package/packages/pilothub/package.json +41 -0
  112. package/packages/pilothub/src/browserAuth.test.ts +96 -0
  113. package/packages/pilothub/src/browserAuth.ts +174 -0
  114. package/packages/pilothub/src/cli/buildInfo.ts +94 -0
  115. package/packages/pilothub/src/cli/commands/auth.ts +97 -0
  116. package/packages/pilothub/src/cli/commands/delete.test.ts +73 -0
  117. package/packages/pilothub/src/cli/commands/delete.ts +83 -0
  118. package/packages/pilothub/src/cli/commands/publish.test.ts +122 -0
  119. package/packages/pilothub/src/cli/commands/publish.ts +108 -0
  120. package/packages/pilothub/src/cli/commands/skills.test.ts +191 -0
  121. package/packages/pilothub/src/cli/commands/skills.ts +380 -0
  122. package/packages/pilothub/src/cli/commands/star.ts +46 -0
  123. package/packages/pilothub/src/cli/commands/sync.test.ts +310 -0
  124. package/packages/pilothub/src/cli/commands/sync.ts +200 -0
  125. package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +26 -0
  126. package/packages/pilothub/src/cli/commands/syncHelpers.ts +427 -0
  127. package/packages/pilothub/src/cli/commands/syncTypes.ts +27 -0
  128. package/packages/pilothub/src/cli/commands/unstar.ts +48 -0
  129. package/packages/pilothub/src/cli/helpStyle.ts +45 -0
  130. package/packages/pilothub/src/cli/pilotbotConfig.test.ts +159 -0
  131. package/packages/pilothub/src/cli/pilotbotConfig.ts +147 -0
  132. package/packages/pilothub/src/cli/registry.test.ts +63 -0
  133. package/packages/pilothub/src/cli/registry.ts +43 -0
  134. package/packages/pilothub/src/cli/scanSkills.test.ts +64 -0
  135. package/packages/pilothub/src/cli/scanSkills.ts +84 -0
  136. package/packages/pilothub/src/cli/slug.ts +16 -0
  137. package/packages/pilothub/src/cli/types.ts +12 -0
  138. package/packages/pilothub/src/cli/ui.ts +75 -0
  139. package/packages/pilothub/src/cli.ts +311 -0
  140. package/packages/pilothub/src/config.ts +36 -0
  141. package/packages/pilothub/src/discovery.test.ts +75 -0
  142. package/packages/pilothub/src/discovery.ts +19 -0
  143. package/packages/pilothub/src/http.test.ts +156 -0
  144. package/packages/pilothub/src/http.ts +301 -0
  145. package/packages/pilothub/src/schema/ark.ts +29 -0
  146. package/packages/pilothub/src/schema/index.ts +5 -0
  147. package/packages/pilothub/src/schema/routes.ts +22 -0
  148. package/packages/pilothub/src/schema/schemas.ts +260 -0
  149. package/packages/pilothub/src/schema/textFiles.test.ts +23 -0
  150. package/packages/pilothub/src/schema/textFiles.ts +66 -0
  151. package/packages/pilothub/src/skills.test.ts +191 -0
  152. package/packages/pilothub/src/skills.ts +172 -0
  153. package/packages/pilothub/src/types.ts +10 -0
  154. package/packages/pilothub/tsconfig.json +14 -0
  155. package/packages/schema/README.md +3 -0
  156. package/packages/schema/dist/ark.d.ts +4 -0
  157. package/packages/schema/dist/ark.js +26 -0
  158. package/packages/schema/dist/ark.js.map +1 -0
  159. package/packages/schema/dist/index.d.ts +5 -0
  160. package/packages/schema/dist/index.js +5 -0
  161. package/packages/schema/dist/index.js.map +1 -0
  162. package/packages/schema/dist/routes.d.ts +21 -0
  163. package/packages/schema/dist/routes.js +22 -0
  164. package/packages/schema/dist/routes.js.map +1 -0
  165. package/packages/schema/dist/schemas.d.ts +297 -0
  166. package/packages/schema/dist/schemas.js +243 -0
  167. package/packages/schema/dist/schemas.js.map +1 -0
  168. package/packages/schema/dist/textFiles.d.ts +5 -0
  169. package/packages/schema/dist/textFiles.js +66 -0
  170. package/packages/schema/dist/textFiles.js.map +1 -0
  171. package/packages/schema/package.json +26 -0
  172. package/packages/schema/src/ark.ts +29 -0
  173. package/packages/schema/src/index.ts +5 -0
  174. package/packages/schema/src/routes.ts +22 -0
  175. package/packages/schema/src/schemas.test.ts +123 -0
  176. package/packages/schema/src/schemas.ts +287 -0
  177. package/packages/schema/src/textFiles.test.ts +23 -0
  178. package/packages/schema/src/textFiles.ts +66 -0
  179. package/packages/schema/tsconfig.json +15 -0
  180. package/pilothub +46 -0
  181. package/playwright.config.ts +33 -0
  182. package/public/.well-known/pilothub.json +6 -0
  183. package/public/api/v1/openapi.json +379 -0
  184. package/public/favicon.ico +0 -0
  185. package/public/logo192.png +0 -0
  186. package/public/logo512.png +0 -0
  187. package/public/manifest.json +25 -0
  188. package/public/og.png +0 -0
  189. package/public/og.svg +98 -0
  190. package/public/pilot-logo.png +0 -0
  191. package/public/pilot-mark.png +0 -0
  192. package/public/robots.txt +3 -0
  193. package/public/tanstack-circle-logo.png +0 -0
  194. package/public/tanstack-word-logo-white.svg +1 -0
  195. package/scripts/check-peer-deps.ts +56 -0
  196. package/scripts/docs-list.ts +148 -0
  197. package/scripts/run-playwright-local.sh +14 -0
  198. package/server/og/fetchSkillOgMeta.ts +27 -0
  199. package/server/og/fetchSoulOgMeta.ts +27 -0
  200. package/server/og/ogAssets.ts +80 -0
  201. package/server/og/skillOgSvg.test.ts +59 -0
  202. package/server/og/skillOgSvg.ts +258 -0
  203. package/server/og/soulOgSvg.ts +209 -0
  204. package/server/routes/og/skill.png.ts +103 -0
  205. package/server/routes/og/soul.png.ts +111 -0
  206. package/src/__tests__/skill-detail-page.test.tsx +86 -0
  207. package/src/__tests__/skills-index.test.tsx +145 -0
  208. package/src/__tests__/upload.route.test.tsx +228 -0
  209. package/src/components/AppProviders.tsx +19 -0
  210. package/src/components/ClientOnly.tsx +18 -0
  211. package/src/components/Footer.tsx +29 -0
  212. package/src/components/Header.tsx +295 -0
  213. package/src/components/InstallSwitcher.tsx +53 -0
  214. package/src/components/SkillCard.tsx +36 -0
  215. package/src/components/SkillDetailPage.tsx +817 -0
  216. package/src/components/SkillDiffCard.tsx +485 -0
  217. package/src/components/SoulCard.tsx +19 -0
  218. package/src/components/SoulDetailPage.tsx +263 -0
  219. package/src/components/UserBootstrap.tsx +18 -0
  220. package/src/components/ui/dropdown-menu.tsx +67 -0
  221. package/src/components/ui/toggle-group.tsx +35 -0
  222. package/src/convex/client.ts +3 -0
  223. package/src/lib/badges.ts +29 -0
  224. package/src/lib/diffing.test.ts +163 -0
  225. package/src/lib/diffing.ts +106 -0
  226. package/src/lib/gravatar.test.ts +9 -0
  227. package/src/lib/gravatar.ts +158 -0
  228. package/src/lib/og.test.ts +142 -0
  229. package/src/lib/og.ts +156 -0
  230. package/src/lib/publicUser.ts +39 -0
  231. package/src/lib/roles.ts +19 -0
  232. package/src/lib/site.test.ts +130 -0
  233. package/src/lib/site.ts +84 -0
  234. package/src/lib/theme-transition.test.ts +134 -0
  235. package/src/lib/theme-transition.ts +134 -0
  236. package/src/lib/theme.test.tsx +88 -0
  237. package/src/lib/theme.ts +43 -0
  238. package/src/lib/uploadFiles.jsdom.test.ts +33 -0
  239. package/src/lib/uploadFiles.test.ts +123 -0
  240. package/src/lib/uploadFiles.ts +245 -0
  241. package/src/lib/uploadUtils.test.ts +78 -0
  242. package/src/lib/uploadUtils.ts +93 -0
  243. package/src/lib/useAuthStatus.ts +12 -0
  244. package/src/lib/utils.test.ts +9 -0
  245. package/src/lib/utils.ts +6 -0
  246. package/src/logo.svg +12 -0
  247. package/src/routeTree.gen.ts +345 -0
  248. package/src/router.tsx +17 -0
  249. package/src/routes/$owner/$slug.tsx +55 -0
  250. package/src/routes/__root.tsx +136 -0
  251. package/src/routes/admin.tsx +11 -0
  252. package/src/routes/cli/auth.tsx +168 -0
  253. package/src/routes/dashboard.tsx +97 -0
  254. package/src/routes/import.tsx +415 -0
  255. package/src/routes/index.tsx +252 -0
  256. package/src/routes/management.tsx +529 -0
  257. package/src/routes/settings.tsx +203 -0
  258. package/src/routes/skills/index.tsx +422 -0
  259. package/src/routes/souls/$slug.tsx +55 -0
  260. package/src/routes/souls/index.tsx +243 -0
  261. package/src/routes/stars.tsx +68 -0
  262. package/src/routes/u/$handle.tsx +307 -0
  263. package/src/routes/upload/utils.ts +81 -0
  264. package/src/routes/upload.tsx +499 -0
  265. package/src/styles.css +2718 -0
  266. package/tsconfig.json +24 -0
  267. package/tsconfig.oxlint.json +16 -0
  268. package/vercel.json +8 -0
  269. package/vite.config.ts +48 -0
  270. package/vitest.config.ts +47 -0
  271. package/vitest.e2e.config.ts +11 -0
  272. package/vitest.setup.ts +1 -0
@@ -0,0 +1,311 @@
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
+ })
@@ -0,0 +1,36 @@
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
+ }
@@ -0,0 +1,75 @@
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
+ })
@@ -0,0 +1,19 @@
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
+ }
@@ -0,0 +1,156 @@
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
+ })