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,112 @@
1
+ export type WebhookEvent = 'skill.publish' | 'skill.highlighted'
2
+
3
+ export type WebhookSkillPayload = {
4
+ slug: string
5
+ displayName: string
6
+ summary?: string
7
+ version?: string
8
+ ownerHandle?: string
9
+ highlighted?: boolean
10
+ tags?: string[]
11
+ }
12
+
13
+ export type WebhookConfig = {
14
+ url: string | null
15
+ highlightedOnly: boolean
16
+ siteUrl: string
17
+ }
18
+
19
+ const DEFAULT_SITE_URL = 'https://pilothub.com'
20
+
21
+ export function getWebhookConfig(env: NodeJS.ProcessEnv = process.env): WebhookConfig {
22
+ const url = env.DISCORD_WEBHOOK_URL?.trim() || null
23
+ const highlightedOnly = parseBoolean(env.DISCORD_WEBHOOK_HIGHLIGHTED_ONLY)
24
+ const siteUrl = env.SITE_URL?.trim() || DEFAULT_SITE_URL
25
+ return { url, highlightedOnly, siteUrl }
26
+ }
27
+
28
+ export function shouldSendWebhook(
29
+ event: WebhookEvent,
30
+ skill: WebhookSkillPayload,
31
+ config: WebhookConfig,
32
+ ) {
33
+ if (!config.url) return false
34
+ if (!config.highlightedOnly) return true
35
+ if (event === 'skill.highlighted') return true
36
+ return Boolean(skill.highlighted)
37
+ }
38
+
39
+ export function buildDiscordPayload(
40
+ event: WebhookEvent,
41
+ skill: WebhookSkillPayload,
42
+ config: WebhookConfig,
43
+ ) {
44
+ const titleBase = skill.displayName || skill.slug
45
+ const title = event === 'skill.highlighted' ? `Highlighted: ${titleBase}` : titleBase
46
+ const description = buildDescription(event, skill)
47
+ const url = buildSkillUrl(skill, config.siteUrl)
48
+ const tags = formatTags(skill.tags)
49
+
50
+ return {
51
+ embeds: [
52
+ {
53
+ title,
54
+ description,
55
+ url,
56
+ color: event === 'skill.highlighted' ? 0xff6b4a : 0x2f76ff,
57
+ fields: [
58
+ {
59
+ name: 'Version',
60
+ value: skill.version ? `v${skill.version}` : '—',
61
+ inline: true,
62
+ },
63
+ {
64
+ name: 'Owner',
65
+ value: skill.ownerHandle ? `@${skill.ownerHandle}` : '—',
66
+ inline: true,
67
+ },
68
+ {
69
+ name: 'Tags',
70
+ value: tags,
71
+ inline: false,
72
+ },
73
+ ],
74
+ footer: {
75
+ text: 'PilotHub',
76
+ },
77
+ timestamp: new Date().toISOString(),
78
+ },
79
+ ],
80
+ }
81
+ }
82
+
83
+ export function buildSkillUrl(skill: WebhookSkillPayload, siteUrl: string) {
84
+ const owner = skill.ownerHandle?.trim()
85
+ if (owner) return `${siteUrl}/${owner}/${skill.slug}`
86
+ return `${siteUrl}/skills/${skill.slug}`
87
+ }
88
+
89
+ function buildDescription(event: WebhookEvent, skill: WebhookSkillPayload) {
90
+ const summary = (skill.summary ?? '').trim()
91
+ if (summary) return truncate(summary, 200)
92
+ if (event === 'skill.highlighted') return 'Newly highlighted skill on PilotHub.'
93
+ if (skill.version) return `New version v${skill.version} published on PilotHub.`
94
+ return 'New skill published on PilotHub.'
95
+ }
96
+
97
+ function parseBoolean(value?: string) {
98
+ if (!value) return false
99
+ const normalized = value.trim().toLowerCase()
100
+ return normalized === '1' || normalized === 'true' || normalized === 'yes' || normalized === 'on'
101
+ }
102
+
103
+ function formatTags(tags?: string[] | null) {
104
+ const cleaned = (tags ?? []).map((tag) => tag.trim()).filter(Boolean)
105
+ if (cleaned.length === 0) return '—'
106
+ return cleaned.slice(0, 8).join(', ')
107
+ }
108
+
109
+ function truncate(value: string, max: number) {
110
+ if (value.length <= max) return value
111
+ return `${value.slice(0, max - 1).trim()}…`
112
+ }
@@ -0,0 +1,270 @@
1
+ /* @vitest-environment node */
2
+ import { describe, expect, it, vi } from 'vitest'
3
+
4
+ vi.mock('./_generated/api', () => ({
5
+ internal: {
6
+ maintenance: {
7
+ getSkillBackfillPageInternal: Symbol('getSkillBackfillPageInternal'),
8
+ applySkillBackfillPatchInternal: Symbol('applySkillBackfillPatchInternal'),
9
+ backfillSkillSummariesInternal: Symbol('backfillSkillSummariesInternal'),
10
+ getSkillFingerprintBackfillPageInternal: Symbol('getSkillFingerprintBackfillPageInternal'),
11
+ applySkillFingerprintBackfillPatchInternal: Symbol(
12
+ 'applySkillFingerprintBackfillPatchInternal',
13
+ ),
14
+ backfillSkillFingerprintsInternal: Symbol('backfillSkillFingerprintsInternal'),
15
+ },
16
+ },
17
+ }))
18
+
19
+ const { backfillSkillFingerprintsInternalHandler, backfillSkillSummariesInternalHandler } =
20
+ await import('./maintenance')
21
+
22
+ function makeBlob(text: string) {
23
+ return { text: () => Promise.resolve(text) } as unknown as Blob
24
+ }
25
+
26
+ describe('maintenance backfill', () => {
27
+ it('repairs summary + parsed by reparsing SKILL.md', async () => {
28
+ const runQuery = vi.fn().mockResolvedValue({
29
+ items: [
30
+ {
31
+ kind: 'ok',
32
+ skillId: 'skills:1',
33
+ versionId: 'skillVersions:1',
34
+ skillSummary: '>',
35
+ versionParsed: { frontmatter: { description: '>' } },
36
+ readmeStorageId: 'storage:1',
37
+ },
38
+ ],
39
+ cursor: null,
40
+ isDone: true,
41
+ })
42
+
43
+ const runMutation = vi.fn().mockResolvedValue({ ok: true })
44
+ const storageGet = vi
45
+ .fn()
46
+ .mockResolvedValue(makeBlob(`---\ndescription: >\n Hello\n world.\n---\nBody`))
47
+
48
+ const result = await backfillSkillSummariesInternalHandler(
49
+ { runQuery, runMutation, storage: { get: storageGet } } as never,
50
+ { dryRun: false, batchSize: 10, maxBatches: 1 },
51
+ )
52
+
53
+ expect(result.ok).toBe(true)
54
+ expect(result.stats.skillsScanned).toBe(1)
55
+ expect(result.stats.skillsPatched).toBe(1)
56
+ expect(result.stats.versionsPatched).toBe(1)
57
+ expect(runMutation).toHaveBeenCalledTimes(1)
58
+ expect(runMutation).toHaveBeenCalledWith(expect.anything(), {
59
+ skillId: 'skills:1',
60
+ versionId: 'skillVersions:1',
61
+ summary: 'Hello world.',
62
+ parsed: {
63
+ frontmatter: { description: 'Hello world.' },
64
+ metadata: undefined,
65
+ pilotbot: undefined,
66
+ },
67
+ })
68
+ })
69
+
70
+ it('dryRun does not patch', async () => {
71
+ const runQuery = vi.fn().mockResolvedValue({
72
+ items: [
73
+ {
74
+ kind: 'ok',
75
+ skillId: 'skills:1',
76
+ versionId: 'skillVersions:1',
77
+ skillSummary: '>',
78
+ versionParsed: { frontmatter: { description: '>' } },
79
+ readmeStorageId: 'storage:1',
80
+ },
81
+ ],
82
+ cursor: null,
83
+ isDone: true,
84
+ })
85
+
86
+ const runMutation = vi.fn()
87
+ const storageGet = vi.fn().mockResolvedValue(makeBlob(`---\ndescription: Hello\n---\nBody`))
88
+
89
+ const result = await backfillSkillSummariesInternalHandler(
90
+ { runQuery, runMutation, storage: { get: storageGet } } as never,
91
+ { dryRun: true, batchSize: 10, maxBatches: 1 },
92
+ )
93
+
94
+ expect(result.ok).toBe(true)
95
+ expect(result.stats.skillsPatched).toBe(1)
96
+ expect(runMutation).not.toHaveBeenCalled()
97
+ })
98
+
99
+ it('counts missing storage blob', async () => {
100
+ const runQuery = vi.fn().mockResolvedValue({
101
+ items: [
102
+ {
103
+ kind: 'ok',
104
+ skillId: 'skills:1',
105
+ versionId: 'skillVersions:1',
106
+ skillSummary: null,
107
+ versionParsed: { frontmatter: {} },
108
+ readmeStorageId: 'storage:missing',
109
+ },
110
+ ],
111
+ cursor: null,
112
+ isDone: true,
113
+ })
114
+
115
+ const runMutation = vi.fn()
116
+ const storageGet = vi.fn().mockResolvedValue(null)
117
+
118
+ const result = await backfillSkillSummariesInternalHandler(
119
+ { runQuery, runMutation, storage: { get: storageGet } } as never,
120
+ { dryRun: false, batchSize: 10, maxBatches: 1 },
121
+ )
122
+
123
+ expect(result.stats.missingStorageBlob).toBe(1)
124
+ expect(runMutation).not.toHaveBeenCalled()
125
+ })
126
+ })
127
+
128
+ describe('maintenance fingerprint backfill', () => {
129
+ it('backfills fingerprint field and inserts index entry', async () => {
130
+ const { hashSkillFiles } = await import('./lib/skills')
131
+ const expected = await hashSkillFiles([{ path: 'SKILL.md', sha256: 'abc' }])
132
+
133
+ const runQuery = vi.fn().mockResolvedValue({
134
+ items: [
135
+ {
136
+ skillId: 'skills:1',
137
+ versionId: 'skillVersions:1',
138
+ versionFingerprint: undefined,
139
+ files: [{ path: 'SKILL.md', sha256: 'abc' }],
140
+ existingEntries: [],
141
+ },
142
+ ],
143
+ cursor: null,
144
+ isDone: true,
145
+ })
146
+
147
+ const runMutation = vi.fn().mockResolvedValue({ ok: true })
148
+
149
+ const result = await backfillSkillFingerprintsInternalHandler(
150
+ { runQuery, runMutation } as never,
151
+ { dryRun: false, batchSize: 10, maxBatches: 1 },
152
+ )
153
+
154
+ expect(result.ok).toBe(true)
155
+ expect(result.stats.versionsScanned).toBe(1)
156
+ expect(result.stats.versionsPatched).toBe(1)
157
+ expect(result.stats.fingerprintsInserted).toBe(1)
158
+ expect(result.stats.fingerprintMismatches).toBe(0)
159
+ expect(runMutation).toHaveBeenCalledTimes(1)
160
+ expect(runMutation).toHaveBeenCalledWith(expect.anything(), {
161
+ versionId: 'skillVersions:1',
162
+ fingerprint: expected,
163
+ patchVersion: true,
164
+ replaceEntries: true,
165
+ existingEntryIds: [],
166
+ })
167
+ })
168
+
169
+ it('dryRun does not patch', async () => {
170
+ const runQuery = vi.fn().mockResolvedValue({
171
+ items: [
172
+ {
173
+ skillId: 'skills:1',
174
+ versionId: 'skillVersions:1',
175
+ versionFingerprint: undefined,
176
+ files: [{ path: 'SKILL.md', sha256: 'abc' }],
177
+ existingEntries: [],
178
+ },
179
+ ],
180
+ cursor: null,
181
+ isDone: true,
182
+ })
183
+
184
+ const runMutation = vi.fn()
185
+
186
+ const result = await backfillSkillFingerprintsInternalHandler(
187
+ { runQuery, runMutation } as never,
188
+ { dryRun: true, batchSize: 10, maxBatches: 1 },
189
+ )
190
+
191
+ expect(result.ok).toBe(true)
192
+ expect(result.stats.versionsPatched).toBe(1)
193
+ expect(result.stats.fingerprintsInserted).toBe(1)
194
+ expect(runMutation).not.toHaveBeenCalled()
195
+ })
196
+
197
+ it('patches missing version fingerprint without touching correct entries', async () => {
198
+ const { hashSkillFiles } = await import('./lib/skills')
199
+ const expected = await hashSkillFiles([{ path: 'SKILL.md', sha256: 'abc' }])
200
+
201
+ const runQuery = vi.fn().mockResolvedValue({
202
+ items: [
203
+ {
204
+ skillId: 'skills:1',
205
+ versionId: 'skillVersions:1',
206
+ versionFingerprint: undefined,
207
+ files: [{ path: 'SKILL.md', sha256: 'abc' }],
208
+ existingEntries: [{ id: 'skillVersionFingerprints:1', fingerprint: expected }],
209
+ },
210
+ ],
211
+ cursor: null,
212
+ isDone: true,
213
+ })
214
+
215
+ const runMutation = vi.fn().mockResolvedValue({ ok: true })
216
+
217
+ const result = await backfillSkillFingerprintsInternalHandler(
218
+ { runQuery, runMutation } as never,
219
+ { dryRun: false, batchSize: 10, maxBatches: 1 },
220
+ )
221
+
222
+ expect(result.ok).toBe(true)
223
+ expect(result.stats.versionsPatched).toBe(1)
224
+ expect(result.stats.fingerprintsInserted).toBe(0)
225
+ expect(result.stats.fingerprintMismatches).toBe(0)
226
+ expect(runMutation).toHaveBeenCalledWith(expect.anything(), {
227
+ versionId: 'skillVersions:1',
228
+ fingerprint: expected,
229
+ patchVersion: true,
230
+ replaceEntries: false,
231
+ existingEntryIds: [],
232
+ })
233
+ })
234
+
235
+ it('replaces mismatched fingerprint entries', async () => {
236
+ const { hashSkillFiles } = await import('./lib/skills')
237
+ const expected = await hashSkillFiles([{ path: 'SKILL.md', sha256: 'abc' }])
238
+
239
+ const runQuery = vi.fn().mockResolvedValue({
240
+ items: [
241
+ {
242
+ skillId: 'skills:1',
243
+ versionId: 'skillVersions:1',
244
+ versionFingerprint: 'wrong',
245
+ files: [{ path: 'SKILL.md', sha256: 'abc' }],
246
+ existingEntries: [{ id: 'skillVersionFingerprints:1', fingerprint: 'wrong' }],
247
+ },
248
+ ],
249
+ cursor: null,
250
+ isDone: true,
251
+ })
252
+
253
+ const runMutation = vi.fn().mockResolvedValue({ ok: true })
254
+
255
+ const result = await backfillSkillFingerprintsInternalHandler(
256
+ { runQuery, runMutation } as never,
257
+ { dryRun: false, batchSize: 10, maxBatches: 1 },
258
+ )
259
+
260
+ expect(result.ok).toBe(true)
261
+ expect(result.stats.fingerprintMismatches).toBe(1)
262
+ expect(runMutation).toHaveBeenCalledWith(expect.anything(), {
263
+ versionId: 'skillVersions:1',
264
+ fingerprint: expected,
265
+ patchVersion: true,
266
+ replaceEntries: true,
267
+ existingEntryIds: ['skillVersionFingerprints:1'],
268
+ })
269
+ })
270
+ })