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,228 +0,0 @@
1
- import { fireEvent, render, screen, waitFor } from '@testing-library/react'
2
- import { strToU8, zipSync } from 'fflate'
3
- import { vi } from 'vitest'
4
-
5
- import { Upload } from '../routes/upload'
6
-
7
- vi.mock('@tanstack/react-router', () => ({
8
- createFileRoute: () => (config: { component: unknown }) => config,
9
- useNavigate: () => vi.fn(),
10
- useSearch: () => ({ updateSlug: undefined }),
11
- }))
12
-
13
- const generateUploadUrl = vi.fn()
14
- const publishVersion = vi.fn()
15
- const generateChangelogPreview = vi.fn()
16
- const fetchMock = vi.fn()
17
- const useQueryMock = vi.fn()
18
- const useAuthStatusMock = vi.fn()
19
- let useActionCallCount = 0
20
-
21
- vi.mock('convex/react', () => ({
22
- useQuery: (...args: unknown[]) => useQueryMock(...args),
23
- useMutation: () => generateUploadUrl,
24
- useAction: () => {
25
- useActionCallCount += 1
26
- return useActionCallCount % 2 === 1 ? publishVersion : generateChangelogPreview
27
- },
28
- }))
29
-
30
- vi.mock('../lib/useAuthStatus', () => ({
31
- useAuthStatus: () => useAuthStatusMock(),
32
- }))
33
-
34
- describe('Upload route', () => {
35
- beforeEach(() => {
36
- generateUploadUrl.mockReset()
37
- publishVersion.mockReset()
38
- generateChangelogPreview.mockReset()
39
- fetchMock.mockReset()
40
- useQueryMock.mockReset()
41
- useAuthStatusMock.mockReset()
42
- useActionCallCount = 0
43
- useAuthStatusMock.mockReturnValue({
44
- isAuthenticated: true,
45
- isLoading: false,
46
- me: { _id: 'users:1' },
47
- })
48
- useQueryMock.mockImplementation((_fn: unknown, args: unknown) => {
49
- if (args === 'skip') return undefined
50
- return null
51
- })
52
- fetchMock.mockResolvedValue({
53
- ok: true,
54
- json: async () => ({ storageId: 'storage-id' }),
55
- })
56
- vi.stubGlobal('fetch', fetchMock)
57
- })
58
-
59
- afterEach(() => {
60
- vi.unstubAllGlobals()
61
- })
62
-
63
- it('shows validation issues before submit', async () => {
64
- render(<Upload />)
65
- const publishButton = screen.getByRole('button', { name: /publish/i })
66
- expect(publishButton.getAttribute('disabled')).not.toBeNull()
67
- expect(screen.getByText(/Slug is required/i)).toBeTruthy()
68
- expect(screen.getByText(/Display name is required/i)).toBeTruthy()
69
- })
70
-
71
- it('marks the input for folder uploads', async () => {
72
- render(<Upload />)
73
- const input = screen.getByTestId('upload-input')
74
- await waitFor(() => {
75
- expect(input.getAttribute('webkitdirectory')).not.toBeNull()
76
- })
77
- })
78
-
79
- it('enables publish when fields and files are valid', async () => {
80
- generateUploadUrl.mockResolvedValue('https://upload.local')
81
- render(<Upload />)
82
- fireEvent.change(screen.getByPlaceholderText('skill-name'), {
83
- target: { value: 'cool-skill' },
84
- })
85
- fireEvent.change(screen.getByPlaceholderText('My skill'), {
86
- target: { value: 'Cool Skill' },
87
- })
88
- fireEvent.change(screen.getByPlaceholderText('1.0.0'), {
89
- target: { value: '1.2.3' },
90
- })
91
- fireEvent.change(screen.getByPlaceholderText('latest, stable'), {
92
- target: { value: 'latest' },
93
- })
94
- const file = new File(['hello'], 'SKILL.md', { type: 'text/markdown' })
95
- const input = screen.getByTestId('upload-input') as HTMLInputElement
96
- fireEvent.change(input, { target: { files: [file] } })
97
-
98
- const publishButton = screen.getByRole('button', { name: /publish/i }) as HTMLButtonElement
99
- expect(await screen.findByText(/All checks passed/i)).toBeTruthy()
100
- expect(publishButton.getAttribute('disabled')).toBeNull()
101
- })
102
-
103
- it('extracts zip uploads and unwraps top-level folders', async () => {
104
- render(<Upload />)
105
- fireEvent.change(screen.getByPlaceholderText('skill-name'), {
106
- target: { value: 'cool-skill' },
107
- })
108
- fireEvent.change(screen.getByPlaceholderText('My skill'), {
109
- target: { value: 'Cool Skill' },
110
- })
111
- fireEvent.change(screen.getByPlaceholderText('1.0.0'), {
112
- target: { value: '1.2.3' },
113
- })
114
- fireEvent.change(screen.getByPlaceholderText('latest, stable'), {
115
- target: { value: 'latest' },
116
- })
117
-
118
- const zip = zipSync({
119
- 'hetzner-cloud-skill/SKILL.md': new Uint8Array(strToU8('hello')),
120
- 'hetzner-cloud-skill/notes.txt': new Uint8Array(strToU8('notes')),
121
- })
122
- const zipBytes = Uint8Array.from(zip).buffer
123
- const zipFile = new File([zipBytes], 'bundle.zip', { type: 'application/zip' })
124
-
125
- const input = screen.getByTestId('upload-input') as HTMLInputElement
126
- fireEvent.change(input, { target: { files: [zipFile] } })
127
-
128
- expect(await screen.findByText('notes.txt', {}, { timeout: 3000 })).toBeTruthy()
129
- expect(screen.getByText('SKILL.md')).toBeTruthy()
130
- expect(await screen.findByText(/All checks passed/i, {}, { timeout: 3000 })).toBeTruthy()
131
- })
132
-
133
- it('unwraps folder uploads so SKILL.md can be at the top-level', async () => {
134
- generateUploadUrl.mockResolvedValue('https://upload.local')
135
- publishVersion.mockResolvedValue(undefined)
136
- render(<Upload />)
137
- fireEvent.change(screen.getByPlaceholderText('skill-name'), {
138
- target: { value: 'ynab' },
139
- })
140
- fireEvent.change(screen.getByPlaceholderText('My skill'), {
141
- target: { value: 'YNAB' },
142
- })
143
- fireEvent.change(screen.getByPlaceholderText('1.0.0'), {
144
- target: { value: '1.0.0' },
145
- })
146
- fireEvent.change(screen.getByPlaceholderText('latest, stable'), {
147
- target: { value: 'latest' },
148
- })
149
-
150
- const file = new File(['hello'], 'SKILL.md', { type: 'text/markdown' })
151
- Object.defineProperty(file, 'webkitRelativePath', { value: 'ynab/SKILL.md' })
152
-
153
- const input = screen.getByTestId('upload-input') as HTMLInputElement
154
- fireEvent.change(input, { target: { files: [file] } })
155
-
156
- expect(await screen.findByText('SKILL.md')).toBeTruthy()
157
- expect(await screen.findByText(/All checks passed/i)).toBeTruthy()
158
-
159
- fireEvent.click(screen.getByRole('button', { name: /publish/i }))
160
- await waitFor(() => {
161
- expect(
162
- publishVersion.mock.calls.some((call) =>
163
- Array.isArray((call[0] as { files?: unknown }).files),
164
- ),
165
- ).toBe(true)
166
- })
167
- const args = publishVersion.mock.calls
168
- .map((call) => call[0] as { files?: Array<{ path: string }> })
169
- .find((call) => Array.isArray(call.files))
170
- expect(args?.files?.[0]?.path).toBe('SKILL.md')
171
- })
172
-
173
- it('blocks non-text folder uploads (png)', async () => {
174
- render(<Upload />)
175
- fireEvent.change(screen.getByPlaceholderText('skill-name'), {
176
- target: { value: 'cool-skill' },
177
- })
178
- fireEvent.change(screen.getByPlaceholderText('My skill'), {
179
- target: { value: 'Cool Skill' },
180
- })
181
- fireEvent.change(screen.getByPlaceholderText('1.0.0'), {
182
- target: { value: '1.2.3' },
183
- })
184
- fireEvent.change(screen.getByPlaceholderText('latest, stable'), {
185
- target: { value: 'latest' },
186
- })
187
-
188
- const skill = new File(['hello'], 'SKILL.md', { type: 'text/markdown' })
189
- const png = new File([new Uint8Array([137, 80, 78, 71]).buffer], 'screenshot.png', {
190
- type: 'image/png',
191
- })
192
- const input = screen.getByTestId('upload-input') as HTMLInputElement
193
- fireEvent.change(input, { target: { files: [skill, png] } })
194
-
195
- expect(await screen.findByText('screenshot.png')).toBeTruthy()
196
- fireEvent.click(screen.getByRole('button', { name: /publish/i }))
197
- expect(await screen.findByText(/Remove non-text files: screenshot\.png/i)).toBeTruthy()
198
- expect(screen.getByText('screenshot.png')).toBeTruthy()
199
- })
200
-
201
- it('surfaces publish errors and stays on page', async () => {
202
- publishVersion.mockRejectedValueOnce(new Error('Changelog is required'))
203
- generateUploadUrl.mockResolvedValue('https://upload.local')
204
- render(<Upload />)
205
- fireEvent.change(screen.getByPlaceholderText('skill-name'), {
206
- target: { value: 'cool-skill' },
207
- })
208
- fireEvent.change(screen.getByPlaceholderText('My skill'), {
209
- target: { value: 'Cool Skill' },
210
- })
211
- fireEvent.change(screen.getByPlaceholderText('1.0.0'), {
212
- target: { value: '1.2.3' },
213
- })
214
- fireEvent.change(screen.getByPlaceholderText('latest, stable'), {
215
- target: { value: 'latest' },
216
- })
217
- fireEvent.change(screen.getByPlaceholderText('Describe what changed in this skill...'), {
218
- target: { value: 'Initial drop.' },
219
- })
220
- const file = new File(['hello'], 'SKILL.md', { type: 'text/markdown' })
221
- const input = screen.getByTestId('upload-input') as HTMLInputElement
222
- fireEvent.change(input, { target: { files: [file] } })
223
- const publishButton = screen.getByRole('button', { name: /publish/i }) as HTMLButtonElement
224
- await screen.findByText(/All checks passed/i)
225
- fireEvent.click(publishButton)
226
- expect(await screen.findByText(/Changelog is required/i)).toBeTruthy()
227
- })
228
- })
@@ -1,19 +0,0 @@
1
- import { ConvexAuthProvider } from '@convex-dev/auth/react'
2
- import { convex } from '../convex/client'
3
- import { UserBootstrap } from './UserBootstrap'
4
-
5
- export function AppProviders({ children }: { children: React.ReactNode }) {
6
- return (
7
- <ConvexAuthProvider
8
- client={convex}
9
- replaceURL={(relativeUrl) => {
10
- if (typeof window !== 'undefined') {
11
- window.history.replaceState(null, '', relativeUrl)
12
- }
13
- }}
14
- >
15
- <UserBootstrap />
16
- {children}
17
- </ConvexAuthProvider>
18
- )
19
- }
@@ -1,18 +0,0 @@
1
- import { useEffect, useState } from 'react'
2
-
3
- export function ClientOnly({
4
- children,
5
- fallback = null,
6
- }: {
7
- children: React.ReactNode
8
- fallback?: React.ReactNode
9
- }) {
10
- const [ready, setReady] = useState(false)
11
-
12
- useEffect(() => {
13
- setReady(true)
14
- }, [])
15
-
16
- if (!ready) return <>{fallback}</>
17
- return <>{children}</>
18
- }
@@ -1,29 +0,0 @@
1
- import { getSiteName } from '../lib/site'
2
-
3
- export function Footer() {
4
- const siteName = getSiteName()
5
- return (
6
- <footer className="site-footer">
7
- <div className="site-footer-inner">
8
- <div className="site-footer-divider" aria-hidden="true" />
9
- <div className="site-footer-row">
10
- <div className="site-footer-copy">
11
- {siteName} · A{' '}
12
- <a href="https://pilot.bot" target="_blank" rel="noreferrer">
13
- Pilotbot
14
- </a>{' '}
15
- project ·{' '}
16
- <a href="https://github.com/pilotbot/pilothub" target="_blank" rel="noreferrer">
17
- Open source (MIT)
18
- </a>{' '}
19
- ·{' '}
20
- <a href="https://steipete.me" target="_blank" rel="noreferrer">
21
- Peter Steinberger
22
- </a>
23
- .
24
- </div>
25
- </div>
26
- </div>
27
- </footer>
28
- )
29
- }
@@ -1,295 +0,0 @@
1
- import { useAuthActions } from '@convex-dev/auth/react'
2
- import { Link } from '@tanstack/react-router'
3
- import { Menu, Monitor, Moon, Sun } from 'lucide-react'
4
- import { useMemo, useRef } from 'react'
5
- import { gravatarUrl } from '../lib/gravatar'
6
- import { isModerator } from '../lib/roles'
7
- import { getPilotHubSiteUrl, getSiteMode, getSiteName } from '../lib/site'
8
- import { applyTheme, useThemeMode } from '../lib/theme'
9
- import { startThemeTransition } from '../lib/theme-transition'
10
- import { useAuthStatus } from '../lib/useAuthStatus'
11
- import {
12
- DropdownMenu,
13
- DropdownMenuContent,
14
- DropdownMenuItem,
15
- DropdownMenuSeparator,
16
- DropdownMenuTrigger,
17
- } from './ui/dropdown-menu'
18
- import { ToggleGroup, ToggleGroupItem } from './ui/toggle-group'
19
-
20
- export default function Header() {
21
- const { isAuthenticated, isLoading, me } = useAuthStatus()
22
- const { signIn, signOut } = useAuthActions()
23
- const { mode, setMode } = useThemeMode()
24
- const toggleRef = useRef<HTMLDivElement | null>(null)
25
- const siteMode = getSiteMode()
26
- const siteName = useMemo(() => getSiteName(siteMode), [siteMode])
27
- const isSoulMode = siteMode === 'souls'
28
- const pilotHubUrl = getPilotHubSiteUrl()
29
-
30
- const avatar = me?.image ?? (me?.email ? gravatarUrl(me.email) : undefined)
31
- const handle = me?.handle ?? me?.displayName ?? 'user'
32
- const initial = (me?.displayName ?? me?.name ?? handle).charAt(0).toUpperCase()
33
- const isStaff = isModerator(me)
34
-
35
- const setTheme = (next: 'system' | 'light' | 'dark') => {
36
- startThemeTransition({
37
- nextTheme: next,
38
- currentTheme: mode,
39
- setTheme: (value) => {
40
- const nextMode = value as 'system' | 'light' | 'dark'
41
- applyTheme(nextMode)
42
- setMode(nextMode)
43
- },
44
- context: { element: toggleRef.current },
45
- })
46
- }
47
-
48
- return (
49
- <header className="navbar">
50
- <div className="navbar-inner">
51
- <Link
52
- to="/"
53
- search={{ q: undefined, highlighted: undefined, search: undefined }}
54
- className="brand"
55
- >
56
- <span className="brand-mark">
57
- <img src="/pilot-logo.png" alt="" aria-hidden="true" />
58
- </span>
59
- <span className="brand-name">{siteName}</span>
60
- </Link>
61
- <nav className="nav-links">
62
- {isSoulMode ? <a href={pilotHubUrl}>PilotHub</a> : null}
63
- {isSoulMode ? (
64
- <Link
65
- to="/souls"
66
- search={{
67
- q: undefined,
68
- sort: undefined,
69
- dir: undefined,
70
- view: undefined,
71
- focus: undefined,
72
- }}
73
- >
74
- Souls
75
- </Link>
76
- ) : (
77
- <Link
78
- to="/skills"
79
- search={{
80
- q: undefined,
81
- sort: undefined,
82
- dir: undefined,
83
- highlighted: undefined,
84
- view: undefined,
85
- focus: undefined,
86
- }}
87
- >
88
- Skills
89
- </Link>
90
- )}
91
- <Link to="/upload" search={{ updateSlug: undefined }}>
92
- Upload
93
- </Link>
94
- {isSoulMode ? null : <Link to="/import">Import</Link>}
95
- <Link
96
- to={isSoulMode ? '/souls' : '/skills'}
97
- search={
98
- isSoulMode
99
- ? {
100
- q: undefined,
101
- sort: undefined,
102
- dir: undefined,
103
- view: undefined,
104
- focus: 'search',
105
- }
106
- : {
107
- q: undefined,
108
- sort: undefined,
109
- dir: undefined,
110
- highlighted: undefined,
111
- view: undefined,
112
- focus: 'search',
113
- }
114
- }
115
- >
116
- Search
117
- </Link>
118
- {me ? <Link to="/stars">Stars</Link> : null}
119
- {isStaff ? (
120
- <Link to="/management" search={{ skill: undefined }}>
121
- Management
122
- </Link>
123
- ) : null}
124
- </nav>
125
- <div className="nav-actions">
126
- <div className="nav-mobile">
127
- <DropdownMenu>
128
- <DropdownMenuTrigger asChild>
129
- <button className="nav-mobile-trigger" type="button" aria-label="Open menu">
130
- <Menu className="h-4 w-4" aria-hidden="true" />
131
- </button>
132
- </DropdownMenuTrigger>
133
- <DropdownMenuContent align="end">
134
- {isSoulMode ? (
135
- <DropdownMenuItem asChild>
136
- <a href={pilotHubUrl}>PilotHub</a>
137
- </DropdownMenuItem>
138
- ) : null}
139
- <DropdownMenuItem asChild>
140
- {isSoulMode ? (
141
- <Link
142
- to="/souls"
143
- search={{
144
- q: undefined,
145
- sort: undefined,
146
- dir: undefined,
147
- view: undefined,
148
- focus: undefined,
149
- }}
150
- >
151
- Souls
152
- </Link>
153
- ) : (
154
- <Link
155
- to="/skills"
156
- search={{
157
- q: undefined,
158
- sort: undefined,
159
- dir: undefined,
160
- highlighted: undefined,
161
- view: undefined,
162
- focus: undefined,
163
- }}
164
- >
165
- Skills
166
- </Link>
167
- )}
168
- </DropdownMenuItem>
169
- <DropdownMenuItem asChild>
170
- <Link to="/upload" search={{ updateSlug: undefined }}>
171
- Upload
172
- </Link>
173
- </DropdownMenuItem>
174
- {isSoulMode ? null : (
175
- <DropdownMenuItem asChild>
176
- <Link to="/import">Import</Link>
177
- </DropdownMenuItem>
178
- )}
179
- <DropdownMenuItem asChild>
180
- <Link
181
- to={isSoulMode ? '/souls' : '/skills'}
182
- search={
183
- isSoulMode
184
- ? {
185
- q: undefined,
186
- sort: undefined,
187
- dir: undefined,
188
- view: undefined,
189
- focus: 'search',
190
- }
191
- : {
192
- q: undefined,
193
- sort: undefined,
194
- dir: undefined,
195
- highlighted: undefined,
196
- view: undefined,
197
- focus: 'search',
198
- }
199
- }
200
- >
201
- Search
202
- </Link>
203
- </DropdownMenuItem>
204
- {me ? (
205
- <DropdownMenuItem asChild>
206
- <Link to="/stars">Stars</Link>
207
- </DropdownMenuItem>
208
- ) : null}
209
- {isStaff ? (
210
- <DropdownMenuItem asChild>
211
- <Link to="/management" search={{ skill: undefined }}>
212
- Management
213
- </Link>
214
- </DropdownMenuItem>
215
- ) : null}
216
- <DropdownMenuSeparator />
217
- <DropdownMenuItem onClick={() => setTheme('system')}>
218
- <Monitor className="h-4 w-4" aria-hidden="true" />
219
- System
220
- </DropdownMenuItem>
221
- <DropdownMenuItem onClick={() => setTheme('light')}>
222
- <Sun className="h-4 w-4" aria-hidden="true" />
223
- Light
224
- </DropdownMenuItem>
225
- <DropdownMenuItem onClick={() => setTheme('dark')}>
226
- <Moon className="h-4 w-4" aria-hidden="true" />
227
- Dark
228
- </DropdownMenuItem>
229
- </DropdownMenuContent>
230
- </DropdownMenu>
231
- </div>
232
- <div className="theme-toggle" ref={toggleRef}>
233
- <ToggleGroup
234
- type="single"
235
- value={mode}
236
- onValueChange={(value) => {
237
- if (!value) return
238
- setTheme(value as 'system' | 'light' | 'dark')
239
- }}
240
- aria-label="Theme mode"
241
- >
242
- <ToggleGroupItem value="system" aria-label="System theme">
243
- <Monitor className="h-4 w-4" aria-hidden="true" />
244
- <span className="sr-only">System</span>
245
- </ToggleGroupItem>
246
- <ToggleGroupItem value="light" aria-label="Light theme">
247
- <Sun className="h-4 w-4" aria-hidden="true" />
248
- <span className="sr-only">Light</span>
249
- </ToggleGroupItem>
250
- <ToggleGroupItem value="dark" aria-label="Dark theme">
251
- <Moon className="h-4 w-4" aria-hidden="true" />
252
- <span className="sr-only">Dark</span>
253
- </ToggleGroupItem>
254
- </ToggleGroup>
255
- </div>
256
- {isAuthenticated && me ? (
257
- <DropdownMenu>
258
- <DropdownMenuTrigger asChild>
259
- <button className="user-trigger" type="button">
260
- {avatar ? (
261
- <img src={avatar} alt={me.displayName ?? me.name ?? 'User avatar'} />
262
- ) : (
263
- <span className="user-menu-fallback">{initial}</span>
264
- )}
265
- <span className="mono">@{handle}</span>
266
- <span className="user-menu-chevron">▾</span>
267
- </button>
268
- </DropdownMenuTrigger>
269
- <DropdownMenuContent align="end">
270
- <DropdownMenuItem asChild>
271
- <Link to="/dashboard">Dashboard</Link>
272
- </DropdownMenuItem>
273
- <DropdownMenuItem asChild>
274
- <Link to="/settings">Settings</Link>
275
- </DropdownMenuItem>
276
- <DropdownMenuSeparator />
277
- <DropdownMenuItem onClick={() => void signOut()}>Sign out</DropdownMenuItem>
278
- </DropdownMenuContent>
279
- </DropdownMenu>
280
- ) : (
281
- <button
282
- className="btn btn-primary"
283
- type="button"
284
- disabled={isLoading}
285
- onClick={() => void signIn('github')}
286
- >
287
- <span className="sign-in-label">Sign in</span>
288
- <span className="sign-in-provider">with GitHub</span>
289
- </button>
290
- )}
291
- </div>
292
- </div>
293
- </header>
294
- )
295
- }
@@ -1,53 +0,0 @@
1
- import { useMemo, useState } from 'react'
2
-
3
- type PackageManager = 'npm' | 'pnpm' | 'bun'
4
-
5
- type InstallSwitcherProps = {
6
- exampleSlug?: string
7
- }
8
-
9
- const PACKAGE_MANAGERS: Array<{ id: PackageManager; label: string }> = [
10
- { id: 'npm', label: 'npm' },
11
- { id: 'pnpm', label: 'pnpm' },
12
- { id: 'bun', label: 'bun' },
13
- ]
14
-
15
- export function InstallSwitcher({ exampleSlug = 'sonoscli' }: InstallSwitcherProps) {
16
- const [pm, setPm] = useState<PackageManager>('npm')
17
-
18
- const command = useMemo(() => {
19
- switch (pm) {
20
- case 'npm':
21
- return `npx pilothub@latest install ${exampleSlug}`
22
- case 'pnpm':
23
- return `pnpm dlx pilothub@latest install ${exampleSlug}`
24
- case 'bun':
25
- return `bunx pilothub@latest install ${exampleSlug}`
26
- }
27
- }, [exampleSlug, pm])
28
-
29
- return (
30
- <div className="install-switcher">
31
- <div className="install-switcher-row">
32
- <div className="stat">Install any skill folder in one shot:</div>
33
- <div className="install-switcher-toggle" role="tablist" aria-label="Install command">
34
- {PACKAGE_MANAGERS.map((entry) => (
35
- <button
36
- key={entry.id}
37
- type="button"
38
- className={
39
- pm === entry.id ? 'install-switcher-pill is-active' : 'install-switcher-pill'
40
- }
41
- role="tab"
42
- aria-selected={pm === entry.id}
43
- onClick={() => setPm(entry.id)}
44
- >
45
- {entry.label}
46
- </button>
47
- ))}
48
- </div>
49
- </div>
50
- <div className="hero-install-code mono">{command}</div>
51
- </div>
52
- )
53
- }
@@ -1,36 +0,0 @@
1
- import { Link } from '@tanstack/react-router'
2
- import type { ReactNode } from 'react'
3
- import type { PublicSkill } from '../lib/publicUser'
4
-
5
- type SkillCardProps = {
6
- skill: PublicSkill
7
- badge?: string | string[]
8
- chip?: string
9
- summaryFallback: string
10
- meta: ReactNode
11
- href?: string
12
- }
13
-
14
- export function SkillCard({ skill, badge, chip, summaryFallback, meta, href }: SkillCardProps) {
15
- const owner = encodeURIComponent(String(skill.ownerUserId))
16
- const link = href ?? `/${owner}/${skill.slug}`
17
- const badges = Array.isArray(badge) ? badge : badge ? [badge] : []
18
-
19
- return (
20
- <Link to={link} className="card skill-card">
21
- {badges.length || chip ? (
22
- <div className="skill-card-tags">
23
- {badges.map((label) => (
24
- <div key={label} className="tag">
25
- {label}
26
- </div>
27
- ))}
28
- {chip ? <div className="tag tag-accent tag-compact">{chip}</div> : null}
29
- </div>
30
- ) : null}
31
- <h3 className="skill-card-title">{skill.displayName}</h3>
32
- <p className="skill-card-summary">{skill.summary ?? summaryFallback}</p>
33
- <div className="skill-card-footer">{meta}</div>
34
- </Link>
35
- )
36
- }