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,380 @@
1
+ import { mkdir, rm, stat } from 'node:fs/promises'
2
+ import { join } from 'node:path'
3
+ import semver from 'semver'
4
+ import { apiRequest, downloadZip } from '../../http.js'
5
+ import {
6
+ ApiRoutes,
7
+ ApiV1SearchResponseSchema,
8
+ ApiV1SkillListResponseSchema,
9
+ ApiV1SkillResolveResponseSchema,
10
+ ApiV1SkillResponseSchema,
11
+ } from '../../schema/index.js'
12
+ import {
13
+ extractZipToDir,
14
+ hashSkillFiles,
15
+ listTextFiles,
16
+ readLockfile,
17
+ readSkillOrigin,
18
+ writeLockfile,
19
+ writeSkillOrigin,
20
+ } from '../../skills.js'
21
+ import { getRegistry } from '../registry.js'
22
+ import type { GlobalOpts, ResolveResult } from '../types.js'
23
+ import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js'
24
+
25
+ export async function cmdSearch(opts: GlobalOpts, query: string, limit?: number) {
26
+ if (!query) fail('Query required')
27
+
28
+ const registry = await getRegistry(opts, { cache: true })
29
+ const spinner = createSpinner('Searching')
30
+ try {
31
+ const url = new URL(ApiRoutes.search, registry)
32
+ url.searchParams.set('q', query)
33
+ if (typeof limit === 'number' && Number.isFinite(limit)) {
34
+ url.searchParams.set('limit', String(limit))
35
+ }
36
+ const result = await apiRequest(
37
+ registry,
38
+ { method: 'GET', url: url.toString() },
39
+ ApiV1SearchResponseSchema,
40
+ )
41
+
42
+ spinner.stop()
43
+ for (const entry of result.results) {
44
+ const slug = entry.slug ?? 'unknown'
45
+ const name = entry.displayName ?? slug
46
+ const version = entry.version ? ` v${entry.version}` : ''
47
+ console.log(`${slug}${version} ${name} (${entry.score.toFixed(3)})`)
48
+ }
49
+ } catch (error) {
50
+ spinner.fail(formatError(error))
51
+ throw error
52
+ }
53
+ }
54
+
55
+ export async function cmdInstall(
56
+ opts: GlobalOpts,
57
+ slug: string,
58
+ versionFlag?: string,
59
+ force = false,
60
+ ) {
61
+ const trimmed = slug.trim()
62
+ if (!trimmed) fail('Slug required')
63
+
64
+ const registry = await getRegistry(opts, { cache: true })
65
+ await mkdir(opts.dir, { recursive: true })
66
+ const target = join(opts.dir, trimmed)
67
+ if (!force) {
68
+ const exists = await fileExists(target)
69
+ if (exists) fail(`Already installed: ${target} (use --force)`)
70
+ } else {
71
+ await rm(target, { recursive: true, force: true })
72
+ }
73
+
74
+ const spinner = createSpinner(`Resolving ${trimmed}`)
75
+ try {
76
+ const resolvedVersion =
77
+ versionFlag ??
78
+ (
79
+ await apiRequest(
80
+ registry,
81
+ { method: 'GET', path: `${ApiRoutes.skills}/${encodeURIComponent(trimmed)}` },
82
+ ApiV1SkillResponseSchema,
83
+ )
84
+ ).latestVersion?.version ??
85
+ null
86
+ if (!resolvedVersion) fail('Could not resolve latest version')
87
+
88
+ spinner.text = `Downloading ${trimmed}@${resolvedVersion}`
89
+ const zip = await downloadZip(registry, { slug: trimmed, version: resolvedVersion })
90
+ await extractZipToDir(zip, target)
91
+
92
+ await writeSkillOrigin(target, {
93
+ version: 1,
94
+ registry,
95
+ slug: trimmed,
96
+ installedVersion: resolvedVersion,
97
+ installedAt: Date.now(),
98
+ })
99
+
100
+ const lock = await readLockfile(opts.workdir)
101
+ lock.skills[trimmed] = {
102
+ version: resolvedVersion,
103
+ installedAt: Date.now(),
104
+ }
105
+ await writeLockfile(opts.workdir, lock)
106
+ spinner.succeed(`OK. Installed ${trimmed} -> ${target}`)
107
+ } catch (error) {
108
+ spinner.fail(formatError(error))
109
+ throw error
110
+ }
111
+ }
112
+
113
+ export async function cmdUpdate(
114
+ opts: GlobalOpts,
115
+ slugArg: string | undefined,
116
+ options: { all?: boolean; version?: string; force?: boolean },
117
+ inputAllowed: boolean,
118
+ ) {
119
+ const slug = slugArg?.trim()
120
+ const all = Boolean(options.all)
121
+ if (!slug && !all) fail('Provide <slug> or --all')
122
+ if (slug && all) fail('Use either <slug> or --all')
123
+ if (options.version && !slug) fail('--version requires a single <slug>')
124
+ if (options.version && !semver.valid(options.version)) fail('--version must be valid semver')
125
+ const allowPrompt = isInteractive() && inputAllowed !== false
126
+
127
+ const registry = await getRegistry(opts, { cache: true })
128
+ const lock = await readLockfile(opts.workdir)
129
+ const slugs = slug ? [slug] : Object.keys(lock.skills)
130
+ if (slugs.length === 0) {
131
+ console.log('No installed skills.')
132
+ return
133
+ }
134
+
135
+ for (const entry of slugs) {
136
+ const spinner = createSpinner(`Checking ${entry}`)
137
+ try {
138
+ const target = join(opts.dir, entry)
139
+ const exists = await fileExists(target)
140
+
141
+ let localFingerprint: string | null = null
142
+ if (exists) {
143
+ const filesOnDisk = await listTextFiles(target)
144
+ if (filesOnDisk.length > 0) {
145
+ const hashed = hashSkillFiles(filesOnDisk)
146
+ localFingerprint = hashed.fingerprint
147
+ }
148
+ }
149
+
150
+ let resolveResult: ResolveResult
151
+ if (localFingerprint) {
152
+ resolveResult = await resolveSkillVersion(registry, entry, localFingerprint)
153
+ } else {
154
+ const meta = await apiRequest(
155
+ registry,
156
+ { method: 'GET', path: `${ApiRoutes.skills}/${encodeURIComponent(entry)}` },
157
+ ApiV1SkillResponseSchema,
158
+ )
159
+ resolveResult = { match: null, latestVersion: meta.latestVersion ?? null }
160
+ }
161
+
162
+ const latest = resolveResult.latestVersion?.version ?? null
163
+ const matched = resolveResult.match?.version ?? null
164
+
165
+ if (matched && lock.skills[entry]?.version !== matched) {
166
+ lock.skills[entry] = {
167
+ version: matched,
168
+ installedAt: lock.skills[entry]?.installedAt ?? Date.now(),
169
+ }
170
+ }
171
+
172
+ if (!latest) {
173
+ spinner.fail(`${entry}: not found`)
174
+ continue
175
+ }
176
+
177
+ if (!matched && localFingerprint && !options.force) {
178
+ spinner.stop()
179
+ if (!allowPrompt) {
180
+ console.log(`${entry}: local changes (no match). Use --force to overwrite.`)
181
+ continue
182
+ }
183
+ const confirm = await promptConfirm(
184
+ `${entry}: local changes (no match). Overwrite with ${options.version ?? latest}?`,
185
+ )
186
+ if (!confirm) {
187
+ console.log(`${entry}: skipped`)
188
+ continue
189
+ }
190
+ spinner.start(`Updating ${entry} -> ${options.version ?? latest}`)
191
+ }
192
+
193
+ const targetVersion = options.version ?? latest
194
+ if (options.version) {
195
+ if (matched && matched === targetVersion) {
196
+ spinner.succeed(`${entry}: already at ${matched}`)
197
+ continue
198
+ }
199
+ } else if (matched && semver.valid(matched) && semver.gte(matched, targetVersion)) {
200
+ spinner.succeed(`${entry}: up to date (${matched})`)
201
+ continue
202
+ }
203
+
204
+ if (spinner.isSpinning) {
205
+ spinner.text = `Updating ${entry} -> ${targetVersion}`
206
+ } else {
207
+ spinner.start(`Updating ${entry} -> ${targetVersion}`)
208
+ }
209
+ await rm(target, { recursive: true, force: true })
210
+ const zip = await downloadZip(registry, { slug: entry, version: targetVersion })
211
+ await extractZipToDir(zip, target)
212
+
213
+ const existingOrigin = await readSkillOrigin(target)
214
+ await writeSkillOrigin(target, {
215
+ version: 1,
216
+ registry: existingOrigin?.registry ?? registry,
217
+ slug: existingOrigin?.slug ?? entry,
218
+ installedVersion: targetVersion,
219
+ installedAt: existingOrigin?.installedAt ?? Date.now(),
220
+ })
221
+
222
+ lock.skills[entry] = { version: targetVersion, installedAt: Date.now() }
223
+ spinner.succeed(`${entry}: updated -> ${targetVersion}`)
224
+ } catch (error) {
225
+ spinner.fail(formatError(error))
226
+ throw error
227
+ }
228
+ }
229
+
230
+ await writeLockfile(opts.workdir, lock)
231
+ }
232
+
233
+ export async function cmdList(opts: GlobalOpts) {
234
+ const lock = await readLockfile(opts.workdir)
235
+ const entries = Object.entries(lock.skills)
236
+ if (entries.length === 0) {
237
+ console.log('No installed skills.')
238
+ return
239
+ }
240
+ for (const [slug, entry] of entries) {
241
+ console.log(`${slug} ${entry.version ?? 'latest'}`)
242
+ }
243
+ }
244
+
245
+ type ExploreSort = 'newest' | 'downloads' | 'rating' | 'installs' | 'installsAllTime' | 'trending'
246
+ type ApiExploreSort =
247
+ | 'updated'
248
+ | 'downloads'
249
+ | 'stars'
250
+ | 'installsCurrent'
251
+ | 'installsAllTime'
252
+ | 'trending'
253
+
254
+ export async function cmdExplore(
255
+ opts: GlobalOpts,
256
+ options: { limit?: number; sort?: string; json?: boolean } = {},
257
+ ) {
258
+ const registry = await getRegistry(opts, { cache: true })
259
+ const spinner = createSpinner('Fetching latest skills')
260
+ try {
261
+ const url = new URL(ApiRoutes.skills, registry)
262
+ const boundedLimit = clampLimit(options.limit ?? 25)
263
+ const { apiSort } = resolveExploreSort(options.sort)
264
+ url.searchParams.set('limit', String(boundedLimit))
265
+ if (apiSort !== 'updated') url.searchParams.set('sort', apiSort)
266
+ const result = await apiRequest(
267
+ registry,
268
+ { method: 'GET', url: url.toString() },
269
+ ApiV1SkillListResponseSchema,
270
+ )
271
+
272
+ spinner.stop()
273
+ if (options.json) {
274
+ console.log(JSON.stringify(result, null, 2))
275
+ return
276
+ }
277
+ if (result.items.length === 0) {
278
+ console.log('No skills found.')
279
+ return
280
+ }
281
+
282
+ for (const item of result.items) {
283
+ console.log(formatExploreLine(item))
284
+ }
285
+ } catch (error) {
286
+ spinner.fail(formatError(error))
287
+ throw error
288
+ }
289
+ }
290
+
291
+ export function formatExploreLine(item: {
292
+ slug: string
293
+ summary?: string | null
294
+ updatedAt: number
295
+ latestVersion?: { version: string } | null
296
+ }) {
297
+ const version = item.latestVersion?.version ?? '?'
298
+ const age = formatRelativeTime(item.updatedAt)
299
+ const summary = item.summary ? ` ${truncate(item.summary, 50)}` : ''
300
+ return `${item.slug} v${version} ${age}${summary}`
301
+ }
302
+
303
+ export function clampLimit(limit: number, fallback = 25) {
304
+ if (!Number.isFinite(limit)) return fallback
305
+ return Math.min(Math.max(1, limit), 200)
306
+ }
307
+
308
+ function formatRelativeTime(timestamp: number): string {
309
+ const now = Date.now()
310
+ const diff = now - timestamp
311
+ const seconds = Math.floor(diff / 1000)
312
+ const minutes = Math.floor(seconds / 60)
313
+ const hours = Math.floor(minutes / 60)
314
+ const days = Math.floor(hours / 24)
315
+
316
+ if (days > 30) {
317
+ const months = Math.floor(days / 30)
318
+ return `${months}mo ago`
319
+ }
320
+ if (days > 0) return `${days}d ago`
321
+ if (hours > 0) return `${hours}h ago`
322
+ if (minutes > 0) return `${minutes}m ago`
323
+ return 'just now'
324
+ }
325
+
326
+ function truncate(str: string, maxLen: number): string {
327
+ if (str.length <= maxLen) return str
328
+ return `${str.slice(0, maxLen - 1)}…`
329
+ }
330
+
331
+ function resolveExploreSort(raw?: string): { sort: ExploreSort; apiSort: ApiExploreSort } {
332
+ const normalized = raw?.trim().toLowerCase()
333
+ if (!normalized || normalized === 'newest' || normalized === 'updated') {
334
+ return { sort: 'newest', apiSort: 'updated' }
335
+ }
336
+ if (normalized === 'downloads' || normalized === 'download') {
337
+ return { sort: 'downloads', apiSort: 'downloads' }
338
+ }
339
+ if (normalized === 'rating' || normalized === 'stars' || normalized === 'star') {
340
+ return { sort: 'rating', apiSort: 'stars' }
341
+ }
342
+ if (
343
+ normalized === 'installs' ||
344
+ normalized === 'install' ||
345
+ normalized === 'installscurrent' ||
346
+ normalized === 'installs-current' ||
347
+ normalized === 'current'
348
+ ) {
349
+ return { sort: 'installs', apiSort: 'installsCurrent' }
350
+ }
351
+ if (normalized === 'installsalltime' || normalized === 'installs-all-time') {
352
+ return { sort: 'installsAllTime', apiSort: 'installsAllTime' }
353
+ }
354
+ if (normalized === 'trending') {
355
+ return { sort: 'trending', apiSort: 'trending' }
356
+ }
357
+ fail(
358
+ `Invalid sort "${raw}". Use newest, downloads, rating, installs, installsAllTime, or trending.`,
359
+ )
360
+ }
361
+
362
+ async function resolveSkillVersion(registry: string, slug: string, hash: string) {
363
+ const url = new URL(ApiRoutes.resolve, registry)
364
+ url.searchParams.set('slug', slug)
365
+ url.searchParams.set('hash', hash)
366
+ return apiRequest(
367
+ registry,
368
+ { method: 'GET', url: url.toString() },
369
+ ApiV1SkillResolveResponseSchema,
370
+ )
371
+ }
372
+
373
+ async function fileExists(path: string) {
374
+ try {
375
+ await stat(path)
376
+ return true
377
+ } catch {
378
+ return false
379
+ }
380
+ }
@@ -0,0 +1,46 @@
1
+ import { readGlobalConfig } from '../../config.js'
2
+ import { apiRequest } from '../../http.js'
3
+ import { ApiRoutes, ApiV1StarResponseSchema } from '../../schema/index.js'
4
+ import { getRegistry } from '../registry.js'
5
+ import type { GlobalOpts } from '../types.js'
6
+ import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js'
7
+
8
+ async function requireToken() {
9
+ const cfg = await readGlobalConfig()
10
+ const token = cfg?.token
11
+ if (!token) fail('Not logged in. Run: pilothub login')
12
+ return token
13
+ }
14
+
15
+ export async function cmdStarSkill(
16
+ opts: GlobalOpts,
17
+ slugArg: string,
18
+ options: { yes?: boolean },
19
+ inputAllowed: boolean,
20
+ ) {
21
+ const slug = slugArg.trim().toLowerCase()
22
+ if (!slug) fail('Slug required')
23
+ const allowPrompt = isInteractive() && inputAllowed !== false
24
+
25
+ if (!options.yes) {
26
+ if (!allowPrompt) fail('Pass --yes (no input)')
27
+ const ok = await promptConfirm(`Star ${slug}?`)
28
+ if (!ok) return
29
+ }
30
+
31
+ const token = await requireToken()
32
+ const registry = await getRegistry(opts, { cache: true })
33
+ const spinner = createSpinner(`Starring ${slug}`)
34
+ try {
35
+ const result = await apiRequest(
36
+ registry,
37
+ { method: 'POST', path: `${ApiRoutes.stars}/${encodeURIComponent(slug)}`, token },
38
+ ApiV1StarResponseSchema,
39
+ )
40
+ spinner.succeed(result.alreadyStarred ? `OK. ${slug} already starred.` : `OK. Starred ${slug}`)
41
+ return result
42
+ } catch (error) {
43
+ spinner.fail(formatError(error))
44
+ throw error
45
+ }
46
+ }