pilothub 0.0.1 → 0.0.2

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 (388) hide show
  1. package/LICENSE +1 -0
  2. package/README.md +36 -129
  3. package/dist/browserAuth.d.ts +20 -0
  4. package/dist/browserAuth.js +156 -0
  5. package/dist/browserAuth.js.map +1 -0
  6. package/dist/browserAuth.test.d.ts +1 -0
  7. package/dist/browserAuth.test.js +83 -0
  8. package/dist/browserAuth.test.js.map +1 -0
  9. package/dist/cli/buildInfo.d.ts +3 -0
  10. package/dist/cli/buildInfo.js +103 -0
  11. package/dist/cli/buildInfo.js.map +1 -0
  12. package/dist/cli/commands/auth.d.ts +9 -0
  13. package/dist/cli/commands/auth.js +75 -0
  14. package/dist/cli/commands/auth.js.map +1 -0
  15. package/dist/cli/commands/delete.d.ts +11 -0
  16. package/dist/cli/commands/delete.js +67 -0
  17. package/dist/cli/commands/delete.js.map +1 -0
  18. package/dist/cli/commands/delete.test.d.ts +1 -0
  19. package/dist/cli/commands/delete.test.js +52 -0
  20. package/dist/cli/commands/delete.test.js.map +1 -0
  21. package/dist/cli/commands/publish.d.ts +9 -0
  22. package/dist/cli/commands/publish.js +87 -0
  23. package/dist/cli/commands/publish.js.map +1 -0
  24. package/dist/cli/commands/publish.test.d.ts +1 -0
  25. package/dist/cli/commands/publish.test.js +104 -0
  26. package/dist/cli/commands/publish.test.js.map +1 -0
  27. package/dist/cli/commands/skills.d.ts +23 -0
  28. package/dist/cli/commands/skills.js +298 -0
  29. package/dist/cli/commands/skills.js.map +1 -0
  30. package/dist/cli/commands/skills.test.d.ts +1 -0
  31. package/dist/cli/commands/skills.test.js +156 -0
  32. package/dist/cli/commands/skills.test.js.map +1 -0
  33. package/dist/cli/commands/star.d.ts +8 -0
  34. package/dist/cli/commands/star.js +38 -0
  35. package/dist/cli/commands/star.js.map +1 -0
  36. package/dist/cli/commands/sync.d.ts +3 -0
  37. package/dist/cli/commands/sync.js +160 -0
  38. package/dist/cli/commands/sync.js.map +1 -0
  39. package/dist/cli/commands/sync.test.d.ts +1 -0
  40. package/dist/cli/commands/sync.test.js +277 -0
  41. package/dist/cli/commands/sync.test.js.map +1 -0
  42. package/dist/cli/commands/syncHelpers.d.ts +76 -0
  43. package/dist/cli/commands/syncHelpers.js +349 -0
  44. package/dist/cli/commands/syncHelpers.js.map +1 -0
  45. package/dist/cli/commands/syncHelpers.test.d.ts +1 -0
  46. package/dist/cli/commands/syncHelpers.test.js +22 -0
  47. package/dist/cli/commands/syncHelpers.test.js.map +1 -0
  48. package/dist/cli/commands/syncTypes.d.ts +24 -0
  49. package/dist/cli/commands/syncTypes.js +2 -0
  50. package/dist/cli/commands/syncTypes.js.map +1 -0
  51. package/dist/cli/commands/unstar.d.ts +8 -0
  52. package/dist/cli/commands/unstar.js +38 -0
  53. package/dist/cli/commands/unstar.js.map +1 -0
  54. package/dist/cli/helpStyle.d.ts +13 -0
  55. package/dist/cli/helpStyle.js +38 -0
  56. package/dist/cli/helpStyle.js.map +1 -0
  57. package/dist/cli/pilotbotConfig.d.ts +6 -0
  58. package/dist/cli/pilotbotConfig.js +110 -0
  59. package/dist/cli/pilotbotConfig.js.map +1 -0
  60. package/dist/cli/pilotbotConfig.test.d.ts +1 -0
  61. package/dist/cli/pilotbotConfig.test.js +133 -0
  62. package/dist/cli/pilotbotConfig.test.js.map +1 -0
  63. package/dist/cli/registry.d.ts +7 -0
  64. package/dist/cli/registry.js +42 -0
  65. package/dist/cli/registry.js.map +1 -0
  66. package/dist/cli/registry.test.d.ts +1 -0
  67. package/dist/cli/registry.test.js +48 -0
  68. package/dist/cli/registry.test.js.map +1 -0
  69. package/dist/cli/scanSkills.d.ts +7 -0
  70. package/dist/cli/scanSkills.js +75 -0
  71. package/dist/cli/scanSkills.js.map +1 -0
  72. package/dist/cli/scanSkills.test.d.ts +1 -0
  73. package/dist/cli/scanSkills.test.js +60 -0
  74. package/dist/cli/scanSkills.test.js.map +1 -0
  75. package/dist/cli/slug.d.ts +2 -0
  76. package/dist/cli/slug.js +16 -0
  77. package/dist/cli/slug.js.map +1 -0
  78. package/dist/cli/types.d.ts +15 -0
  79. package/dist/cli/types.js +2 -0
  80. package/dist/cli/types.js.map +1 -0
  81. package/dist/cli/ui.d.ts +7 -0
  82. package/dist/cli/ui.js +72 -0
  83. package/dist/cli/ui.js.map +1 -0
  84. package/dist/cli.d.ts +2 -0
  85. package/dist/cli.js +268 -0
  86. package/dist/cli.js.map +1 -0
  87. package/dist/config.d.ts +4 -0
  88. package/dist/config.js +38 -0
  89. package/dist/config.js.map +1 -0
  90. package/dist/discovery.d.ts +5 -0
  91. package/dist/discovery.js +21 -0
  92. package/dist/discovery.js.map +1 -0
  93. package/dist/discovery.test.d.ts +1 -0
  94. package/dist/discovery.test.js +46 -0
  95. package/dist/discovery.test.js.map +1 -0
  96. package/dist/http.d.ts +32 -0
  97. package/dist/http.js +261 -0
  98. package/dist/http.js.map +1 -0
  99. package/dist/http.test.d.ts +1 -0
  100. package/dist/http.test.js +135 -0
  101. package/dist/http.test.js.map +1 -0
  102. package/dist/schema/ark.js.map +1 -0
  103. package/dist/schema/index.js.map +1 -0
  104. package/dist/schema/routes.js.map +1 -0
  105. package/{packages/schema/dist → dist/schema}/schemas.d.ts +0 -39
  106. package/{packages/schema/dist → dist/schema}/schemas.js +0 -22
  107. package/dist/schema/schemas.js.map +1 -0
  108. package/dist/schema/textFiles.js.map +1 -0
  109. package/dist/schema/textFiles.test.d.ts +1 -0
  110. package/dist/schema/textFiles.test.js +20 -0
  111. package/dist/schema/textFiles.test.js.map +1 -0
  112. package/dist/skills.d.ts +43 -0
  113. package/dist/skills.js +163 -0
  114. package/dist/skills.js.map +1 -0
  115. package/dist/skills.test.d.ts +1 -0
  116. package/dist/skills.test.js +144 -0
  117. package/dist/skills.test.js.map +1 -0
  118. package/dist/types.d.ts +7 -0
  119. package/dist/types.js +2 -0
  120. package/dist/types.js.map +1 -0
  121. package/package.json +27 -70
  122. package/.env.local.example +0 -19
  123. package/.github/workflows/ci.yml +0 -40
  124. package/.oxlintrc.json +0 -3
  125. package/AGENTS.md +0 -45
  126. package/CHANGELOG.md +0 -138
  127. package/DEPRECATIONS.md +0 -7
  128. package/biome.json +0 -41
  129. package/convex/_generated/api.d.ts +0 -153
  130. package/convex/_generated/api.js +0 -23
  131. package/convex/_generated/dataModel.d.ts +0 -60
  132. package/convex/_generated/server.d.ts +0 -143
  133. package/convex/_generated/server.js +0 -93
  134. package/convex/auth.config.ts +0 -8
  135. package/convex/auth.ts +0 -19
  136. package/convex/comments.ts +0 -88
  137. package/convex/crons.ts +0 -34
  138. package/convex/devSeed.ts +0 -459
  139. package/convex/devSeedExtra.ts +0 -541
  140. package/convex/downloads.ts +0 -78
  141. package/convex/githubBackups.ts +0 -170
  142. package/convex/githubBackupsNode.ts +0 -183
  143. package/convex/githubImport.ts +0 -317
  144. package/convex/githubSoulBackups.ts +0 -170
  145. package/convex/githubSoulBackupsNode.ts +0 -186
  146. package/convex/http.ts +0 -194
  147. package/convex/httpApi.handlers.test.ts +0 -488
  148. package/convex/httpApi.test.ts +0 -70
  149. package/convex/httpApi.ts +0 -305
  150. package/convex/httpApiV1.handlers.test.ts +0 -584
  151. package/convex/httpApiV1.ts +0 -1172
  152. package/convex/leaderboards.ts +0 -39
  153. package/convex/lib/access.ts +0 -36
  154. package/convex/lib/apiTokenAuth.ts +0 -36
  155. package/convex/lib/badges.ts +0 -50
  156. package/convex/lib/changelog.test.ts +0 -34
  157. package/convex/lib/changelog.ts +0 -278
  158. package/convex/lib/embeddings.ts +0 -38
  159. package/convex/lib/githubBackup.ts +0 -443
  160. package/convex/lib/githubImport.test.ts +0 -247
  161. package/convex/lib/githubImport.ts +0 -425
  162. package/convex/lib/githubSoulBackup.ts +0 -443
  163. package/convex/lib/leaderboards.ts +0 -103
  164. package/convex/lib/moderation.ts +0 -42
  165. package/convex/lib/public.ts +0 -89
  166. package/convex/lib/searchText.test.ts +0 -46
  167. package/convex/lib/searchText.ts +0 -27
  168. package/convex/lib/skillBackfill.test.ts +0 -34
  169. package/convex/lib/skillBackfill.ts +0 -67
  170. package/convex/lib/skillPublish.test.ts +0 -28
  171. package/convex/lib/skillPublish.ts +0 -284
  172. package/convex/lib/skillStats.ts +0 -80
  173. package/convex/lib/skills.test.ts +0 -197
  174. package/convex/lib/skills.ts +0 -273
  175. package/convex/lib/soulChangelog.ts +0 -273
  176. package/convex/lib/soulPublish.ts +0 -236
  177. package/convex/lib/tokens.test.ts +0 -33
  178. package/convex/lib/tokens.ts +0 -51
  179. package/convex/lib/webhooks.test.ts +0 -91
  180. package/convex/lib/webhooks.ts +0 -112
  181. package/convex/maintenance.test.ts +0 -270
  182. package/convex/maintenance.ts +0 -840
  183. package/convex/rateLimits.ts +0 -50
  184. package/convex/schema.ts +0 -472
  185. package/convex/search.test.ts +0 -12
  186. package/convex/search.ts +0 -254
  187. package/convex/seed.test.ts +0 -37
  188. package/convex/seed.ts +0 -254
  189. package/convex/seedSouls.ts +0 -111
  190. package/convex/skillStatEvents.ts +0 -568
  191. package/convex/skills.ts +0 -1606
  192. package/convex/soulComments.ts +0 -88
  193. package/convex/soulDownloads.ts +0 -14
  194. package/convex/soulStars.ts +0 -71
  195. package/convex/souls.ts +0 -570
  196. package/convex/stars.ts +0 -108
  197. package/convex/statsMaintenance.ts +0 -205
  198. package/convex/telemetry.ts +0 -434
  199. package/convex/tokens.ts +0 -88
  200. package/convex/tsconfig.json +0 -7
  201. package/convex/uploads.ts +0 -20
  202. package/convex/users.ts +0 -122
  203. package/convex/webhooks.ts +0 -50
  204. package/convex.json +0 -3
  205. package/docs/README.md +0 -32
  206. package/docs/api.md +0 -51
  207. package/docs/architecture.md +0 -61
  208. package/docs/auth.md +0 -54
  209. package/docs/cli.md +0 -117
  210. package/docs/deploy.md +0 -78
  211. package/docs/diffing.md +0 -84
  212. package/docs/github-import.md +0 -171
  213. package/docs/http-api.md +0 -187
  214. package/docs/manual-testing.md +0 -64
  215. package/docs/mintlify.md +0 -43
  216. package/docs/quickstart.md +0 -120
  217. package/docs/skill-format.md +0 -58
  218. package/docs/soul-format.md +0 -37
  219. package/docs/spec.md +0 -177
  220. package/docs/telemetry.md +0 -91
  221. package/docs/troubleshooting.md +0 -49
  222. package/docs/webhook.md +0 -51
  223. package/e2e/menu-smoke.pw.test.ts +0 -49
  224. package/e2e/pilothub.e2e.test.ts +0 -494
  225. package/e2e/search-exact.pw.test.ts +0 -97
  226. package/packages/pilothub/LICENSE +0 -22
  227. package/packages/pilothub/README.md +0 -57
  228. package/packages/pilothub/package.json +0 -41
  229. package/packages/pilothub/src/browserAuth.test.ts +0 -96
  230. package/packages/pilothub/src/browserAuth.ts +0 -174
  231. package/packages/pilothub/src/cli/buildInfo.ts +0 -94
  232. package/packages/pilothub/src/cli/commands/auth.ts +0 -97
  233. package/packages/pilothub/src/cli/commands/delete.test.ts +0 -73
  234. package/packages/pilothub/src/cli/commands/delete.ts +0 -83
  235. package/packages/pilothub/src/cli/commands/publish.test.ts +0 -122
  236. package/packages/pilothub/src/cli/commands/publish.ts +0 -108
  237. package/packages/pilothub/src/cli/commands/skills.test.ts +0 -191
  238. package/packages/pilothub/src/cli/commands/skills.ts +0 -380
  239. package/packages/pilothub/src/cli/commands/star.ts +0 -46
  240. package/packages/pilothub/src/cli/commands/sync.test.ts +0 -310
  241. package/packages/pilothub/src/cli/commands/sync.ts +0 -200
  242. package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +0 -26
  243. package/packages/pilothub/src/cli/commands/syncHelpers.ts +0 -427
  244. package/packages/pilothub/src/cli/commands/syncTypes.ts +0 -27
  245. package/packages/pilothub/src/cli/commands/unstar.ts +0 -48
  246. package/packages/pilothub/src/cli/helpStyle.ts +0 -45
  247. package/packages/pilothub/src/cli/pilotbotConfig.test.ts +0 -159
  248. package/packages/pilothub/src/cli/pilotbotConfig.ts +0 -147
  249. package/packages/pilothub/src/cli/registry.test.ts +0 -63
  250. package/packages/pilothub/src/cli/registry.ts +0 -43
  251. package/packages/pilothub/src/cli/scanSkills.test.ts +0 -64
  252. package/packages/pilothub/src/cli/scanSkills.ts +0 -84
  253. package/packages/pilothub/src/cli/slug.ts +0 -16
  254. package/packages/pilothub/src/cli/types.ts +0 -12
  255. package/packages/pilothub/src/cli/ui.ts +0 -75
  256. package/packages/pilothub/src/cli.ts +0 -311
  257. package/packages/pilothub/src/config.ts +0 -36
  258. package/packages/pilothub/src/discovery.test.ts +0 -75
  259. package/packages/pilothub/src/discovery.ts +0 -19
  260. package/packages/pilothub/src/http.test.ts +0 -156
  261. package/packages/pilothub/src/http.ts +0 -301
  262. package/packages/pilothub/src/schema/ark.ts +0 -29
  263. package/packages/pilothub/src/schema/index.ts +0 -5
  264. package/packages/pilothub/src/schema/routes.ts +0 -22
  265. package/packages/pilothub/src/schema/schemas.ts +0 -260
  266. package/packages/pilothub/src/schema/textFiles.test.ts +0 -23
  267. package/packages/pilothub/src/schema/textFiles.ts +0 -66
  268. package/packages/pilothub/src/skills.test.ts +0 -191
  269. package/packages/pilothub/src/skills.ts +0 -172
  270. package/packages/pilothub/src/types.ts +0 -10
  271. package/packages/pilothub/tsconfig.json +0 -14
  272. package/packages/schema/README.md +0 -3
  273. package/packages/schema/dist/ark.js.map +0 -1
  274. package/packages/schema/dist/index.js.map +0 -1
  275. package/packages/schema/dist/routes.js.map +0 -1
  276. package/packages/schema/dist/schemas.js.map +0 -1
  277. package/packages/schema/dist/textFiles.js.map +0 -1
  278. package/packages/schema/package.json +0 -26
  279. package/packages/schema/src/ark.ts +0 -29
  280. package/packages/schema/src/index.ts +0 -5
  281. package/packages/schema/src/routes.ts +0 -22
  282. package/packages/schema/src/schemas.test.ts +0 -123
  283. package/packages/schema/src/schemas.ts +0 -287
  284. package/packages/schema/src/textFiles.test.ts +0 -23
  285. package/packages/schema/src/textFiles.ts +0 -66
  286. package/packages/schema/tsconfig.json +0 -15
  287. package/pilothub +0 -46
  288. package/playwright.config.ts +0 -33
  289. package/public/.well-known/pilothub.json +0 -6
  290. package/public/api/v1/openapi.json +0 -379
  291. package/public/favicon.ico +0 -0
  292. package/public/logo192.png +0 -0
  293. package/public/logo512.png +0 -0
  294. package/public/manifest.json +0 -25
  295. package/public/og.png +0 -0
  296. package/public/og.svg +0 -98
  297. package/public/pilot-logo.png +0 -0
  298. package/public/pilot-mark.png +0 -0
  299. package/public/robots.txt +0 -3
  300. package/public/tanstack-circle-logo.png +0 -0
  301. package/public/tanstack-word-logo-white.svg +0 -1
  302. package/scripts/check-peer-deps.ts +0 -56
  303. package/scripts/docs-list.ts +0 -148
  304. package/scripts/run-playwright-local.sh +0 -14
  305. package/server/og/fetchSkillOgMeta.ts +0 -27
  306. package/server/og/fetchSoulOgMeta.ts +0 -27
  307. package/server/og/ogAssets.ts +0 -80
  308. package/server/og/skillOgSvg.test.ts +0 -59
  309. package/server/og/skillOgSvg.ts +0 -258
  310. package/server/og/soulOgSvg.ts +0 -209
  311. package/server/routes/og/skill.png.ts +0 -103
  312. package/server/routes/og/soul.png.ts +0 -111
  313. package/src/__tests__/skill-detail-page.test.tsx +0 -86
  314. package/src/__tests__/skills-index.test.tsx +0 -145
  315. package/src/__tests__/upload.route.test.tsx +0 -228
  316. package/src/components/AppProviders.tsx +0 -19
  317. package/src/components/ClientOnly.tsx +0 -18
  318. package/src/components/Footer.tsx +0 -29
  319. package/src/components/Header.tsx +0 -295
  320. package/src/components/InstallSwitcher.tsx +0 -53
  321. package/src/components/SkillCard.tsx +0 -36
  322. package/src/components/SkillDetailPage.tsx +0 -817
  323. package/src/components/SkillDiffCard.tsx +0 -485
  324. package/src/components/SoulCard.tsx +0 -19
  325. package/src/components/SoulDetailPage.tsx +0 -263
  326. package/src/components/UserBootstrap.tsx +0 -18
  327. package/src/components/ui/dropdown-menu.tsx +0 -67
  328. package/src/components/ui/toggle-group.tsx +0 -35
  329. package/src/convex/client.ts +0 -3
  330. package/src/lib/badges.ts +0 -29
  331. package/src/lib/diffing.test.ts +0 -163
  332. package/src/lib/diffing.ts +0 -106
  333. package/src/lib/gravatar.test.ts +0 -9
  334. package/src/lib/gravatar.ts +0 -158
  335. package/src/lib/og.test.ts +0 -142
  336. package/src/lib/og.ts +0 -156
  337. package/src/lib/publicUser.ts +0 -39
  338. package/src/lib/roles.ts +0 -19
  339. package/src/lib/site.test.ts +0 -130
  340. package/src/lib/site.ts +0 -84
  341. package/src/lib/theme-transition.test.ts +0 -134
  342. package/src/lib/theme-transition.ts +0 -134
  343. package/src/lib/theme.test.tsx +0 -88
  344. package/src/lib/theme.ts +0 -43
  345. package/src/lib/uploadFiles.jsdom.test.ts +0 -33
  346. package/src/lib/uploadFiles.test.ts +0 -123
  347. package/src/lib/uploadFiles.ts +0 -245
  348. package/src/lib/uploadUtils.test.ts +0 -78
  349. package/src/lib/uploadUtils.ts +0 -93
  350. package/src/lib/useAuthStatus.ts +0 -12
  351. package/src/lib/utils.test.ts +0 -9
  352. package/src/lib/utils.ts +0 -6
  353. package/src/logo.svg +0 -12
  354. package/src/routeTree.gen.ts +0 -345
  355. package/src/router.tsx +0 -17
  356. package/src/routes/$owner/$slug.tsx +0 -55
  357. package/src/routes/__root.tsx +0 -136
  358. package/src/routes/admin.tsx +0 -11
  359. package/src/routes/cli/auth.tsx +0 -168
  360. package/src/routes/dashboard.tsx +0 -97
  361. package/src/routes/import.tsx +0 -415
  362. package/src/routes/index.tsx +0 -252
  363. package/src/routes/management.tsx +0 -529
  364. package/src/routes/settings.tsx +0 -203
  365. package/src/routes/skills/index.tsx +0 -422
  366. package/src/routes/souls/$slug.tsx +0 -55
  367. package/src/routes/souls/index.tsx +0 -243
  368. package/src/routes/stars.tsx +0 -68
  369. package/src/routes/u/$handle.tsx +0 -307
  370. package/src/routes/upload/utils.ts +0 -81
  371. package/src/routes/upload.tsx +0 -499
  372. package/src/styles.css +0 -2718
  373. package/tsconfig.json +0 -24
  374. package/tsconfig.oxlint.json +0 -16
  375. package/vercel.json +0 -8
  376. package/vite.config.ts +0 -48
  377. package/vitest.config.ts +0 -47
  378. package/vitest.e2e.config.ts +0 -11
  379. package/vitest.setup.ts +0 -1
  380. /package/{packages/pilothub/bin → bin}/pilothub.js +0 -0
  381. /package/{packages/schema/dist → dist/schema}/ark.d.ts +0 -0
  382. /package/{packages/schema/dist → dist/schema}/ark.js +0 -0
  383. /package/{packages/schema/dist → dist/schema}/index.d.ts +0 -0
  384. /package/{packages/schema/dist → dist/schema}/index.js +0 -0
  385. /package/{packages/schema/dist → dist/schema}/routes.d.ts +0 -0
  386. /package/{packages/schema/dist → dist/schema}/routes.js +0 -0
  387. /package/{packages/schema/dist → dist/schema}/textFiles.d.ts +0 -0
  388. /package/{packages/schema/dist → dist/schema}/textFiles.js +0 -0
@@ -1,499 +0,0 @@
1
- import { createFileRoute, useNavigate, useSearch } from '@tanstack/react-router'
2
- import { useAction, useMutation, useQuery } from 'convex/react'
3
- import { useEffect, useMemo, useRef, useState } from 'react'
4
- import semver from 'semver'
5
- import { api } from '../../convex/_generated/api'
6
- import { getSiteMode } from '../lib/site'
7
- import { expandDroppedItems, expandFiles } from '../lib/uploadFiles'
8
- import { useAuthStatus } from '../lib/useAuthStatus'
9
- import {
10
- formatBytes,
11
- formatPublishError,
12
- hashFile,
13
- isTextFile,
14
- readText,
15
- uploadFile,
16
- } from './upload/utils'
17
-
18
- const SLUG_PATTERN = /^[a-z0-9]+(?:-[a-z0-9]+)*$/
19
-
20
- export const Route = createFileRoute('/upload')({
21
- validateSearch: (search) => ({
22
- updateSlug: typeof search.updateSlug === 'string' ? search.updateSlug : undefined,
23
- }),
24
- component: Upload,
25
- })
26
-
27
- export function Upload() {
28
- const { isAuthenticated, me } = useAuthStatus()
29
- const { updateSlug } = useSearch({ from: '/upload' })
30
- const siteMode = getSiteMode()
31
- const isSoulMode = siteMode === 'souls'
32
- const requiredFileLabel = isSoulMode ? 'SOUL.md' : 'SKILL.md'
33
- const contentLabel = isSoulMode ? 'soul' : 'skill'
34
-
35
- const generateUploadUrl = useMutation(api.uploads.generateUploadUrl)
36
- const publishVersion = useAction(
37
- isSoulMode ? api.souls.publishVersion : api.skills.publishVersion,
38
- )
39
- const generateChangelogPreview = useAction(
40
- isSoulMode ? api.souls.generateChangelogPreview : api.skills.generateChangelogPreview,
41
- )
42
- const existingSkill = useQuery(
43
- api.skills.getBySlug,
44
- !isSoulMode && updateSlug ? { slug: updateSlug } : 'skip',
45
- )
46
- const existingSoul = useQuery(
47
- api.souls.getBySlug,
48
- isSoulMode && updateSlug ? { slug: updateSlug } : 'skip',
49
- )
50
- const existing = (isSoulMode ? existingSoul : existingSkill) as
51
- | {
52
- skill?: { slug: string; displayName: string }
53
- soul?: { slug: string; displayName: string }
54
- latestVersion?: { version: string }
55
- }
56
- | null
57
- | undefined
58
-
59
- const [hasAttempted, setHasAttempted] = useState(false)
60
- const [files, setFiles] = useState<File[]>([])
61
- const [slug, setSlug] = useState(updateSlug ?? '')
62
- const [displayName, setDisplayName] = useState('')
63
- const [version, setVersion] = useState('1.0.0')
64
- const [tags, setTags] = useState('latest')
65
- const [changelog, setChangelog] = useState('')
66
- const [changelogStatus, setChangelogStatus] = useState<'idle' | 'loading' | 'ready' | 'error'>(
67
- 'idle',
68
- )
69
- const [changelogSource, setChangelogSource] = useState<'auto' | 'user' | null>(null)
70
- const changelogTouchedRef = useRef(false)
71
- const changelogRequestRef = useRef(0)
72
- const changelogKeyRef = useRef<string | null>(null)
73
- const [status, setStatus] = useState<string | null>(null)
74
- const isSubmitting = status !== null
75
- const [error, setError] = useState<string | null>(null)
76
- const [isDragging, setIsDragging] = useState(false)
77
- const fileInputRef = useRef<HTMLInputElement | null>(null)
78
- const validationRef = useRef<HTMLDivElement | null>(null)
79
- const navigate = useNavigate()
80
- const maxBytes = 50 * 1024 * 1024
81
- const totalBytes = useMemo(() => files.reduce((sum, file) => sum + file.size, 0), [files])
82
- const stripRoot = useMemo(() => {
83
- if (files.length === 0) return null
84
- const paths = files.map((file) => (file.webkitRelativePath || file.name).replace(/^\.\//, ''))
85
- if (!paths.every((path) => path.includes('/'))) return null
86
- const firstSegment = paths[0]?.split('/')[0]
87
- if (!firstSegment) return null
88
- if (!paths.every((path) => path.startsWith(`${firstSegment}/`))) return null
89
- return firstSegment
90
- }, [files])
91
- const normalizedPaths = useMemo(
92
- () =>
93
- files.map((file) => {
94
- const raw = (file.webkitRelativePath || file.name).replace(/^\.\//, '')
95
- if (stripRoot && raw.startsWith(`${stripRoot}/`)) {
96
- return raw.slice(stripRoot.length + 1)
97
- }
98
- return raw
99
- }),
100
- [files, stripRoot],
101
- )
102
- const hasRequiredFile = useMemo(
103
- () =>
104
- normalizedPaths.some((path) => {
105
- const lower = path.trim().toLowerCase()
106
- return isSoulMode ? lower === 'soul.md' : lower === 'skill.md' || lower === 'skills.md'
107
- }),
108
- [isSoulMode, normalizedPaths],
109
- )
110
- const sizeLabel = totalBytes ? formatBytes(totalBytes) : '0 B'
111
- const trimmedSlug = slug.trim()
112
- const trimmedName = displayName.trim()
113
- const trimmedChangelog = changelog.trim()
114
-
115
- useEffect(() => {
116
- if (!existing?.latestVersion || (!existing?.skill && !existing?.soul)) return
117
- const name = existing.skill?.displayName ?? existing.soul?.displayName
118
- const nextSlug = existing.skill?.slug ?? existing.soul?.slug
119
- if (nextSlug) setSlug(nextSlug)
120
- if (name) setDisplayName(name)
121
- const nextVersion = semver.inc(existing.latestVersion.version, 'patch')
122
- if (nextVersion) setVersion(nextVersion)
123
- }, [existing])
124
-
125
- useEffect(() => {
126
- if (changelogTouchedRef.current) return
127
- if (trimmedChangelog) return
128
- if (!trimmedSlug || !SLUG_PATTERN.test(trimmedSlug)) return
129
- if (!semver.valid(version)) return
130
- if (!hasRequiredFile) return
131
- if (files.length === 0) return
132
-
133
- const requiredIndex = normalizedPaths.findIndex((path) => {
134
- const lower = path.trim().toLowerCase()
135
- return isSoulMode ? lower === 'soul.md' : lower === 'skill.md' || lower === 'skills.md'
136
- })
137
- if (requiredIndex < 0) return
138
-
139
- const requiredFile = files[requiredIndex]
140
- if (!requiredFile) return
141
-
142
- const key = `${trimmedSlug}:${version}:${requiredFile.size}:${requiredFile.lastModified}:${normalizedPaths.length}`
143
- if (changelogKeyRef.current === key) return
144
- changelogKeyRef.current = key
145
-
146
- const requestId = ++changelogRequestRef.current
147
- setChangelogStatus('loading')
148
-
149
- void readText(requiredFile)
150
- .then((text) => {
151
- if (changelogRequestRef.current !== requestId) return null
152
- return generateChangelogPreview({
153
- slug: trimmedSlug,
154
- version,
155
- readmeText: text.slice(0, 20_000),
156
- filePaths: normalizedPaths,
157
- })
158
- })
159
- .then((result) => {
160
- if (!result) return
161
- if (changelogRequestRef.current !== requestId) return
162
- setChangelog(result.changelog)
163
- setChangelogSource('auto')
164
- setChangelogStatus('ready')
165
- })
166
- .catch(() => {
167
- if (changelogRequestRef.current !== requestId) return
168
- setChangelogStatus('error')
169
- })
170
- }, [
171
- files,
172
- generateChangelogPreview,
173
- hasRequiredFile,
174
- isSoulMode,
175
- normalizedPaths,
176
- trimmedChangelog,
177
- trimmedSlug,
178
- version,
179
- ])
180
- const parsedTags = useMemo(
181
- () =>
182
- tags
183
- .split(',')
184
- .map((tag) => tag.trim())
185
- .filter(Boolean),
186
- [tags],
187
- )
188
- const validation = useMemo(() => {
189
- const issues: string[] = []
190
- if (!trimmedSlug) {
191
- issues.push('Slug is required.')
192
- } else if (!SLUG_PATTERN.test(trimmedSlug)) {
193
- issues.push('Slug must be lowercase and use dashes only.')
194
- }
195
- if (!trimmedName) {
196
- issues.push('Display name is required.')
197
- }
198
- if (!semver.valid(version)) {
199
- issues.push('Version must be valid semver (e.g. 1.0.0).')
200
- }
201
- if (parsedTags.length === 0) {
202
- issues.push('At least one tag is required.')
203
- }
204
- if (files.length === 0) {
205
- issues.push('Add at least one file.')
206
- }
207
- if (!hasRequiredFile) {
208
- issues.push(`${requiredFileLabel} is required.`)
209
- }
210
- const invalidFiles = files.filter((file) => !isTextFile(file))
211
- if (invalidFiles.length > 0) {
212
- issues.push(
213
- `Remove non-text files: ${invalidFiles
214
- .slice(0, 3)
215
- .map((file) => file.name)
216
- .join(', ')}`,
217
- )
218
- }
219
- if (totalBytes > maxBytes) {
220
- issues.push('Total file size exceeds 50MB.')
221
- }
222
- return {
223
- issues,
224
- ready: issues.length === 0,
225
- }
226
- }, [
227
- trimmedSlug,
228
- trimmedName,
229
- version,
230
- parsedTags.length,
231
- files,
232
- hasRequiredFile,
233
- totalBytes,
234
- requiredFileLabel,
235
- ])
236
-
237
- useEffect(() => {
238
- if (!fileInputRef.current) return
239
- fileInputRef.current.setAttribute('webkitdirectory', '')
240
- fileInputRef.current.setAttribute('directory', '')
241
- }, [])
242
-
243
- if (!isAuthenticated) {
244
- return (
245
- <main className="section">
246
- <div className="card">Sign in to upload a {contentLabel}.</div>
247
- </main>
248
- )
249
- }
250
-
251
- async function handleSubmit(event: React.FormEvent) {
252
- event.preventDefault()
253
- setHasAttempted(true)
254
- if (!validation.ready) {
255
- if (validationRef.current && 'scrollIntoView' in validationRef.current) {
256
- validationRef.current.scrollIntoView({ behavior: 'smooth', block: 'center' })
257
- }
258
- return
259
- }
260
- setError(null)
261
- if (totalBytes > maxBytes) {
262
- setError('Total size exceeds 50MB per version.')
263
- return
264
- }
265
- if (!hasRequiredFile) {
266
- setError(`${requiredFileLabel} is required.`)
267
- return
268
- }
269
- setStatus('Uploading files…')
270
-
271
- const uploaded = [] as Array<{
272
- path: string
273
- size: number
274
- storageId: string
275
- sha256: string
276
- contentType?: string
277
- }>
278
-
279
- for (const file of files) {
280
- const uploadUrl = await generateUploadUrl()
281
- const rawPath = (file.webkitRelativePath || file.name).replace(/^\.\//, '')
282
- const path =
283
- stripRoot && rawPath.startsWith(`${stripRoot}/`)
284
- ? rawPath.slice(stripRoot.length + 1)
285
- : rawPath
286
- const sha256 = await hashFile(file)
287
- const storageId = await uploadFile(uploadUrl, file)
288
- uploaded.push({
289
- path,
290
- size: file.size,
291
- storageId,
292
- sha256,
293
- contentType: file.type || undefined,
294
- })
295
- }
296
-
297
- setStatus('Publishing…')
298
- try {
299
- const result = await publishVersion({
300
- slug: trimmedSlug,
301
- displayName: trimmedName,
302
- version,
303
- changelog: trimmedChangelog,
304
- tags: parsedTags,
305
- files: uploaded,
306
- })
307
- setStatus(null)
308
- setError(null)
309
- setHasAttempted(false)
310
- setChangelogSource('user')
311
- if (result) {
312
- const ownerParam = me?.handle ?? (me?._id ? String(me._id) : 'unknown')
313
- void navigate({
314
- to: isSoulMode ? '/souls/$slug' : '/$owner/$slug',
315
- params: isSoulMode ? { slug: trimmedSlug } : { owner: ownerParam, slug: trimmedSlug },
316
- })
317
- }
318
- } catch (error) {
319
- setStatus(null)
320
- setError(formatPublishError(error))
321
- }
322
- }
323
-
324
- return (
325
- <main className="section">
326
- <h1 className="section-title">Publish a {contentLabel}</h1>
327
- <p className="section-subtitle">
328
- Drop a folder with {requiredFileLabel} and text files. We will handle the rest.
329
- </p>
330
-
331
- <form onSubmit={handleSubmit} className="upload-grid">
332
- <div className="card">
333
- <label className="form-label" htmlFor="slug">
334
- Slug
335
- </label>
336
- <input
337
- className="form-input"
338
- id="slug"
339
- value={slug}
340
- onChange={(event) => setSlug(event.target.value)}
341
- placeholder={`${contentLabel}-name`}
342
- />
343
-
344
- <label className="form-label" htmlFor="displayName">
345
- Display name
346
- </label>
347
- <input
348
- className="form-input"
349
- id="displayName"
350
- value={displayName}
351
- onChange={(event) => setDisplayName(event.target.value)}
352
- placeholder={`My ${contentLabel}`}
353
- />
354
-
355
- <label className="form-label" htmlFor="version">
356
- Version
357
- </label>
358
- <input
359
- className="form-input"
360
- id="version"
361
- value={version}
362
- onChange={(event) => setVersion(event.target.value)}
363
- placeholder="1.0.0"
364
- />
365
-
366
- <label className="form-label" htmlFor="tags">
367
- Tags
368
- </label>
369
- <input
370
- className="form-input"
371
- id="tags"
372
- value={tags}
373
- onChange={(event) => setTags(event.target.value)}
374
- placeholder="latest, stable"
375
- />
376
- </div>
377
-
378
- <div className="card">
379
- <label
380
- className={`upload-dropzone${isDragging ? ' is-dragging' : ''}`}
381
- onDragOver={(event) => {
382
- event.preventDefault()
383
- setIsDragging(true)
384
- }}
385
- onDragLeave={() => setIsDragging(false)}
386
- onDrop={(event) => {
387
- event.preventDefault()
388
- setIsDragging(false)
389
- const items = event.dataTransfer.items
390
- void (async () => {
391
- const dropped = items?.length
392
- ? await expandDroppedItems(items)
393
- : Array.from(event.dataTransfer.files)
394
- const next = await expandFiles(dropped)
395
- setFiles(next)
396
- })()
397
- }}
398
- >
399
- <input
400
- ref={fileInputRef}
401
- className="upload-input"
402
- id="upload-files"
403
- data-testid="upload-input"
404
- type="file"
405
- multiple
406
- // @ts-expect-error - non-standard attribute to allow folder selection
407
- webkitdirectory=""
408
- directory=""
409
- onChange={(event) => {
410
- const picked = Array.from(event.target.files ?? [])
411
- void expandFiles(picked).then((next) => setFiles(next))
412
- }}
413
- />
414
- <div className="upload-dropzone-copy">
415
- <strong>Drop a folder</strong>
416
- <span>
417
- {files.length} files · {sizeLabel}
418
- </span>
419
- <button className="btn" type="button" onClick={() => fileInputRef.current?.click()}>
420
- Choose folder
421
- </button>
422
- </div>
423
- </label>
424
-
425
- <div className="upload-file-list">
426
- {files.length === 0 ? (
427
- <div className="stat">No files selected.</div>
428
- ) : (
429
- normalizedPaths.map((path) => (
430
- <div key={path} className="upload-file-row">
431
- <span>{path}</span>
432
- </div>
433
- ))
434
- )}
435
- </div>
436
- </div>
437
-
438
- <div className="card" ref={validationRef}>
439
- <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
440
- Validation
441
- </h2>
442
- {validation.issues.length === 0 ? (
443
- <div className="stat">All checks passed.</div>
444
- ) : (
445
- <ul className="validation-list">
446
- {validation.issues.map((issue) => (
447
- <li key={issue}>{issue}</li>
448
- ))}
449
- </ul>
450
- )}
451
- </div>
452
-
453
- <div className="card">
454
- <label className="form-label" htmlFor="changelog">
455
- Changelog
456
- </label>
457
- <textarea
458
- className="form-input"
459
- id="changelog"
460
- rows={6}
461
- value={changelog}
462
- onChange={(event) => {
463
- changelogTouchedRef.current = true
464
- setChangelogSource('user')
465
- setChangelog(event.target.value)
466
- }}
467
- placeholder={`Describe what changed in this ${contentLabel}...`}
468
- />
469
- {changelogStatus === 'loading' ? <div className="stat">Generating changelog…</div> : null}
470
- {changelogStatus === 'error' ? (
471
- <div className="stat">Could not auto-generate changelog.</div>
472
- ) : null}
473
- {changelogSource === 'auto' && changelog ? (
474
- <div className="stat">Auto-generated changelog (edit as needed).</div>
475
- ) : null}
476
- </div>
477
-
478
- <div className="card">
479
- {error ? (
480
- <div className="error" role="alert">
481
- {error}
482
- </div>
483
- ) : null}
484
- {status ? <div className="stat">{status}</div> : null}
485
- <button
486
- className="btn btn-primary"
487
- type="submit"
488
- disabled={!validation.ready || isSubmitting}
489
- >
490
- Publish {contentLabel}
491
- </button>
492
- {hasAttempted && !validation.ready ? (
493
- <div className="stat">Fix validation issues to continue.</div>
494
- ) : null}
495
- </div>
496
- </form>
497
- </main>
498
- )
499
- }