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,203 +0,0 @@
1
- import { createFileRoute } from '@tanstack/react-router'
2
- import { useMutation, useQuery } from 'convex/react'
3
- import { useEffect, useState } from 'react'
4
- import { api } from '../../convex/_generated/api'
5
- import type { Id } from '../../convex/_generated/dataModel'
6
- import { gravatarUrl } from '../lib/gravatar'
7
-
8
- export const Route = createFileRoute('/settings')({
9
- component: Settings,
10
- })
11
-
12
- function Settings() {
13
- const me = useQuery(api.users.me)
14
- const updateProfile = useMutation(api.users.updateProfile)
15
- const deleteAccount = useMutation(api.users.deleteAccount)
16
- const tokens = useQuery(api.tokens.listMine) as
17
- | Array<{
18
- _id: Id<'apiTokens'>
19
- label: string
20
- prefix: string
21
- createdAt: number
22
- lastUsedAt?: number
23
- revokedAt?: number
24
- }>
25
- | undefined
26
- const createToken = useMutation(api.tokens.create)
27
- const revokeToken = useMutation(api.tokens.revoke)
28
- const [displayName, setDisplayName] = useState('')
29
- const [bio, setBio] = useState('')
30
- const [status, setStatus] = useState<string | null>(null)
31
- const [tokenLabel, setTokenLabel] = useState('CLI token')
32
- const [newToken, setNewToken] = useState<string | null>(null)
33
-
34
- useEffect(() => {
35
- if (!me) return
36
- setDisplayName(me.displayName ?? '')
37
- setBio(me.bio ?? '')
38
- }, [me])
39
-
40
- if (!me) {
41
- return (
42
- <main className="section">
43
- <div className="card">Sign in to access settings.</div>
44
- </main>
45
- )
46
- }
47
-
48
- const avatar = me.image ?? (me.email ? gravatarUrl(me.email, 160) : undefined)
49
- const identityName = me.displayName ?? me.name ?? me.handle ?? 'Profile'
50
- const handle = me.handle ?? (me.email ? me.email.split('@')[0] : undefined)
51
-
52
- async function onSave(event: React.FormEvent) {
53
- event.preventDefault()
54
- await updateProfile({ displayName, bio })
55
- setStatus('Saved.')
56
- }
57
-
58
- async function onDelete() {
59
- const ok = window.confirm('Soft delete your account? This cannot be undone.')
60
- if (!ok) return
61
- await deleteAccount()
62
- }
63
-
64
- async function onCreateToken() {
65
- const label = tokenLabel.trim() || 'CLI token'
66
- const result = await createToken({ label })
67
- setNewToken(result.token)
68
- }
69
-
70
- return (
71
- <main className="section settings-shell">
72
- <h1 className="section-title">Settings</h1>
73
- <div className="card settings-profile">
74
- <div className="settings-avatar">
75
- {avatar ? (
76
- <img src={avatar} alt={identityName} />
77
- ) : (
78
- <span>{identityName[0]?.toUpperCase() ?? 'U'}</span>
79
- )}
80
- </div>
81
- <div className="settings-profile-body">
82
- <div className="settings-name">{identityName}</div>
83
- {handle ? <div className="settings-handle">@{handle}</div> : null}
84
- {me.email ? <div className="settings-email">{me.email}</div> : null}
85
- </div>
86
- </div>
87
- <form className="card settings-card" onSubmit={onSave}>
88
- <label className="settings-field">
89
- <span>Display name</span>
90
- <input
91
- className="settings-input"
92
- value={displayName}
93
- onChange={(event) => setDisplayName(event.target.value)}
94
- />
95
- </label>
96
- <label className="settings-field">
97
- <span>Bio</span>
98
- <textarea
99
- className="settings-input"
100
- rows={5}
101
- value={bio}
102
- onChange={(event) => setBio(event.target.value)}
103
- placeholder="Tell people what you're building."
104
- />
105
- </label>
106
- <div className="settings-actions">
107
- <button className="btn btn-primary settings-save" type="submit">
108
- Save
109
- </button>
110
- {status ? <div className="stat">{status}</div> : null}
111
- </div>
112
- </form>
113
-
114
- <div className="card settings-card">
115
- <h2 className="section-title danger-title" style={{ marginTop: 0 }}>
116
- API tokens
117
- </h2>
118
- <p className="section-subtitle">
119
- Use these tokens for the `pilothub` CLI. Tokens are shown once on creation.
120
- </p>
121
-
122
- <div className="settings-field">
123
- <span>Label</span>
124
- <input
125
- className="settings-input"
126
- value={tokenLabel}
127
- onChange={(event) => setTokenLabel(event.target.value)}
128
- placeholder="CLI token"
129
- />
130
- </div>
131
- <div className="settings-actions">
132
- <button
133
- className="btn btn-primary settings-save"
134
- type="button"
135
- onClick={() => void onCreateToken()}
136
- >
137
- Create token
138
- </button>
139
- {newToken ? (
140
- <div className="stat" style={{ overflowX: 'auto' }}>
141
- <div style={{ marginBottom: 8 }}>Copy this token now:</div>
142
- <code>{newToken}</code>
143
- </div>
144
- ) : null}
145
- </div>
146
-
147
- {(tokens ?? []).length ? (
148
- <div style={{ display: 'grid', gap: 10, marginTop: 16 }}>
149
- {(tokens ?? []).map((token) => (
150
- <div
151
- key={token._id}
152
- className="stat"
153
- style={{ display: 'flex', justifyContent: 'space-between', gap: 12 }}
154
- >
155
- <div>
156
- <div>
157
- <strong>{token.label}</strong>{' '}
158
- <span style={{ opacity: 0.7 }}>({token.prefix}…)</span>
159
- </div>
160
- <div style={{ opacity: 0.7 }}>
161
- Created {formatDate(token.createdAt)}
162
- {token.lastUsedAt ? ` · Used ${formatDate(token.lastUsedAt)}` : ''}
163
- {token.revokedAt ? ` · Revoked ${formatDate(token.revokedAt)}` : ''}
164
- </div>
165
- </div>
166
- <div>
167
- <button
168
- className="btn"
169
- type="button"
170
- disabled={Boolean(token.revokedAt)}
171
- onClick={() => void revokeToken({ tokenId: token._id })}
172
- >
173
- {token.revokedAt ? 'Revoked' : 'Revoke'}
174
- </button>
175
- </div>
176
- </div>
177
- ))}
178
- </div>
179
- ) : (
180
- <p className="section-subtitle" style={{ marginTop: 16 }}>
181
- No tokens yet.
182
- </p>
183
- )}
184
- </div>
185
-
186
- <div className="card danger-card">
187
- <h2 className="section-title danger-title">Danger zone</h2>
188
- <p className="section-subtitle">Soft delete your account. Skills remain public.</p>
189
- <button className="btn btn-danger" type="button" onClick={() => void onDelete()}>
190
- Delete account
191
- </button>
192
- </div>
193
- </main>
194
- )
195
- }
196
-
197
- function formatDate(value: number) {
198
- try {
199
- return new Date(value).toLocaleString()
200
- } catch {
201
- return String(value)
202
- }
203
- }
@@ -1,422 +0,0 @@
1
- import { createFileRoute, Link } from '@tanstack/react-router'
2
- import { useAction } from 'convex/react'
3
- import { usePaginatedQuery } from 'convex-helpers/react'
4
- import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
5
- import { api } from '../../../convex/_generated/api'
6
- import type { Doc } from '../../../convex/_generated/dataModel'
7
- import { SkillCard } from '../../components/SkillCard'
8
- import { getSkillBadges, isSkillHighlighted } from '../../lib/badges'
9
- import type { PublicSkill } from '../../lib/publicUser'
10
-
11
- const sortKeys = ['newest', 'downloads', 'installs', 'stars', 'name', 'updated'] as const
12
- const pageSize = 25
13
- type SortKey = (typeof sortKeys)[number]
14
- type SortDir = 'asc' | 'desc'
15
-
16
- function parseSort(value: unknown): SortKey {
17
- if (typeof value !== 'string') return 'newest'
18
- if ((sortKeys as readonly string[]).includes(value)) return value as SortKey
19
- return 'newest'
20
- }
21
-
22
- function parseDir(value: unknown, sort: SortKey): SortDir {
23
- if (value === 'asc' || value === 'desc') return value
24
- return sort === 'name' ? 'asc' : 'desc'
25
- }
26
-
27
- type SkillListEntry = {
28
- skill: PublicSkill
29
- latestVersion: Doc<'skillVersions'> | null
30
- ownerHandle?: string | null
31
- }
32
-
33
- type SkillSearchEntry = {
34
- skill: PublicSkill
35
- version: Doc<'skillVersions'> | null
36
- score: number
37
- ownerHandle?: string | null
38
- }
39
-
40
- function buildSkillHref(skill: PublicSkill, ownerHandle?: string | null) {
41
- const owner = ownerHandle?.trim() || String(skill.ownerUserId)
42
- return `/${encodeURIComponent(owner)}/${encodeURIComponent(skill.slug)}`
43
- }
44
-
45
- export const Route = createFileRoute('/skills/')({
46
- validateSearch: (search) => {
47
- return {
48
- q: typeof search.q === 'string' && search.q.trim() ? search.q : undefined,
49
- sort: typeof search.sort === 'string' ? parseSort(search.sort) : undefined,
50
- dir: search.dir === 'asc' || search.dir === 'desc' ? search.dir : undefined,
51
- highlighted:
52
- search.highlighted === '1' || search.highlighted === 'true' || search.highlighted === true
53
- ? true
54
- : undefined,
55
- view: search.view === 'cards' || search.view === 'list' ? search.view : undefined,
56
- focus: search.focus === 'search' ? 'search' : undefined,
57
- }
58
- },
59
- component: SkillsIndex,
60
- })
61
-
62
- export function SkillsIndex() {
63
- const navigate = Route.useNavigate()
64
- const search = Route.useSearch()
65
- const sort = search.sort ?? 'newest'
66
- const dir = parseDir(search.dir, sort)
67
- const view = search.view ?? 'list'
68
- const highlightedOnly = search.highlighted ?? false
69
- const [query, setQuery] = useState(search.q ?? '')
70
- const searchSkills = useAction(api.search.searchSkills)
71
- const [searchResults, setSearchResults] = useState<Array<SkillSearchEntry>>([])
72
- const [searchLimit, setSearchLimit] = useState(pageSize)
73
- const [isSearching, setIsSearching] = useState(false)
74
- const searchRequest = useRef(0)
75
- const loadMoreRef = useRef<HTMLDivElement | null>(null)
76
-
77
- const searchInputRef = useRef<HTMLInputElement>(null)
78
- const trimmedQuery = useMemo(() => query.trim(), [query])
79
- const hasQuery = trimmedQuery.length > 0
80
- const searchKey = trimmedQuery ? `${trimmedQuery}::${highlightedOnly ? '1' : '0'}` : ''
81
-
82
- // Use convex-helpers usePaginatedQuery for better cache behavior
83
- const {
84
- results: paginatedResults,
85
- status: paginationStatus,
86
- loadMore: loadMorePaginated,
87
- } = usePaginatedQuery(api.skills.listPublicPageV2, hasQuery ? 'skip' : {}, {
88
- initialNumItems: pageSize,
89
- })
90
-
91
- // Derive loading states from pagination status
92
- // status: 'LoadingFirstPage' | 'CanLoadMore' | 'LoadingMore' | 'Exhausted'
93
- const isLoadingList = paginationStatus === 'LoadingFirstPage'
94
- const canLoadMoreList = paginationStatus === 'CanLoadMore'
95
- const isLoadingMoreList = paginationStatus === 'LoadingMore'
96
-
97
- useEffect(() => {
98
- setQuery(search.q ?? '')
99
- }, [search.q])
100
-
101
- // Auto-focus search input when focus=search param is present
102
- useEffect(() => {
103
- if (search.focus === 'search' && searchInputRef.current) {
104
- searchInputRef.current.focus()
105
- // Clear the focus param from URL to avoid re-focusing on navigation
106
- void navigate({ search: (prev) => ({ ...prev, focus: undefined }), replace: true })
107
- }
108
- }, [search.focus, navigate])
109
-
110
- useEffect(() => {
111
- if (!searchKey) {
112
- setSearchResults([])
113
- setIsSearching(false)
114
- return
115
- }
116
- setSearchResults([])
117
- setSearchLimit(pageSize)
118
- }, [searchKey])
119
-
120
- useEffect(() => {
121
- if (!hasQuery) return
122
- searchRequest.current += 1
123
- const requestId = searchRequest.current
124
- setIsSearching(true)
125
- const handle = window.setTimeout(() => {
126
- void (async () => {
127
- try {
128
- const data = (await searchSkills({
129
- query: trimmedQuery,
130
- highlightedOnly,
131
- limit: searchLimit,
132
- })) as Array<SkillSearchEntry>
133
- if (requestId === searchRequest.current) {
134
- setSearchResults(data)
135
- }
136
- } finally {
137
- if (requestId === searchRequest.current) {
138
- setIsSearching(false)
139
- }
140
- }
141
- })()
142
- }, 220)
143
- return () => window.clearTimeout(handle)
144
- }, [hasQuery, highlightedOnly, searchLimit, searchSkills, trimmedQuery])
145
-
146
- const baseItems = useMemo(() => {
147
- if (hasQuery) {
148
- return searchResults.map((entry) => ({
149
- skill: entry.skill,
150
- latestVersion: entry.version,
151
- ownerHandle: entry.ownerHandle ?? null,
152
- }))
153
- }
154
- // paginatedResults is an array of page items from usePaginatedQuery
155
- return paginatedResults as Array<SkillListEntry>
156
- }, [hasQuery, paginatedResults, searchResults])
157
-
158
- const filtered = useMemo(
159
- () => baseItems.filter((entry) => (highlightedOnly ? isSkillHighlighted(entry.skill) : true)),
160
- [baseItems, highlightedOnly],
161
- )
162
-
163
- const sorted = useMemo(() => {
164
- const multiplier = dir === 'asc' ? 1 : -1
165
- const results = [...filtered]
166
- results.sort((a, b) => {
167
- switch (sort) {
168
- case 'downloads':
169
- return (a.skill.stats.downloads - b.skill.stats.downloads) * multiplier
170
- case 'installs':
171
- return (
172
- ((a.skill.stats.installsAllTime ?? 0) - (b.skill.stats.installsAllTime ?? 0)) *
173
- multiplier
174
- )
175
- case 'stars':
176
- return (a.skill.stats.stars - b.skill.stats.stars) * multiplier
177
- case 'updated':
178
- return (a.skill.updatedAt - b.skill.updatedAt) * multiplier
179
- case 'name':
180
- return (
181
- (a.skill.displayName.localeCompare(b.skill.displayName) ||
182
- a.skill.slug.localeCompare(b.skill.slug)) * multiplier
183
- )
184
- default:
185
- return (a.skill.createdAt - b.skill.createdAt) * multiplier
186
- }
187
- })
188
- return results
189
- }, [dir, filtered, sort])
190
-
191
- const isLoadingSkills = hasQuery ? isSearching && searchResults.length === 0 : isLoadingList
192
- const canLoadMore = hasQuery
193
- ? !isSearching && searchResults.length === searchLimit && searchResults.length > 0
194
- : canLoadMoreList
195
- const isLoadingMore = hasQuery ? isSearching && searchResults.length > 0 : isLoadingMoreList
196
- const canAutoLoad = typeof IntersectionObserver !== 'undefined'
197
-
198
- const loadMore = useCallback(() => {
199
- if (isLoadingMore || !canLoadMore) return
200
- if (hasQuery) {
201
- setSearchLimit((value) => value + pageSize)
202
- } else {
203
- loadMorePaginated(pageSize)
204
- }
205
- }, [canLoadMore, hasQuery, isLoadingMore, loadMorePaginated])
206
-
207
- useEffect(() => {
208
- if (!canLoadMore || typeof IntersectionObserver === 'undefined') return
209
- const target = loadMoreRef.current
210
- if (!target) return
211
- const observer = new IntersectionObserver(
212
- (entries) => {
213
- if (entries.some((entry) => entry.isIntersecting)) {
214
- loadMore()
215
- }
216
- },
217
- { rootMargin: '200px' },
218
- )
219
- observer.observe(target)
220
- return () => observer.disconnect()
221
- }, [canLoadMore, loadMore])
222
-
223
- return (
224
- <main className="section">
225
- <header className="skills-header">
226
- <div>
227
- <h1 className="section-title" style={{ marginBottom: 8 }}>
228
- Skills
229
- </h1>
230
- <p className="section-subtitle" style={{ marginBottom: 0 }}>
231
- {isLoadingSkills
232
- ? 'Loading skills…'
233
- : `Browse the skill library${highlightedOnly ? ' (highlighted)' : ''}.`}
234
- </p>
235
- </div>
236
- <div className="skills-toolbar">
237
- <div className="skills-search">
238
- <input
239
- ref={searchInputRef}
240
- className="skills-search-input"
241
- value={query}
242
- onChange={(event) => {
243
- const next = event.target.value
244
- const trimmed = next.trim()
245
- setQuery(next)
246
- void navigate({
247
- search: (prev) => ({ ...prev, q: trimmed ? next : undefined }),
248
- replace: true,
249
- })
250
- }}
251
- placeholder="Filter by name, slug, or summary…"
252
- />
253
- </div>
254
- <div className="skills-toolbar-row">
255
- <button
256
- className={`search-filter-button${highlightedOnly ? ' is-active' : ''}`}
257
- type="button"
258
- aria-pressed={highlightedOnly}
259
- onClick={() => {
260
- void navigate({
261
- search: (prev) => ({
262
- ...prev,
263
- highlighted: highlightedOnly ? undefined : true,
264
- }),
265
- replace: true,
266
- })
267
- }}
268
- >
269
- Highlighted
270
- </button>
271
- <select
272
- className="skills-sort"
273
- value={sort}
274
- onChange={(event) => {
275
- const sort = parseSort(event.target.value)
276
- void navigate({
277
- search: (prev) => ({
278
- ...prev,
279
- sort,
280
- dir: parseDir(prev.dir, sort),
281
- }),
282
- replace: true,
283
- })
284
- }}
285
- aria-label="Sort skills"
286
- >
287
- <option value="newest">Newest</option>
288
- <option value="updated">Recently updated</option>
289
- <option value="downloads">Downloads</option>
290
- <option value="installs">Installs</option>
291
- <option value="stars">Stars</option>
292
- <option value="name">Name</option>
293
- </select>
294
- <button
295
- className="skills-dir"
296
- type="button"
297
- aria-label={`Sort direction ${dir}`}
298
- onClick={() => {
299
- void navigate({
300
- search: (prev) => ({
301
- ...prev,
302
- dir: parseDir(prev.dir, sort) === 'asc' ? 'desc' : 'asc',
303
- }),
304
- replace: true,
305
- })
306
- }}
307
- >
308
- {dir === 'asc' ? '↑' : '↓'}
309
- </button>
310
- <button
311
- className={`skills-view${view === 'cards' ? ' is-active' : ''}`}
312
- type="button"
313
- onClick={() => {
314
- void navigate({
315
- search: (prev) => ({
316
- ...prev,
317
- view: prev.view === 'cards' ? undefined : 'cards',
318
- }),
319
- replace: true,
320
- })
321
- }}
322
- >
323
- {view === 'cards' ? 'List' : 'Cards'}
324
- </button>
325
- </div>
326
- </div>
327
- </header>
328
-
329
- {isLoadingSkills ? (
330
- <div className="card">
331
- <div className="loading-indicator">Loading skills…</div>
332
- </div>
333
- ) : sorted.length === 0 ? (
334
- <div className="card">No skills match that filter.</div>
335
- ) : view === 'cards' ? (
336
- <div className="grid">
337
- {sorted.map((entry) => {
338
- const skill = entry.skill
339
- const isPlugin = Boolean(entry.latestVersion?.parsed?.pilotbot?.nix?.plugin)
340
- const skillHref = buildSkillHref(skill, entry.ownerHandle)
341
- return (
342
- <SkillCard
343
- key={skill._id}
344
- skill={skill}
345
- href={skillHref}
346
- badge={getSkillBadges(skill)}
347
- chip={isPlugin ? 'Plugin bundle (nix)' : undefined}
348
- summaryFallback="Agent-ready skill pack."
349
- meta={
350
- <div className="stat">
351
- ⭐ {skill.stats.stars} · ⤓ {skill.stats.downloads} · ⤒{' '}
352
- {skill.stats.installsAllTime ?? 0}
353
- </div>
354
- }
355
- />
356
- )
357
- })}
358
- </div>
359
- ) : (
360
- <div className="skills-list">
361
- {sorted.map((entry) => {
362
- const skill = entry.skill
363
- const isPlugin = Boolean(entry.latestVersion?.parsed?.pilotbot?.nix?.plugin)
364
- const skillHref = buildSkillHref(skill, entry.ownerHandle)
365
- return (
366
- <Link key={skill._id} className="skills-row" to={skillHref}>
367
- <div className="skills-row-main">
368
- <div className="skills-row-title">
369
- <span>{skill.displayName}</span>
370
- <span className="skills-row-slug">/{skill.slug}</span>
371
- {getSkillBadges(skill).map((badge) => (
372
- <span key={badge} className="tag">
373
- {badge}
374
- </span>
375
- ))}
376
- {isPlugin ? (
377
- <span className="tag tag-accent tag-compact">Plugin bundle (nix)</span>
378
- ) : null}
379
- </div>
380
- <div className="skills-row-summary">
381
- {skill.summary ?? 'No summary provided.'}
382
- </div>
383
- {isPlugin ? (
384
- <div className="skills-row-meta">
385
- Bundle includes SKILL.md, CLI, and config.
386
- </div>
387
- ) : null}
388
- </div>
389
- <div className="skills-row-metrics">
390
- <span>⤓ {skill.stats.downloads}</span>
391
- <span>⤒ {skill.stats.installsAllTime ?? 0}</span>
392
- <span>★ {skill.stats.stars}</span>
393
- <span>{skill.stats.versions} v</span>
394
- </div>
395
- </Link>
396
- )
397
- })}
398
- </div>
399
- )}
400
-
401
- {canLoadMore ? (
402
- <div
403
- ref={canAutoLoad ? loadMoreRef : null}
404
- className="card"
405
- style={{ marginTop: 16, display: 'flex', justifyContent: 'center' }}
406
- >
407
- {canAutoLoad ? (
408
- isLoadingMore ? (
409
- 'Loading more…'
410
- ) : (
411
- 'Scroll to load more'
412
- )
413
- ) : (
414
- <button className="btn" type="button" onClick={loadMore} disabled={isLoadingMore}>
415
- {isLoadingMore ? 'Loading…' : 'Load more'}
416
- </button>
417
- )}
418
- </div>
419
- ) : null}
420
- </main>
421
- )
422
- }
@@ -1,55 +0,0 @@
1
- import { createFileRoute } from '@tanstack/react-router'
2
- import { SoulDetailPage } from '../../components/SoulDetailPage'
3
- import { buildSoulMeta, fetchSoulMeta } from '../../lib/og'
4
-
5
- export const Route = createFileRoute('/souls/$slug')({
6
- loader: async ({ params }) => {
7
- const data = await fetchSoulMeta(params.slug)
8
- return {
9
- owner: data?.owner ?? null,
10
- displayName: data?.displayName ?? null,
11
- summary: data?.summary ?? null,
12
- version: data?.version ?? null,
13
- }
14
- },
15
- head: ({ params, loaderData }) => {
16
- const meta = buildSoulMeta({
17
- slug: params.slug,
18
- owner: loaderData?.owner ?? null,
19
- displayName: loaderData?.displayName,
20
- summary: loaderData?.summary,
21
- version: loaderData?.version ?? null,
22
- })
23
- return {
24
- links: [
25
- {
26
- rel: 'canonical',
27
- href: meta.url,
28
- },
29
- ],
30
- meta: [
31
- { title: meta.title },
32
- { name: 'description', content: meta.description },
33
- { property: 'og:title', content: meta.title },
34
- { property: 'og:description', content: meta.description },
35
- { property: 'og:type', content: 'website' },
36
- { property: 'og:url', content: meta.url },
37
- { property: 'og:image', content: meta.image },
38
- { property: 'og:image:width', content: '1200' },
39
- { property: 'og:image:height', content: '630' },
40
- { property: 'og:image:alt', content: meta.title },
41
- { name: 'twitter:card', content: 'summary_large_image' },
42
- { name: 'twitter:title', content: meta.title },
43
- { name: 'twitter:description', content: meta.description },
44
- { name: 'twitter:image', content: meta.image },
45
- { name: 'twitter:image:alt', content: meta.title },
46
- ],
47
- }
48
- },
49
- component: SoulDetail,
50
- })
51
-
52
- function SoulDetail() {
53
- const { slug } = Route.useParams()
54
- return <SoulDetailPage slug={slug} />
55
- }