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,83 @@
1
+ import { readGlobalConfig } from '../../config.js'
2
+ import { apiRequest } from '../../http.js'
3
+ import { ApiRoutes, ApiV1DeleteResponseSchema, parseArk } 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 cmdDeleteSkill(
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(`Delete ${slug}? (soft delete)`)
28
+ if (!ok) return
29
+ }
30
+
31
+ const token = await requireToken()
32
+ const registry = await getRegistry(opts, { cache: true })
33
+ const spinner = createSpinner(`Deleting ${slug}`)
34
+ try {
35
+ const result = await apiRequest(
36
+ registry,
37
+ { method: 'DELETE', path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}`, token },
38
+ ApiV1DeleteResponseSchema,
39
+ )
40
+ spinner.succeed(`OK. Deleted ${slug}`)
41
+ return parseArk(ApiV1DeleteResponseSchema, result, 'Delete response')
42
+ } catch (error) {
43
+ spinner.fail(formatError(error))
44
+ throw error
45
+ }
46
+ }
47
+
48
+ export async function cmdUndeleteSkill(
49
+ opts: GlobalOpts,
50
+ slugArg: string,
51
+ options: { yes?: boolean },
52
+ inputAllowed: boolean,
53
+ ) {
54
+ const slug = slugArg.trim().toLowerCase()
55
+ if (!slug) fail('Slug required')
56
+ const allowPrompt = isInteractive() && inputAllowed !== false
57
+
58
+ if (!options.yes) {
59
+ if (!allowPrompt) fail('Pass --yes (no input)')
60
+ const ok = await promptConfirm(`Undelete ${slug}?`)
61
+ if (!ok) return
62
+ }
63
+
64
+ const token = await requireToken()
65
+ const registry = await getRegistry(opts, { cache: true })
66
+ const spinner = createSpinner(`Undeleting ${slug}`)
67
+ try {
68
+ const result = await apiRequest(
69
+ registry,
70
+ {
71
+ method: 'POST',
72
+ path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}/undelete`,
73
+ token,
74
+ },
75
+ ApiV1DeleteResponseSchema,
76
+ )
77
+ spinner.succeed(`OK. Undeleted ${slug}`)
78
+ return parseArk(ApiV1DeleteResponseSchema, result, 'Undelete response')
79
+ } catch (error) {
80
+ spinner.fail(formatError(error))
81
+ throw error
82
+ }
83
+ }
@@ -0,0 +1,122 @@
1
+ /* @vitest-environment node */
2
+
3
+ import { mkdir, mkdtemp, rm, writeFile } from 'node:fs/promises'
4
+ import { tmpdir } from 'node:os'
5
+ import { join } from 'node:path'
6
+ import { afterEach, describe, expect, it, vi } from 'vitest'
7
+ import type { GlobalOpts } from '../types'
8
+
9
+ vi.mock('../../config.js', () => ({
10
+ readGlobalConfig: vi.fn(async () => ({ registry: 'https://pilothub.com', token: 'tkn' })),
11
+ }))
12
+
13
+ const mockGetRegistry = vi.fn(async (_opts: unknown, _params?: unknown) => 'https://pilothub.com')
14
+ vi.mock('../registry.js', () => ({
15
+ getRegistry: (opts: unknown, params?: unknown) => mockGetRegistry(opts, params),
16
+ }))
17
+
18
+ const mockApiRequestForm = vi.fn()
19
+ vi.mock('../../http.js', () => ({
20
+ apiRequestForm: (registry: unknown, args: unknown, schema?: unknown) =>
21
+ mockApiRequestForm(registry, args, schema),
22
+ }))
23
+
24
+ const mockFail = vi.fn((message: string) => {
25
+ throw new Error(message)
26
+ })
27
+ const mockSpinner = { text: '', succeed: vi.fn(), fail: vi.fn() }
28
+ vi.mock('../ui.js', () => ({
29
+ createSpinner: vi.fn(() => mockSpinner),
30
+ fail: (message: string) => mockFail(message),
31
+ formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
32
+ }))
33
+
34
+ const { cmdPublish } = await import('./publish')
35
+
36
+ async function makeTmpWorkdir() {
37
+ const root = await mkdtemp(join(tmpdir(), 'pilothub-publish-'))
38
+ return root
39
+ }
40
+
41
+ function makeOpts(workdir: string): GlobalOpts {
42
+ return {
43
+ workdir,
44
+ dir: join(workdir, 'skills'),
45
+ site: 'https://pilothub.com',
46
+ registry: 'https://pilothub.com',
47
+ registrySource: 'default',
48
+ }
49
+ }
50
+
51
+ afterEach(() => {
52
+ vi.unstubAllGlobals()
53
+ vi.clearAllMocks()
54
+ })
55
+
56
+ describe('cmdPublish', () => {
57
+ it('publishes SKILL.md from disk (mocked HTTP)', async () => {
58
+ const workdir = await makeTmpWorkdir()
59
+ try {
60
+ const folder = join(workdir, 'my-skill')
61
+ await mkdir(folder, { recursive: true })
62
+ const skillContent = '# Skill\n\nHello\n'
63
+ const notesContent = 'notes\n'
64
+ await writeFile(join(folder, 'SKILL.md'), skillContent, 'utf8')
65
+ await writeFile(join(folder, 'notes.md'), notesContent, 'utf8')
66
+
67
+ mockApiRequestForm.mockResolvedValueOnce({ ok: true, skillId: 'skill_1', versionId: 'ver_1' })
68
+
69
+ await cmdPublish(makeOpts(workdir), 'my-skill', {
70
+ slug: 'my-skill',
71
+ name: 'My Skill',
72
+ version: '1.0.0',
73
+ changelog: '',
74
+ tags: 'latest',
75
+ })
76
+
77
+ const publishCall = mockApiRequestForm.mock.calls.find((call) => {
78
+ const req = call[1] as { path?: string } | undefined
79
+ return req?.path === '/api/v1/skills'
80
+ })
81
+ if (!publishCall) throw new Error('Missing publish call')
82
+ const publishForm = (publishCall[1] as { form?: FormData }).form as FormData
83
+ const payloadEntry = publishForm.get('payload')
84
+ if (typeof payloadEntry !== 'string') throw new Error('Missing publish payload')
85
+ const payload = JSON.parse(payloadEntry)
86
+ expect(payload.slug).toBe('my-skill')
87
+ expect(payload.displayName).toBe('My Skill')
88
+ expect(payload.version).toBe('1.0.0')
89
+ expect(payload.changelog).toBe('')
90
+ expect(payload.tags).toEqual(['latest'])
91
+ const files = publishForm.getAll('files') as Array<Blob & { name?: string }>
92
+ expect(files.map((file) => String(file.name ?? '')).sort()).toEqual(['SKILL.md', 'notes.md'])
93
+ } finally {
94
+ await rm(workdir, { recursive: true, force: true })
95
+ }
96
+ })
97
+
98
+ it('allows empty changelog when updating an existing skill', async () => {
99
+ const workdir = await makeTmpWorkdir()
100
+ try {
101
+ const folder = join(workdir, 'existing-skill')
102
+ await mkdir(folder, { recursive: true })
103
+ await writeFile(join(folder, 'SKILL.md'), '# Skill\n', 'utf8')
104
+
105
+ mockApiRequestForm.mockResolvedValueOnce({ ok: true, skillId: 'skill_1', versionId: 'ver_2' })
106
+
107
+ await cmdPublish(makeOpts(workdir), 'existing-skill', {
108
+ version: '1.0.1',
109
+ changelog: '',
110
+ tags: 'latest',
111
+ })
112
+
113
+ expect(mockApiRequestForm).toHaveBeenCalledWith(
114
+ expect.anything(),
115
+ expect.objectContaining({ path: '/api/v1/skills', method: 'POST' }),
116
+ expect.anything(),
117
+ )
118
+ } finally {
119
+ await rm(workdir, { recursive: true, force: true })
120
+ }
121
+ })
122
+ })
@@ -0,0 +1,108 @@
1
+ import { stat } from 'node:fs/promises'
2
+ import { basename, resolve } from 'node:path'
3
+ import semver from 'semver'
4
+ import { readGlobalConfig } from '../../config.js'
5
+ import { apiRequestForm } from '../../http.js'
6
+ import { ApiRoutes, ApiV1PublishResponseSchema } from '../../schema/index.js'
7
+ import { listTextFiles } from '../../skills.js'
8
+ import { getRegistry } from '../registry.js'
9
+ import { sanitizeSlug, titleCase } from '../slug.js'
10
+ import type { GlobalOpts } from '../types.js'
11
+ import { createSpinner, fail, formatError } from '../ui.js'
12
+
13
+ export async function cmdPublish(
14
+ opts: GlobalOpts,
15
+ folderArg: string,
16
+ options: {
17
+ slug?: string
18
+ name?: string
19
+ version?: string
20
+ changelog?: string
21
+ tags?: string
22
+ forkOf?: string
23
+ },
24
+ ) {
25
+ const folder = folderArg ? resolve(opts.workdir, folderArg) : null
26
+ if (!folder) fail('Path required')
27
+ const folderStat = await stat(folder).catch(() => null)
28
+ if (!folderStat || !folderStat.isDirectory()) fail('Path must be a folder')
29
+
30
+ const cfg = await readGlobalConfig()
31
+ const token = cfg?.token
32
+ if (!token) fail('Not logged in. Run: pilothub login')
33
+ const registry = await getRegistry(opts, { cache: true })
34
+
35
+ const slug = options.slug ?? sanitizeSlug(basename(folder))
36
+ const displayName = options.name ?? titleCase(basename(folder))
37
+ const version = options.version
38
+ const changelog = options.changelog ?? ''
39
+ const tagsValue = options.tags ?? 'latest'
40
+ const tags = tagsValue
41
+ .split(',')
42
+ .map((tag) => tag.trim())
43
+ .filter(Boolean)
44
+
45
+ const forkOfRaw = options.forkOf?.trim()
46
+ const forkOf = forkOfRaw ? parseForkOf(forkOfRaw) : undefined
47
+
48
+ if (!slug) fail('--slug required')
49
+ if (!displayName) fail('--name required')
50
+ if (!version || !semver.valid(version)) fail('--version must be valid semver')
51
+
52
+ const spinner = createSpinner(`Preparing ${slug}@${version}`)
53
+ try {
54
+ const filesOnDisk = await listTextFiles(folder)
55
+ if (filesOnDisk.length === 0) fail('No files found')
56
+ if (
57
+ !filesOnDisk.some((file) => {
58
+ const lower = file.relPath.toLowerCase()
59
+ return lower === 'skill.md' || lower === 'skills.md'
60
+ })
61
+ ) {
62
+ fail('SKILL.md required')
63
+ }
64
+
65
+ const form = new FormData()
66
+ form.set(
67
+ 'payload',
68
+ JSON.stringify({
69
+ slug,
70
+ displayName,
71
+ version,
72
+ changelog,
73
+ tags,
74
+ ...(forkOf ? { forkOf } : {}),
75
+ }),
76
+ )
77
+
78
+ let index = 0
79
+ for (const file of filesOnDisk) {
80
+ index += 1
81
+ spinner.text = `Uploading ${file.relPath} (${index}/${filesOnDisk.length})`
82
+ const blob = new Blob([Buffer.from(file.bytes)], { type: file.contentType ?? 'text/plain' })
83
+ form.append('files', blob, file.relPath)
84
+ }
85
+
86
+ spinner.text = `Publishing ${slug}@${version}`
87
+ const result = await apiRequestForm(
88
+ registry,
89
+ { method: 'POST', path: ApiRoutes.skills, token, form },
90
+ ApiV1PublishResponseSchema,
91
+ )
92
+
93
+ spinner.succeed(`OK. Published ${slug}@${version} (${result.versionId})`)
94
+ } catch (error) {
95
+ spinner.fail(formatError(error))
96
+ throw error
97
+ }
98
+ }
99
+
100
+ function parseForkOf(value: string) {
101
+ const trimmed = value.trim()
102
+ const [slugRaw, versionRaw] = trimmed.split('@')
103
+ const slug = (slugRaw ?? '').trim().toLowerCase()
104
+ if (!slug) fail('--fork-of must be <slug> or <slug@version>')
105
+ const version = (versionRaw ?? '').trim()
106
+ if (version && !semver.valid(version)) fail('--fork-of version must be valid semver')
107
+ return { slug, version: version || undefined }
108
+ }
@@ -0,0 +1,191 @@
1
+ /* @vitest-environment node */
2
+
3
+ import { afterEach, describe, expect, it, vi } from 'vitest'
4
+ import { ApiRoutes } from '../../schema/index.js'
5
+ import type { GlobalOpts } from '../types.js'
6
+
7
+ const mockApiRequest = vi.fn()
8
+ const mockDownloadZip = vi.fn()
9
+ vi.mock('../../http.js', () => ({
10
+ apiRequest: (...args: unknown[]) => mockApiRequest(...args),
11
+ downloadZip: (...args: unknown[]) => mockDownloadZip(...args),
12
+ }))
13
+
14
+ const mockGetRegistry = vi.fn(async () => 'https://pilothub.com')
15
+ vi.mock('../registry.js', () => ({
16
+ getRegistry: () => mockGetRegistry(),
17
+ }))
18
+
19
+ const mockSpinner = {
20
+ stop: vi.fn(),
21
+ fail: vi.fn(),
22
+ start: vi.fn(),
23
+ succeed: vi.fn(),
24
+ isSpinning: false,
25
+ text: '',
26
+ }
27
+ vi.mock('../ui.js', () => ({
28
+ createSpinner: vi.fn(() => mockSpinner),
29
+ fail: (message: string) => {
30
+ throw new Error(message)
31
+ },
32
+ formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
33
+ isInteractive: () => false,
34
+ promptConfirm: vi.fn(async () => false),
35
+ }))
36
+
37
+ vi.mock('../../skills.js', () => ({
38
+ extractZipToDir: vi.fn(),
39
+ hashSkillFiles: vi.fn(),
40
+ listTextFiles: vi.fn(),
41
+ readLockfile: vi.fn(),
42
+ readSkillOrigin: vi.fn(),
43
+ writeLockfile: vi.fn(),
44
+ writeSkillOrigin: vi.fn(),
45
+ }))
46
+
47
+ vi.mock('node:fs/promises', () => ({
48
+ mkdir: vi.fn(),
49
+ rm: vi.fn(),
50
+ stat: vi.fn(),
51
+ }))
52
+
53
+ const { clampLimit, cmdExplore, cmdUpdate, formatExploreLine } = await import('./skills.js')
54
+ const {
55
+ extractZipToDir,
56
+ hashSkillFiles,
57
+ listTextFiles,
58
+ readLockfile,
59
+ readSkillOrigin,
60
+ writeLockfile,
61
+ writeSkillOrigin,
62
+ } = await import('../../skills.js')
63
+ const { rm, stat } = await import('node:fs/promises')
64
+
65
+ const mockLog = vi.spyOn(console, 'log').mockImplementation(() => {})
66
+
67
+ function makeOpts(): GlobalOpts {
68
+ return {
69
+ workdir: '/work',
70
+ dir: '/work/skills',
71
+ site: 'https://pilothub.com',
72
+ registry: 'https://pilothub.com',
73
+ registrySource: 'default',
74
+ }
75
+ }
76
+
77
+ afterEach(() => {
78
+ vi.clearAllMocks()
79
+ })
80
+
81
+ describe('explore helpers', () => {
82
+ it('clamps explore limits and handles non-finite values', () => {
83
+ expect(clampLimit(-5)).toBe(1)
84
+ expect(clampLimit(0)).toBe(1)
85
+ expect(clampLimit(1)).toBe(1)
86
+ expect(clampLimit(50)).toBe(50)
87
+ expect(clampLimit(99)).toBe(99)
88
+ expect(clampLimit(200)).toBe(200)
89
+ expect(clampLimit(250)).toBe(200)
90
+ expect(clampLimit(Number.NaN)).toBe(25)
91
+ expect(clampLimit(Number.POSITIVE_INFINITY)).toBe(25)
92
+ expect(clampLimit(Number.NaN, 10)).toBe(10)
93
+ })
94
+
95
+ it('formats explore lines with relative time and truncation', () => {
96
+ const now = 4 * 60 * 60 * 1000
97
+ const nowSpy = vi.spyOn(Date, 'now').mockReturnValue(now)
98
+ const summary = 'a'.repeat(60)
99
+ const line = formatExploreLine({
100
+ slug: 'weather',
101
+ summary,
102
+ updatedAt: now - 2 * 60 * 60 * 1000,
103
+ latestVersion: null,
104
+ })
105
+ expect(line).toBe(`weather v? 2h ago ${'a'.repeat(49)}…`)
106
+ nowSpy.mockRestore()
107
+ })
108
+ })
109
+
110
+ describe('cmdExplore', () => {
111
+ it('clamps limit and handles empty results', async () => {
112
+ mockApiRequest.mockResolvedValue({ items: [] })
113
+
114
+ await cmdExplore(makeOpts(), { limit: 0 })
115
+
116
+ const [, args] = mockApiRequest.mock.calls[0] ?? []
117
+ const url = new URL(String(args?.url))
118
+ expect(url.searchParams.get('limit')).toBe('1')
119
+ expect(mockLog).toHaveBeenCalledWith('No skills found.')
120
+ })
121
+
122
+ it('prints formatted results', async () => {
123
+ const now = 10 * 60 * 1000
124
+ const nowSpy = vi.spyOn(Date, 'now').mockReturnValue(now)
125
+ const item = {
126
+ slug: 'gog',
127
+ summary: 'Google Workspace CLI for Gmail, Calendar, Drive and more.',
128
+ updatedAt: now - 90 * 1000,
129
+ latestVersion: { version: '1.2.3' },
130
+ }
131
+ mockApiRequest.mockResolvedValue({ items: [item] })
132
+
133
+ await cmdExplore(makeOpts(), { limit: 250 })
134
+
135
+ const [, args] = mockApiRequest.mock.calls[0] ?? []
136
+ const url = new URL(String(args?.url))
137
+ expect(url.searchParams.get('limit')).toBe('200')
138
+ expect(mockLog).toHaveBeenCalledWith(formatExploreLine(item))
139
+ nowSpy.mockRestore()
140
+ })
141
+
142
+ it('supports sort and json output', async () => {
143
+ const payload = { items: [], nextCursor: null }
144
+ mockApiRequest.mockResolvedValue(payload)
145
+
146
+ await cmdExplore(makeOpts(), { limit: 10, sort: 'installs', json: true })
147
+
148
+ const [, args] = mockApiRequest.mock.calls[0] ?? []
149
+ const url = new URL(String(args?.url))
150
+ expect(url.searchParams.get('limit')).toBe('10')
151
+ expect(url.searchParams.get('sort')).toBe('installsCurrent')
152
+ expect(mockLog).toHaveBeenCalledWith(JSON.stringify(payload, null, 2))
153
+ })
154
+
155
+ it('supports all-time installs and trending sorts', async () => {
156
+ mockApiRequest.mockResolvedValue({ items: [], nextCursor: null })
157
+
158
+ await cmdExplore(makeOpts(), { limit: 5, sort: 'installsAllTime' })
159
+ await cmdExplore(makeOpts(), { limit: 5, sort: 'trending' })
160
+
161
+ const first = new URL(String(mockApiRequest.mock.calls[0]?.[1]?.url))
162
+ const second = new URL(String(mockApiRequest.mock.calls[1]?.[1]?.url))
163
+ expect(first.searchParams.get('sort')).toBe('installsAllTime')
164
+ expect(second.searchParams.get('sort')).toBe('trending')
165
+ })
166
+ })
167
+
168
+ describe('cmdUpdate', () => {
169
+ it('uses path-based skill lookup when no local fingerprint is available', async () => {
170
+ mockApiRequest.mockResolvedValue({ latestVersion: { version: '1.0.0' } })
171
+ mockDownloadZip.mockResolvedValue(new Uint8Array([1, 2, 3]))
172
+ vi.mocked(readLockfile).mockResolvedValue({
173
+ version: 1,
174
+ skills: { demo: { version: '0.1.0', installedAt: 123 } },
175
+ })
176
+ vi.mocked(writeLockfile).mockResolvedValue()
177
+ vi.mocked(readSkillOrigin).mockResolvedValue(null)
178
+ vi.mocked(writeSkillOrigin).mockResolvedValue()
179
+ vi.mocked(extractZipToDir).mockResolvedValue()
180
+ vi.mocked(listTextFiles).mockResolvedValue([])
181
+ vi.mocked(hashSkillFiles).mockReturnValue({ fingerprint: 'hash', files: [] })
182
+ vi.mocked(stat).mockRejectedValue(new Error('missing'))
183
+ vi.mocked(rm).mockResolvedValue()
184
+
185
+ await cmdUpdate(makeOpts(), 'demo', {}, false)
186
+
187
+ const [, args] = mockApiRequest.mock.calls[0] ?? []
188
+ expect(args?.path).toBe(`${ApiRoutes.skills}/${encodeURIComponent('demo')}`)
189
+ expect(args?.url).toBeUndefined()
190
+ })
191
+ })