pilothub 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/.env.local.example +19 -0
  2. package/.github/workflows/ci.yml +40 -0
  3. package/.oxlintrc.json +3 -0
  4. package/AGENTS.md +45 -0
  5. package/CHANGELOG.md +138 -0
  6. package/DEPRECATIONS.md +7 -0
  7. package/LICENSE +21 -0
  8. package/README.md +150 -0
  9. package/biome.json +41 -0
  10. package/convex/_generated/api.d.ts +153 -0
  11. package/convex/_generated/api.js +23 -0
  12. package/convex/_generated/dataModel.d.ts +60 -0
  13. package/convex/_generated/server.d.ts +143 -0
  14. package/convex/_generated/server.js +93 -0
  15. package/convex/auth.config.ts +8 -0
  16. package/convex/auth.ts +19 -0
  17. package/convex/comments.ts +88 -0
  18. package/convex/crons.ts +34 -0
  19. package/convex/devSeed.ts +459 -0
  20. package/convex/devSeedExtra.ts +541 -0
  21. package/convex/downloads.ts +78 -0
  22. package/convex/githubBackups.ts +170 -0
  23. package/convex/githubBackupsNode.ts +183 -0
  24. package/convex/githubImport.ts +317 -0
  25. package/convex/githubSoulBackups.ts +170 -0
  26. package/convex/githubSoulBackupsNode.ts +186 -0
  27. package/convex/http.ts +194 -0
  28. package/convex/httpApi.handlers.test.ts +488 -0
  29. package/convex/httpApi.test.ts +70 -0
  30. package/convex/httpApi.ts +305 -0
  31. package/convex/httpApiV1.handlers.test.ts +584 -0
  32. package/convex/httpApiV1.ts +1172 -0
  33. package/convex/leaderboards.ts +39 -0
  34. package/convex/lib/access.ts +36 -0
  35. package/convex/lib/apiTokenAuth.ts +36 -0
  36. package/convex/lib/badges.ts +50 -0
  37. package/convex/lib/changelog.test.ts +34 -0
  38. package/convex/lib/changelog.ts +278 -0
  39. package/convex/lib/embeddings.ts +38 -0
  40. package/convex/lib/githubBackup.ts +443 -0
  41. package/convex/lib/githubImport.test.ts +247 -0
  42. package/convex/lib/githubImport.ts +425 -0
  43. package/convex/lib/githubSoulBackup.ts +443 -0
  44. package/convex/lib/leaderboards.ts +103 -0
  45. package/convex/lib/moderation.ts +42 -0
  46. package/convex/lib/public.ts +89 -0
  47. package/convex/lib/searchText.test.ts +46 -0
  48. package/convex/lib/searchText.ts +27 -0
  49. package/convex/lib/skillBackfill.test.ts +34 -0
  50. package/convex/lib/skillBackfill.ts +67 -0
  51. package/convex/lib/skillPublish.test.ts +28 -0
  52. package/convex/lib/skillPublish.ts +284 -0
  53. package/convex/lib/skillStats.ts +80 -0
  54. package/convex/lib/skills.test.ts +197 -0
  55. package/convex/lib/skills.ts +273 -0
  56. package/convex/lib/soulChangelog.ts +273 -0
  57. package/convex/lib/soulPublish.ts +236 -0
  58. package/convex/lib/tokens.test.ts +33 -0
  59. package/convex/lib/tokens.ts +51 -0
  60. package/convex/lib/webhooks.test.ts +91 -0
  61. package/convex/lib/webhooks.ts +112 -0
  62. package/convex/maintenance.test.ts +270 -0
  63. package/convex/maintenance.ts +840 -0
  64. package/convex/rateLimits.ts +50 -0
  65. package/convex/schema.ts +472 -0
  66. package/convex/search.test.ts +12 -0
  67. package/convex/search.ts +254 -0
  68. package/convex/seed.test.ts +37 -0
  69. package/convex/seed.ts +254 -0
  70. package/convex/seedSouls.ts +111 -0
  71. package/convex/skillStatEvents.ts +568 -0
  72. package/convex/skills.ts +1606 -0
  73. package/convex/soulComments.ts +88 -0
  74. package/convex/soulDownloads.ts +14 -0
  75. package/convex/soulStars.ts +71 -0
  76. package/convex/souls.ts +570 -0
  77. package/convex/stars.ts +108 -0
  78. package/convex/statsMaintenance.ts +205 -0
  79. package/convex/telemetry.ts +434 -0
  80. package/convex/tokens.ts +88 -0
  81. package/convex/tsconfig.json +7 -0
  82. package/convex/uploads.ts +20 -0
  83. package/convex/users.ts +122 -0
  84. package/convex/webhooks.ts +50 -0
  85. package/convex.json +3 -0
  86. package/docs/README.md +32 -0
  87. package/docs/api.md +51 -0
  88. package/docs/architecture.md +61 -0
  89. package/docs/auth.md +54 -0
  90. package/docs/cli.md +117 -0
  91. package/docs/deploy.md +78 -0
  92. package/docs/diffing.md +84 -0
  93. package/docs/github-import.md +171 -0
  94. package/docs/http-api.md +187 -0
  95. package/docs/manual-testing.md +64 -0
  96. package/docs/mintlify.md +43 -0
  97. package/docs/quickstart.md +120 -0
  98. package/docs/skill-format.md +58 -0
  99. package/docs/soul-format.md +37 -0
  100. package/docs/spec.md +177 -0
  101. package/docs/telemetry.md +91 -0
  102. package/docs/troubleshooting.md +49 -0
  103. package/docs/webhook.md +51 -0
  104. package/e2e/menu-smoke.pw.test.ts +49 -0
  105. package/e2e/pilothub.e2e.test.ts +494 -0
  106. package/e2e/search-exact.pw.test.ts +97 -0
  107. package/package.json +84 -0
  108. package/packages/pilothub/LICENSE +22 -0
  109. package/packages/pilothub/README.md +57 -0
  110. package/packages/pilothub/bin/pilothub.js +2 -0
  111. package/packages/pilothub/package.json +41 -0
  112. package/packages/pilothub/src/browserAuth.test.ts +96 -0
  113. package/packages/pilothub/src/browserAuth.ts +174 -0
  114. package/packages/pilothub/src/cli/buildInfo.ts +94 -0
  115. package/packages/pilothub/src/cli/commands/auth.ts +97 -0
  116. package/packages/pilothub/src/cli/commands/delete.test.ts +73 -0
  117. package/packages/pilothub/src/cli/commands/delete.ts +83 -0
  118. package/packages/pilothub/src/cli/commands/publish.test.ts +122 -0
  119. package/packages/pilothub/src/cli/commands/publish.ts +108 -0
  120. package/packages/pilothub/src/cli/commands/skills.test.ts +191 -0
  121. package/packages/pilothub/src/cli/commands/skills.ts +380 -0
  122. package/packages/pilothub/src/cli/commands/star.ts +46 -0
  123. package/packages/pilothub/src/cli/commands/sync.test.ts +310 -0
  124. package/packages/pilothub/src/cli/commands/sync.ts +200 -0
  125. package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +26 -0
  126. package/packages/pilothub/src/cli/commands/syncHelpers.ts +427 -0
  127. package/packages/pilothub/src/cli/commands/syncTypes.ts +27 -0
  128. package/packages/pilothub/src/cli/commands/unstar.ts +48 -0
  129. package/packages/pilothub/src/cli/helpStyle.ts +45 -0
  130. package/packages/pilothub/src/cli/pilotbotConfig.test.ts +159 -0
  131. package/packages/pilothub/src/cli/pilotbotConfig.ts +147 -0
  132. package/packages/pilothub/src/cli/registry.test.ts +63 -0
  133. package/packages/pilothub/src/cli/registry.ts +43 -0
  134. package/packages/pilothub/src/cli/scanSkills.test.ts +64 -0
  135. package/packages/pilothub/src/cli/scanSkills.ts +84 -0
  136. package/packages/pilothub/src/cli/slug.ts +16 -0
  137. package/packages/pilothub/src/cli/types.ts +12 -0
  138. package/packages/pilothub/src/cli/ui.ts +75 -0
  139. package/packages/pilothub/src/cli.ts +311 -0
  140. package/packages/pilothub/src/config.ts +36 -0
  141. package/packages/pilothub/src/discovery.test.ts +75 -0
  142. package/packages/pilothub/src/discovery.ts +19 -0
  143. package/packages/pilothub/src/http.test.ts +156 -0
  144. package/packages/pilothub/src/http.ts +301 -0
  145. package/packages/pilothub/src/schema/ark.ts +29 -0
  146. package/packages/pilothub/src/schema/index.ts +5 -0
  147. package/packages/pilothub/src/schema/routes.ts +22 -0
  148. package/packages/pilothub/src/schema/schemas.ts +260 -0
  149. package/packages/pilothub/src/schema/textFiles.test.ts +23 -0
  150. package/packages/pilothub/src/schema/textFiles.ts +66 -0
  151. package/packages/pilothub/src/skills.test.ts +191 -0
  152. package/packages/pilothub/src/skills.ts +172 -0
  153. package/packages/pilothub/src/types.ts +10 -0
  154. package/packages/pilothub/tsconfig.json +14 -0
  155. package/packages/schema/README.md +3 -0
  156. package/packages/schema/dist/ark.d.ts +4 -0
  157. package/packages/schema/dist/ark.js +26 -0
  158. package/packages/schema/dist/ark.js.map +1 -0
  159. package/packages/schema/dist/index.d.ts +5 -0
  160. package/packages/schema/dist/index.js +5 -0
  161. package/packages/schema/dist/index.js.map +1 -0
  162. package/packages/schema/dist/routes.d.ts +21 -0
  163. package/packages/schema/dist/routes.js +22 -0
  164. package/packages/schema/dist/routes.js.map +1 -0
  165. package/packages/schema/dist/schemas.d.ts +297 -0
  166. package/packages/schema/dist/schemas.js +243 -0
  167. package/packages/schema/dist/schemas.js.map +1 -0
  168. package/packages/schema/dist/textFiles.d.ts +5 -0
  169. package/packages/schema/dist/textFiles.js +66 -0
  170. package/packages/schema/dist/textFiles.js.map +1 -0
  171. package/packages/schema/package.json +26 -0
  172. package/packages/schema/src/ark.ts +29 -0
  173. package/packages/schema/src/index.ts +5 -0
  174. package/packages/schema/src/routes.ts +22 -0
  175. package/packages/schema/src/schemas.test.ts +123 -0
  176. package/packages/schema/src/schemas.ts +287 -0
  177. package/packages/schema/src/textFiles.test.ts +23 -0
  178. package/packages/schema/src/textFiles.ts +66 -0
  179. package/packages/schema/tsconfig.json +15 -0
  180. package/pilothub +46 -0
  181. package/playwright.config.ts +33 -0
  182. package/public/.well-known/pilothub.json +6 -0
  183. package/public/api/v1/openapi.json +379 -0
  184. package/public/favicon.ico +0 -0
  185. package/public/logo192.png +0 -0
  186. package/public/logo512.png +0 -0
  187. package/public/manifest.json +25 -0
  188. package/public/og.png +0 -0
  189. package/public/og.svg +98 -0
  190. package/public/pilot-logo.png +0 -0
  191. package/public/pilot-mark.png +0 -0
  192. package/public/robots.txt +3 -0
  193. package/public/tanstack-circle-logo.png +0 -0
  194. package/public/tanstack-word-logo-white.svg +1 -0
  195. package/scripts/check-peer-deps.ts +56 -0
  196. package/scripts/docs-list.ts +148 -0
  197. package/scripts/run-playwright-local.sh +14 -0
  198. package/server/og/fetchSkillOgMeta.ts +27 -0
  199. package/server/og/fetchSoulOgMeta.ts +27 -0
  200. package/server/og/ogAssets.ts +80 -0
  201. package/server/og/skillOgSvg.test.ts +59 -0
  202. package/server/og/skillOgSvg.ts +258 -0
  203. package/server/og/soulOgSvg.ts +209 -0
  204. package/server/routes/og/skill.png.ts +103 -0
  205. package/server/routes/og/soul.png.ts +111 -0
  206. package/src/__tests__/skill-detail-page.test.tsx +86 -0
  207. package/src/__tests__/skills-index.test.tsx +145 -0
  208. package/src/__tests__/upload.route.test.tsx +228 -0
  209. package/src/components/AppProviders.tsx +19 -0
  210. package/src/components/ClientOnly.tsx +18 -0
  211. package/src/components/Footer.tsx +29 -0
  212. package/src/components/Header.tsx +295 -0
  213. package/src/components/InstallSwitcher.tsx +53 -0
  214. package/src/components/SkillCard.tsx +36 -0
  215. package/src/components/SkillDetailPage.tsx +817 -0
  216. package/src/components/SkillDiffCard.tsx +485 -0
  217. package/src/components/SoulCard.tsx +19 -0
  218. package/src/components/SoulDetailPage.tsx +263 -0
  219. package/src/components/UserBootstrap.tsx +18 -0
  220. package/src/components/ui/dropdown-menu.tsx +67 -0
  221. package/src/components/ui/toggle-group.tsx +35 -0
  222. package/src/convex/client.ts +3 -0
  223. package/src/lib/badges.ts +29 -0
  224. package/src/lib/diffing.test.ts +163 -0
  225. package/src/lib/diffing.ts +106 -0
  226. package/src/lib/gravatar.test.ts +9 -0
  227. package/src/lib/gravatar.ts +158 -0
  228. package/src/lib/og.test.ts +142 -0
  229. package/src/lib/og.ts +156 -0
  230. package/src/lib/publicUser.ts +39 -0
  231. package/src/lib/roles.ts +19 -0
  232. package/src/lib/site.test.ts +130 -0
  233. package/src/lib/site.ts +84 -0
  234. package/src/lib/theme-transition.test.ts +134 -0
  235. package/src/lib/theme-transition.ts +134 -0
  236. package/src/lib/theme.test.tsx +88 -0
  237. package/src/lib/theme.ts +43 -0
  238. package/src/lib/uploadFiles.jsdom.test.ts +33 -0
  239. package/src/lib/uploadFiles.test.ts +123 -0
  240. package/src/lib/uploadFiles.ts +245 -0
  241. package/src/lib/uploadUtils.test.ts +78 -0
  242. package/src/lib/uploadUtils.ts +93 -0
  243. package/src/lib/useAuthStatus.ts +12 -0
  244. package/src/lib/utils.test.ts +9 -0
  245. package/src/lib/utils.ts +6 -0
  246. package/src/logo.svg +12 -0
  247. package/src/routeTree.gen.ts +345 -0
  248. package/src/router.tsx +17 -0
  249. package/src/routes/$owner/$slug.tsx +55 -0
  250. package/src/routes/__root.tsx +136 -0
  251. package/src/routes/admin.tsx +11 -0
  252. package/src/routes/cli/auth.tsx +168 -0
  253. package/src/routes/dashboard.tsx +97 -0
  254. package/src/routes/import.tsx +415 -0
  255. package/src/routes/index.tsx +252 -0
  256. package/src/routes/management.tsx +529 -0
  257. package/src/routes/settings.tsx +203 -0
  258. package/src/routes/skills/index.tsx +422 -0
  259. package/src/routes/souls/$slug.tsx +55 -0
  260. package/src/routes/souls/index.tsx +243 -0
  261. package/src/routes/stars.tsx +68 -0
  262. package/src/routes/u/$handle.tsx +307 -0
  263. package/src/routes/upload/utils.ts +81 -0
  264. package/src/routes/upload.tsx +499 -0
  265. package/src/styles.css +2718 -0
  266. package/tsconfig.json +24 -0
  267. package/tsconfig.oxlint.json +16 -0
  268. package/vercel.json +8 -0
  269. package/vite.config.ts +48 -0
  270. package/vitest.config.ts +47 -0
  271. package/vitest.e2e.config.ts +11 -0
  272. package/vitest.setup.ts +1 -0
@@ -0,0 +1,529 @@
1
+ import { createFileRoute, Link } 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 { Doc, Id } from '../../convex/_generated/dataModel'
6
+ import {
7
+ getSkillBadges,
8
+ isSkillDeprecated,
9
+ isSkillHighlighted,
10
+ isSkillOfficial,
11
+ } from '../lib/badges'
12
+ import { isAdmin, isModerator } from '../lib/roles'
13
+ import { useAuthStatus } from '../lib/useAuthStatus'
14
+
15
+ type ManagementSkillEntry = {
16
+ skill: Doc<'skills'>
17
+ latestVersion: Doc<'skillVersions'> | null
18
+ owner: Doc<'users'> | null
19
+ }
20
+
21
+ type RecentVersionEntry = {
22
+ version: Doc<'skillVersions'>
23
+ skill: Doc<'skills'> | null
24
+ owner: Doc<'users'> | null
25
+ }
26
+
27
+ type DuplicateCandidateEntry = {
28
+ skill: Doc<'skills'>
29
+ latestVersion: Doc<'skillVersions'> | null
30
+ fingerprint: string | null
31
+ matches: Array<{ skill: Doc<'skills'>; owner: Doc<'users'> | null }>
32
+ owner: Doc<'users'> | null
33
+ }
34
+
35
+ type SkillBySlugResult = {
36
+ skill: Doc<'skills'>
37
+ latestVersion: Doc<'skillVersions'> | null
38
+ owner: Doc<'users'> | null
39
+ canonical: {
40
+ skill: { slug: string; displayName: string }
41
+ owner: { handle: string | null; userId: Id<'users'> | null }
42
+ } | null
43
+ } | null
44
+
45
+ function resolveOwnerParam(handle: string | null | undefined, ownerId?: Id<'users'>) {
46
+ return handle?.trim() || (ownerId ? String(ownerId) : 'unknown')
47
+ }
48
+
49
+ export const Route = createFileRoute('/management')({
50
+ validateSearch: (search) => ({
51
+ skill: typeof search.skill === 'string' && search.skill.trim() ? search.skill : undefined,
52
+ }),
53
+ component: Management,
54
+ })
55
+
56
+ function Management() {
57
+ const { me } = useAuthStatus()
58
+ const search = Route.useSearch()
59
+ const staff = isModerator(me)
60
+ const admin = isAdmin(me)
61
+
62
+ const users = useQuery(api.users.list, admin ? { limit: 50 } : 'skip') as
63
+ | Doc<'users'>[]
64
+ | undefined
65
+ const selectedSlug = search.skill?.trim()
66
+ const selectedSkill = useQuery(
67
+ api.skills.getBySlug,
68
+ staff && selectedSlug ? { slug: selectedSlug } : 'skip',
69
+ ) as SkillBySlugResult | undefined
70
+ const recentVersions = useQuery(api.skills.listRecentVersions, staff ? { limit: 20 } : 'skip') as
71
+ | RecentVersionEntry[]
72
+ | undefined
73
+ const reportedSkills = useQuery(api.skills.listReportedSkills, staff ? { limit: 25 } : 'skip') as
74
+ | ManagementSkillEntry[]
75
+ | undefined
76
+ const duplicateCandidates = useQuery(
77
+ api.skills.listDuplicateCandidates,
78
+ staff ? { limit: 20 } : 'skip',
79
+ ) as DuplicateCandidateEntry[] | undefined
80
+
81
+ const setRole = useMutation(api.users.setRole)
82
+ const setBatch = useMutation(api.skills.setBatch)
83
+ const setSoftDeleted = useMutation(api.skills.setSoftDeleted)
84
+ const hardDelete = useMutation(api.skills.hardDelete)
85
+ const changeOwner = useMutation(api.skills.changeOwner)
86
+ const setDuplicate = useMutation(api.skills.setDuplicate)
87
+ const setOfficialBadge = useMutation(api.skills.setOfficialBadge)
88
+ const setDeprecatedBadge = useMutation(api.skills.setDeprecatedBadge)
89
+
90
+ const [selectedDuplicate, setSelectedDuplicate] = useState('')
91
+ const [selectedOwner, setSelectedOwner] = useState('')
92
+
93
+ const selectedSkillId = selectedSkill?.skill?._id ?? null
94
+ const selectedOwnerUserId = selectedSkill?.skill?.ownerUserId ?? null
95
+ const selectedCanonicalSlug = selectedSkill?.canonical?.skill?.slug ?? ''
96
+
97
+ useEffect(() => {
98
+ if (!selectedSkillId || !selectedOwnerUserId) return
99
+ setSelectedDuplicate(selectedCanonicalSlug)
100
+ setSelectedOwner(String(selectedOwnerUserId))
101
+ }, [selectedCanonicalSlug, selectedOwnerUserId, selectedSkillId])
102
+
103
+ if (!staff) {
104
+ return (
105
+ <main className="section">
106
+ <div className="card">Management only.</div>
107
+ </main>
108
+ )
109
+ }
110
+
111
+ if (!recentVersions || !reportedSkills || !duplicateCandidates) {
112
+ return (
113
+ <main className="section">
114
+ <div className="card">Loading management console…</div>
115
+ </main>
116
+ )
117
+ }
118
+
119
+ return (
120
+ <main className="section">
121
+ <h1 className="section-title">Management console</h1>
122
+ <p className="section-subtitle">Moderation, curation, and ownership tools.</p>
123
+
124
+ <div className="card">
125
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
126
+ Reported skills
127
+ </h2>
128
+ <div className="management-list">
129
+ {reportedSkills.length === 0 ? (
130
+ <div className="stat">No reports yet.</div>
131
+ ) : (
132
+ reportedSkills.map((entry) => {
133
+ const { skill, latestVersion, owner } = entry
134
+ const ownerParam = resolveOwnerParam(
135
+ owner?.handle ?? null,
136
+ owner?._id ?? skill.ownerUserId,
137
+ )
138
+ return (
139
+ <div key={skill._id} className="management-item">
140
+ <div className="management-item-main">
141
+ <Link to="/$owner/$slug" params={{ owner: ownerParam, slug: skill.slug }}>
142
+ {skill.displayName}
143
+ </Link>
144
+ <div className="section-subtitle" style={{ margin: 0 }}>
145
+ @{owner?.handle ?? owner?.name ?? 'user'} · v{latestVersion?.version ?? '—'} ·
146
+ {skill.reportCount ?? 0} report{(skill.reportCount ?? 0) === 1 ? '' : 's'}
147
+ {skill.lastReportedAt
148
+ ? ` · last ${formatTimestamp(skill.lastReportedAt)}`
149
+ : ''}
150
+ </div>
151
+ </div>
152
+ <div className="management-actions">
153
+ <button
154
+ className="btn"
155
+ type="button"
156
+ onClick={() =>
157
+ void setSoftDeleted({ skillId: skill._id, deleted: !skill.softDeletedAt })
158
+ }
159
+ >
160
+ {skill.softDeletedAt ? 'Restore' : 'Hide'}
161
+ </button>
162
+ {admin ? (
163
+ <button
164
+ className="btn"
165
+ type="button"
166
+ onClick={() => {
167
+ if (!window.confirm(`Hard delete ${skill.displayName}?`)) return
168
+ void hardDelete({ skillId: skill._id })
169
+ }}
170
+ >
171
+ Hard delete
172
+ </button>
173
+ ) : null}
174
+ </div>
175
+ </div>
176
+ )
177
+ })
178
+ )}
179
+ </div>
180
+ </div>
181
+
182
+ <div className="card" style={{ marginTop: 20 }}>
183
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
184
+ Skill tools
185
+ </h2>
186
+ {selectedSlug ? (
187
+ <div className="section-subtitle" style={{ marginTop: 8 }}>
188
+ Managing "{selectedSlug}" ·{' '}
189
+ <Link to="/management" search={{ skill: undefined }}>
190
+ Clear selection
191
+ </Link>
192
+ </div>
193
+ ) : null}
194
+ <div className="management-list">
195
+ {!selectedSlug ? (
196
+ <div className="stat">Use the Manage button on a skill to open tooling here.</div>
197
+ ) : selectedSkill === undefined ? (
198
+ <div className="stat">Loading skill…</div>
199
+ ) : !selectedSkill?.skill ? (
200
+ <div className="stat">No skill found for "{selectedSlug}".</div>
201
+ ) : (
202
+ (() => {
203
+ const { skill, latestVersion, owner, canonical } = selectedSkill
204
+ const ownerParam = resolveOwnerParam(
205
+ owner?.handle ?? null,
206
+ owner?._id ?? skill.ownerUserId,
207
+ )
208
+ const moderationStatus =
209
+ skill.moderationStatus ?? (skill.softDeletedAt ? 'hidden' : 'active')
210
+ const isHighlighted = isSkillHighlighted(skill)
211
+ const isOfficial = isSkillOfficial(skill)
212
+ const isDeprecated = isSkillDeprecated(skill)
213
+ const badges = getSkillBadges(skill)
214
+
215
+ return (
216
+ <div key={skill._id} className="management-item">
217
+ <div className="management-item-main">
218
+ <Link to="/$owner/$slug" params={{ owner: ownerParam, slug: skill.slug }}>
219
+ {skill.displayName}
220
+ </Link>
221
+ <div className="section-subtitle" style={{ margin: 0 }}>
222
+ @{owner?.handle ?? owner?.name ?? 'user'} · v{latestVersion?.version ?? '—'} ·
223
+ updated {formatTimestamp(skill.updatedAt)} · {moderationStatus}
224
+ {badges.length ? ` · ${badges.join(', ').toLowerCase()}` : ''}
225
+ </div>
226
+ {skill.moderationFlags?.length ? (
227
+ <div className="management-tags">
228
+ {skill.moderationFlags.map((flag: string) => (
229
+ <span key={flag} className="tag">
230
+ {flag}
231
+ </span>
232
+ ))}
233
+ </div>
234
+ ) : null}
235
+ <div className="management-controls">
236
+ <label className="management-control">
237
+ <span className="mono">duplicate of</span>
238
+ <input
239
+ className="search-input"
240
+ value={selectedDuplicate}
241
+ onChange={(event) => setSelectedDuplicate(event.target.value)}
242
+ placeholder={canonical?.skill?.slug ?? 'canonical slug'}
243
+ />
244
+ </label>
245
+ <button
246
+ className="btn"
247
+ type="button"
248
+ onClick={() =>
249
+ void setDuplicate({
250
+ skillId: skill._id,
251
+ canonicalSlug: selectedDuplicate.trim() || undefined,
252
+ })
253
+ }
254
+ >
255
+ Set duplicate
256
+ </button>
257
+ {admin ? (
258
+ <label className="management-control">
259
+ <span className="mono">owner</span>
260
+ <select
261
+ value={selectedOwner}
262
+ onChange={(event) => setSelectedOwner(event.target.value)}
263
+ >
264
+ {(users ?? []).map((user) => (
265
+ <option key={user._id} value={user._id}>
266
+ @{user.handle ?? user.name ?? 'user'}
267
+ </option>
268
+ ))}
269
+ </select>
270
+ <button
271
+ className="btn"
272
+ type="button"
273
+ onClick={() =>
274
+ void changeOwner({
275
+ skillId: skill._id,
276
+ ownerUserId: selectedOwner as Doc<'users'>['_id'],
277
+ })
278
+ }
279
+ >
280
+ Change owner
281
+ </button>
282
+ </label>
283
+ ) : null}
284
+ </div>
285
+ </div>
286
+ <div className="management-actions">
287
+ <Link
288
+ className="btn"
289
+ to="/$owner/$slug"
290
+ params={{ owner: ownerParam, slug: skill.slug }}
291
+ >
292
+ View
293
+ </Link>
294
+ <button
295
+ className="btn"
296
+ type="button"
297
+ onClick={() =>
298
+ void setSoftDeleted({ skillId: skill._id, deleted: !skill.softDeletedAt })
299
+ }
300
+ >
301
+ {skill.softDeletedAt ? 'Restore' : 'Hide'}
302
+ </button>
303
+ <button
304
+ className="btn"
305
+ type="button"
306
+ onClick={() =>
307
+ void setBatch({
308
+ skillId: skill._id,
309
+ batch: isHighlighted ? undefined : 'highlighted',
310
+ })
311
+ }
312
+ >
313
+ {isHighlighted ? 'Unhighlight' : 'Highlight'}
314
+ </button>
315
+ {admin ? (
316
+ <button
317
+ className="btn"
318
+ type="button"
319
+ onClick={() => {
320
+ if (!window.confirm(`Hard delete ${skill.displayName}?`)) return
321
+ void hardDelete({ skillId: skill._id })
322
+ }}
323
+ >
324
+ Hard delete
325
+ </button>
326
+ ) : null}
327
+ {admin ? (
328
+ <>
329
+ <button
330
+ className="btn"
331
+ type="button"
332
+ onClick={() =>
333
+ void setOfficialBadge({
334
+ skillId: skill._id,
335
+ official: !isOfficial,
336
+ })
337
+ }
338
+ >
339
+ {isOfficial ? 'Remove official' : 'Mark official'}
340
+ </button>
341
+ <button
342
+ className="btn"
343
+ type="button"
344
+ onClick={() =>
345
+ void setDeprecatedBadge({
346
+ skillId: skill._id,
347
+ deprecated: !isDeprecated,
348
+ })
349
+ }
350
+ >
351
+ {isDeprecated ? 'Remove deprecated' : 'Mark deprecated'}
352
+ </button>
353
+ </>
354
+ ) : null}
355
+ </div>
356
+ </div>
357
+ )
358
+ })()
359
+ )}
360
+ </div>
361
+ </div>
362
+
363
+ <div className="card" style={{ marginTop: 20 }}>
364
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
365
+ Duplicate candidates
366
+ </h2>
367
+ <div className="management-list">
368
+ {duplicateCandidates.length === 0 ? (
369
+ <div className="stat">No duplicate candidates.</div>
370
+ ) : (
371
+ duplicateCandidates.map((entry) => (
372
+ <div key={entry.skill._id} className="management-item">
373
+ <div className="management-item-main">
374
+ <Link
375
+ to="/$owner/$slug"
376
+ params={{
377
+ owner: resolveOwnerParam(
378
+ entry.owner?.handle ?? null,
379
+ entry.owner?._id ?? entry.skill.ownerUserId,
380
+ ),
381
+ slug: entry.skill.slug,
382
+ }}
383
+ >
384
+ {entry.skill.displayName}
385
+ </Link>
386
+ <div className="section-subtitle" style={{ margin: 0 }}>
387
+ @{entry.owner?.handle ?? entry.owner?.name ?? 'user'} · v
388
+ {entry.latestVersion?.version ?? '—'} · fingerprint{' '}
389
+ {entry.fingerprint?.slice(0, 8)}
390
+ </div>
391
+ <div className="management-sublist">
392
+ {entry.matches.map((match) => (
393
+ <div key={match.skill._id} className="management-subitem">
394
+ <div>
395
+ <strong>{match.skill.displayName}</strong>
396
+ <div className="section-subtitle" style={{ margin: 0 }}>
397
+ @{match.owner?.handle ?? match.owner?.name ?? 'user'} ·{' '}
398
+ {match.skill.slug}
399
+ </div>
400
+ </div>
401
+ <div className="management-actions">
402
+ <Link
403
+ className="btn"
404
+ to="/$owner/$slug"
405
+ params={{
406
+ owner: resolveOwnerParam(
407
+ match.owner?.handle ?? null,
408
+ match.owner?._id ?? match.skill.ownerUserId,
409
+ ),
410
+ slug: match.skill.slug,
411
+ }}
412
+ >
413
+ View
414
+ </Link>
415
+ <button
416
+ className="btn"
417
+ type="button"
418
+ onClick={() =>
419
+ void setDuplicate({
420
+ skillId: entry.skill._id,
421
+ canonicalSlug: match.skill.slug,
422
+ })
423
+ }
424
+ >
425
+ Mark duplicate
426
+ </button>
427
+ </div>
428
+ </div>
429
+ ))}
430
+ </div>
431
+ </div>
432
+ <div className="management-actions">
433
+ <Link
434
+ className="btn"
435
+ to="/$owner/$slug"
436
+ params={{
437
+ owner: resolveOwnerParam(
438
+ entry.owner?.handle ?? null,
439
+ entry.owner?._id ?? entry.skill.ownerUserId,
440
+ ),
441
+ slug: entry.skill.slug,
442
+ }}
443
+ >
444
+ View
445
+ </Link>
446
+ </div>
447
+ </div>
448
+ ))
449
+ )}
450
+ </div>
451
+ </div>
452
+
453
+ <div className="card" style={{ marginTop: 20 }}>
454
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
455
+ Recent pushes
456
+ </h2>
457
+ <div className="management-list">
458
+ {recentVersions.length === 0 ? (
459
+ <div className="stat">No recent versions.</div>
460
+ ) : (
461
+ recentVersions.map((entry) => (
462
+ <div key={entry.version._id} className="management-item">
463
+ <div className="management-item-main">
464
+ <strong>{entry.skill?.displayName ?? 'Unknown skill'}</strong>
465
+ <div className="section-subtitle" style={{ margin: 0 }}>
466
+ v{entry.version.version} · @{entry.owner?.handle ?? entry.owner?.name ?? 'user'}
467
+ </div>
468
+ </div>
469
+ <div className="management-actions">
470
+ {entry.skill ? (
471
+ <Link
472
+ className="btn"
473
+ to="/$owner/$slug"
474
+ params={{
475
+ owner: resolveOwnerParam(
476
+ entry.owner?.handle ?? null,
477
+ entry.owner?._id ?? entry.skill.ownerUserId,
478
+ ),
479
+ slug: entry.skill.slug,
480
+ }}
481
+ >
482
+ View
483
+ </Link>
484
+ ) : null}
485
+ </div>
486
+ </div>
487
+ ))
488
+ )}
489
+ </div>
490
+ </div>
491
+
492
+ {admin ? (
493
+ <div className="card" style={{ marginTop: 20 }}>
494
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
495
+ Users
496
+ </h2>
497
+ <div className="management-list">
498
+ {(users ?? []).map((user) => (
499
+ <div key={user._id} className="management-item">
500
+ <div className="management-item-main">
501
+ <span className="mono">@{user.handle ?? user.name ?? 'user'}</span>
502
+ </div>
503
+ <div className="management-actions">
504
+ <select
505
+ value={user.role ?? 'user'}
506
+ onChange={(event) => {
507
+ const value = event.target.value
508
+ if (value === 'admin' || value === 'moderator' || value === 'user') {
509
+ void setRole({ userId: user._id, role: value })
510
+ }
511
+ }}
512
+ >
513
+ <option value="user">User</option>
514
+ <option value="moderator">Moderator</option>
515
+ <option value="admin">Admin</option>
516
+ </select>
517
+ </div>
518
+ </div>
519
+ ))}
520
+ </div>
521
+ </div>
522
+ ) : null}
523
+ </main>
524
+ )
525
+ }
526
+
527
+ function formatTimestamp(value: number) {
528
+ return new Date(value).toLocaleString()
529
+ }