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,305 @@
1
+ import {
2
+ ApiCliSkillDeleteResponseSchema,
3
+ ApiCliTelemetrySyncResponseSchema,
4
+ CliPublishRequestSchema,
5
+ CliSkillDeleteRequestSchema,
6
+ CliTelemetrySyncRequestSchema,
7
+ parseArk,
8
+ } from 'pilothub-schema'
9
+ import { api, internal } from './_generated/api'
10
+ import type { Id } from './_generated/dataModel'
11
+ import type { ActionCtx } from './_generated/server'
12
+ import { httpAction } from './_generated/server'
13
+ import { requireApiTokenUser } from './lib/apiTokenAuth'
14
+ import { publishVersionForUser } from './skills'
15
+
16
+ type SearchSkillEntry = {
17
+ score: number
18
+ skill: {
19
+ slug?: string
20
+ displayName?: string
21
+ summary?: string | null
22
+ updatedAt?: number
23
+ } | null
24
+ version: { version?: string } | null
25
+ }
26
+
27
+ type GetBySlugResult = {
28
+ skill: {
29
+ _id: Id<'skills'>
30
+ slug: string
31
+ displayName: string
32
+ summary?: string
33
+ tags: Record<string, string>
34
+ stats: unknown
35
+ createdAt: number
36
+ updatedAt: number
37
+ } | null
38
+ latestVersion: { version: string; createdAt: number; changelog: string } | null
39
+ owner: { handle?: string; displayName?: string; image?: string } | null
40
+ } | null
41
+
42
+ async function searchSkillsHandler(ctx: ActionCtx, request: Request) {
43
+ const url = new URL(request.url)
44
+ const query = url.searchParams.get('q')?.trim() ?? ''
45
+ const limit = toOptionalNumber(url.searchParams.get('limit'))
46
+ const approvedOnly = url.searchParams.get('approvedOnly') === 'true'
47
+ const highlightedOnly = url.searchParams.get('highlightedOnly') === 'true' || approvedOnly
48
+
49
+ if (!query) return json({ results: [] })
50
+
51
+ const results = (await ctx.runAction(api.search.searchSkills, {
52
+ query,
53
+ limit,
54
+ highlightedOnly: highlightedOnly || undefined,
55
+ })) as SearchSkillEntry[]
56
+
57
+ return json({
58
+ results: results.map((result) => ({
59
+ score: result.score,
60
+ slug: result.skill?.slug,
61
+ displayName: result.skill?.displayName,
62
+ summary: result.skill?.summary ?? null,
63
+ version: result.version?.version ?? null,
64
+ updatedAt: result.skill?.updatedAt,
65
+ })),
66
+ })
67
+ }
68
+
69
+ export const searchSkillsHttp = httpAction(searchSkillsHandler)
70
+
71
+ async function getSkillHandler(ctx: ActionCtx, request: Request) {
72
+ const url = new URL(request.url)
73
+ const slug = url.searchParams.get('slug')?.trim().toLowerCase()
74
+ if (!slug) return text('Missing slug', 400)
75
+
76
+ const result = (await ctx.runQuery(api.skills.getBySlug, { slug })) as GetBySlugResult
77
+ if (!result?.skill) return text('Skill not found', 404)
78
+
79
+ return json({
80
+ skill: {
81
+ slug: result.skill.slug,
82
+ displayName: result.skill.displayName,
83
+ summary: result.skill.summary ?? null,
84
+ tags: result.skill.tags,
85
+ stats: result.skill.stats,
86
+ createdAt: result.skill.createdAt,
87
+ updatedAt: result.skill.updatedAt,
88
+ },
89
+ latestVersion: result.latestVersion
90
+ ? {
91
+ version: result.latestVersion.version,
92
+ createdAt: result.latestVersion.createdAt,
93
+ changelog: result.latestVersion.changelog,
94
+ }
95
+ : null,
96
+ owner: result.owner
97
+ ? {
98
+ handle: result.owner.handle ?? null,
99
+ displayName: result.owner.displayName ?? null,
100
+ image: result.owner.image ?? null,
101
+ }
102
+ : null,
103
+ })
104
+ }
105
+
106
+ export const getSkillHttp = httpAction(getSkillHandler)
107
+
108
+ async function resolveSkillVersionHandler(ctx: ActionCtx, request: Request) {
109
+ const url = new URL(request.url)
110
+ const slug = url.searchParams.get('slug')?.trim().toLowerCase()
111
+ const hash = url.searchParams.get('hash')?.trim().toLowerCase()
112
+ if (!slug || !hash) return text('Missing slug or hash', 400)
113
+ if (!/^[a-f0-9]{64}$/.test(hash)) return text('Invalid hash', 400)
114
+
115
+ const resolved = await ctx.runQuery(api.skills.resolveVersionByHash, { slug, hash })
116
+ if (!resolved) return text('Skill not found', 404)
117
+
118
+ return json({ slug, match: resolved.match, latestVersion: resolved.latestVersion })
119
+ }
120
+
121
+ export const resolveSkillVersionHttp = httpAction(resolveSkillVersionHandler)
122
+
123
+ async function cliWhoamiHandler(ctx: ActionCtx, request: Request) {
124
+ try {
125
+ const { user } = await requireApiTokenUser(ctx, request)
126
+ return json({
127
+ user: {
128
+ handle: user.handle ?? null,
129
+ displayName: user.displayName ?? null,
130
+ image: user.image ?? null,
131
+ },
132
+ })
133
+ } catch {
134
+ return text('Unauthorized', 401)
135
+ }
136
+ }
137
+
138
+ export const cliWhoamiHttp = httpAction(cliWhoamiHandler)
139
+
140
+ async function cliUploadUrlHandler(ctx: ActionCtx, request: Request) {
141
+ try {
142
+ const { userId } = await requireApiTokenUser(ctx, request)
143
+ const uploadUrl = await ctx.runMutation(internal.uploads.generateUploadUrlForUserInternal, {
144
+ userId,
145
+ })
146
+ return json({ uploadUrl })
147
+ } catch {
148
+ return text('Unauthorized', 401)
149
+ }
150
+ }
151
+
152
+ export const cliUploadUrlHttp = httpAction(cliUploadUrlHandler)
153
+
154
+ async function cliPublishHandler(ctx: ActionCtx, request: Request) {
155
+ let body: unknown
156
+ try {
157
+ body = await request.json()
158
+ } catch {
159
+ return text('Invalid JSON', 400)
160
+ }
161
+
162
+ try {
163
+ const { userId } = await requireApiTokenUser(ctx, request)
164
+ const args = parsePublishBody(body)
165
+ const result = await publishVersionForUser(ctx, userId, args)
166
+ return json({ ok: true, ...result })
167
+ } catch (error) {
168
+ const message = error instanceof Error ? error.message : 'Publish failed'
169
+ if (message.toLowerCase().includes('unauthorized')) return text('Unauthorized', 401)
170
+ return text(message, 400)
171
+ }
172
+ }
173
+
174
+ export const cliPublishHttp = httpAction(cliPublishHandler)
175
+
176
+ async function cliSkillDeleteHandler(ctx: ActionCtx, request: Request, deleted: boolean) {
177
+ let body: unknown
178
+ try {
179
+ body = await request.json()
180
+ } catch {
181
+ return text('Invalid JSON', 400)
182
+ }
183
+
184
+ try {
185
+ const { userId } = await requireApiTokenUser(ctx, request)
186
+ const args = parseArk(CliSkillDeleteRequestSchema, body, 'Delete payload')
187
+ await ctx.runMutation(internal.skills.setSkillSoftDeletedInternal, {
188
+ userId,
189
+ slug: args.slug,
190
+ deleted,
191
+ })
192
+ const ok = parseArk(ApiCliSkillDeleteResponseSchema, { ok: true }, 'Delete response')
193
+ return json(ok)
194
+ } catch (error) {
195
+ const message = error instanceof Error ? error.message : 'Delete failed'
196
+ if (message.toLowerCase().includes('unauthorized')) return text('Unauthorized', 401)
197
+ return text(message, 400)
198
+ }
199
+ }
200
+
201
+ export const cliSkillDeleteHttp = httpAction((ctx, request) =>
202
+ cliSkillDeleteHandler(ctx, request, true),
203
+ )
204
+ export const cliSkillUndeleteHttp = httpAction((ctx, request) =>
205
+ cliSkillDeleteHandler(ctx, request, false),
206
+ )
207
+
208
+ async function cliTelemetrySyncHandler(ctx: ActionCtx, request: Request) {
209
+ let body: unknown
210
+ try {
211
+ body = await request.json()
212
+ } catch {
213
+ return text('Invalid JSON', 400)
214
+ }
215
+
216
+ try {
217
+ const { userId } = await requireApiTokenUser(ctx, request)
218
+ const args = parseArk(CliTelemetrySyncRequestSchema, body, 'Telemetry payload')
219
+ await ctx.runMutation(internal.telemetry.reportCliSyncInternal, {
220
+ userId,
221
+ roots: args.roots.map((root) => ({
222
+ rootId: root.rootId,
223
+ label: root.label,
224
+ skills: root.skills.map((skill) => ({
225
+ slug: skill.slug,
226
+ version: skill.version ?? undefined,
227
+ })),
228
+ })),
229
+ })
230
+ const ok = parseArk(ApiCliTelemetrySyncResponseSchema, { ok: true }, 'Telemetry response')
231
+ return json(ok)
232
+ } catch (error) {
233
+ const message = error instanceof Error ? error.message : 'Telemetry failed'
234
+ if (message.toLowerCase().includes('unauthorized')) return text('Unauthorized', 401)
235
+ return text(message, 400)
236
+ }
237
+ }
238
+
239
+ export const cliTelemetrySyncHttp = httpAction(cliTelemetrySyncHandler)
240
+
241
+ function json(value: unknown, status = 200) {
242
+ return new Response(JSON.stringify(value), {
243
+ status,
244
+ headers: {
245
+ 'Content-Type': 'application/json',
246
+ 'Cache-Control': 'no-store',
247
+ },
248
+ })
249
+ }
250
+
251
+ function text(value: string, status: number) {
252
+ return new Response(value, {
253
+ status,
254
+ headers: {
255
+ 'Content-Type': 'text/plain; charset=utf-8',
256
+ 'Cache-Control': 'no-store',
257
+ },
258
+ })
259
+ }
260
+
261
+ function toOptionalNumber(value: string | null) {
262
+ if (!value) return undefined
263
+ const parsed = Number.parseInt(value, 10)
264
+ return Number.isFinite(parsed) ? parsed : undefined
265
+ }
266
+
267
+ function parsePublishBody(body: unknown) {
268
+ const parsed = parseArk(CliPublishRequestSchema, body, 'Publish payload')
269
+ if (parsed.files.length === 0) throw new Error('files required')
270
+ const tags = parsed.tags && parsed.tags.length > 0 ? parsed.tags : undefined
271
+ return {
272
+ slug: parsed.slug,
273
+ displayName: parsed.displayName,
274
+ version: parsed.version,
275
+ changelog: parsed.changelog,
276
+ tags,
277
+ source: parsed.source ?? undefined,
278
+ forkOf: parsed.forkOf
279
+ ? {
280
+ slug: parsed.forkOf.slug,
281
+ version: parsed.forkOf.version ?? undefined,
282
+ }
283
+ : undefined,
284
+ files: parsed.files.map((file) => ({
285
+ ...file,
286
+ storageId: file.storageId as Id<'_storage'>,
287
+ })),
288
+ }
289
+ }
290
+
291
+ export const __test = {
292
+ parsePublishBody,
293
+ toOptionalNumber,
294
+ }
295
+
296
+ export const __handlers = {
297
+ searchSkillsHandler,
298
+ getSkillHandler,
299
+ resolveSkillVersionHandler,
300
+ cliWhoamiHandler,
301
+ cliUploadUrlHandler,
302
+ cliPublishHandler,
303
+ cliSkillDeleteHandler,
304
+ cliTelemetrySyncHandler,
305
+ }