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,485 @@
1
+ import type { DiffEditorProps } from '@monaco-editor/react'
2
+ import { DiffEditor, useMonaco } from '@monaco-editor/react'
3
+ import { useAction } from 'convex/react'
4
+ import { useEffect, useMemo, useRef, useState } from 'react'
5
+ import { api } from '../../convex/_generated/api'
6
+ import type { Doc, Id } from '../../convex/_generated/dataModel'
7
+ import {
8
+ buildFileDiffList,
9
+ getDefaultDiffSelection,
10
+ MAX_DIFF_FILE_BYTES,
11
+ resolveLatestVersionId,
12
+ resolvePreviousVersionId,
13
+ selectDefaultFilePath,
14
+ sortVersionsBySemver,
15
+ } from '../lib/diffing'
16
+ import { ClientOnly } from './ClientOnly'
17
+
18
+ type SkillDiffCardProps = {
19
+ skill: Doc<'skills'>
20
+ versions: Doc<'skillVersions'>[]
21
+ variant?: 'card' | 'embedded'
22
+ }
23
+
24
+ type VersionOption = {
25
+ value: Id<'skillVersions'>
26
+ label: string
27
+ group: 'Special' | 'Tags' | 'Versions'
28
+ disabled?: boolean
29
+ }
30
+
31
+ type FileSide = 'left' | 'right'
32
+
33
+ type SizeWarning = {
34
+ side: FileSide
35
+ path: string
36
+ }
37
+
38
+ const EMPTY_DIFF_TEXT = ''
39
+
40
+ export function SkillDiffCard({ skill, versions, variant = 'card' }: SkillDiffCardProps) {
41
+ const getFileText = useAction(api.skills.getFileText)
42
+ const monaco = useMonaco()
43
+ const [viewMode, setViewMode] = useState<'split' | 'inline'>('split')
44
+ const [leftVersionId, setLeftVersionId] = useState<Id<'skillVersions'> | null>(null)
45
+ const [rightVersionId, setRightVersionId] = useState<Id<'skillVersions'> | null>(null)
46
+ const [selectedPath, setSelectedPath] = useState<string | null>(null)
47
+ const [leftText, setLeftText] = useState(EMPTY_DIFF_TEXT)
48
+ const [rightText, setRightText] = useState(EMPTY_DIFF_TEXT)
49
+ const [isLoading, setIsLoading] = useState(false)
50
+ const [error, setError] = useState<string | null>(null)
51
+ const [sizeWarning, setSizeWarning] = useState<SizeWarning | null>(null)
52
+ const cacheRef = useRef(new Map<string, string>())
53
+
54
+ const versionEntries = useMemo(
55
+ () => versions.map((entry) => ({ id: entry._id, version: entry.version })),
56
+ [versions],
57
+ )
58
+ const orderedVersions = useMemo(() => sortVersionsBySemver(versionEntries), [versionEntries])
59
+ const versionById = useMemo(
60
+ () => new Map(versions.map((entry) => [entry._id, entry])),
61
+ [versions],
62
+ )
63
+
64
+ const latestId = useMemo(
65
+ () => resolveLatestVersionId(versionEntries, skill.tags),
66
+ [versionEntries, skill.tags],
67
+ )
68
+ const previousId = useMemo(
69
+ () => resolvePreviousVersionId(versionEntries, latestId),
70
+ [versionEntries, latestId],
71
+ )
72
+
73
+ const versionOptions = useMemo(() => {
74
+ const options: VersionOption[] = []
75
+ if (latestId) {
76
+ const version = versionById.get(latestId)?.version
77
+ options.push({
78
+ value: latestId,
79
+ label: version ? `latest (v${version})` : 'latest',
80
+ group: 'Special',
81
+ })
82
+ }
83
+ if (previousId) {
84
+ const version = versionById.get(previousId)?.version
85
+ options.push({
86
+ value: previousId,
87
+ label: version ? `previous (v${version})` : 'previous',
88
+ group: 'Special',
89
+ })
90
+ } else if (versions.length > 0) {
91
+ options.push({
92
+ value: versions[0]._id,
93
+ label: 'previous (unavailable)',
94
+ group: 'Special',
95
+ disabled: true,
96
+ })
97
+ }
98
+
99
+ const tagEntries = Object.entries(skill.tags ?? {})
100
+ .filter(([tag]) => tag !== 'latest')
101
+ .sort(([a], [b]) => a.localeCompare(b))
102
+ for (const [tag, versionId] of tagEntries) {
103
+ const version = versionById.get(versionId)?.version
104
+ options.push({
105
+ value: versionId,
106
+ label: version ? `tag: ${tag} (v${version})` : `tag: ${tag}`,
107
+ group: 'Tags',
108
+ disabled: !versionById.has(versionId),
109
+ })
110
+ }
111
+
112
+ for (const entry of orderedVersions) {
113
+ options.push({
114
+ value: entry.id,
115
+ label: `v${entry.version}`,
116
+ group: 'Versions',
117
+ })
118
+ }
119
+
120
+ return options
121
+ }, [latestId, previousId, orderedVersions, skill.tags, versionById, versions])
122
+
123
+ useEffect(() => {
124
+ if (!versions.length) return
125
+ const defaults = getDefaultDiffSelection(versionEntries, skill.tags)
126
+ setLeftVersionId((current) => {
127
+ if (current && versionById.has(current)) return current
128
+ return defaults.leftId ? (defaults.leftId as Id<'skillVersions'>) : null
129
+ })
130
+ setRightVersionId((current) => {
131
+ if (current && versionById.has(current)) return current
132
+ return defaults.rightId ? (defaults.rightId as Id<'skillVersions'>) : null
133
+ })
134
+ }, [versionEntries, skill.tags, versionById, versions.length])
135
+
136
+ const leftVersion = leftVersionId ? (versionById.get(leftVersionId) ?? null) : null
137
+ const rightVersion = rightVersionId ? (versionById.get(rightVersionId) ?? null) : null
138
+
139
+ const fileDiffItems = useMemo(() => {
140
+ return buildFileDiffList(leftVersion?.files ?? [], rightVersion?.files ?? [])
141
+ }, [leftVersion, rightVersion])
142
+
143
+ useEffect(() => {
144
+ if (!fileDiffItems.length) {
145
+ setSelectedPath(null)
146
+ return
147
+ }
148
+ setSelectedPath((current) => {
149
+ if (current && fileDiffItems.some((item) => item.path === current)) return current
150
+ return selectDefaultFilePath(fileDiffItems)
151
+ })
152
+ }, [fileDiffItems])
153
+
154
+ const selectedItem = useMemo(
155
+ () => fileDiffItems.find((item) => item.path === selectedPath) ?? null,
156
+ [fileDiffItems, selectedPath],
157
+ )
158
+
159
+ useEffect(() => {
160
+ let cancelled = false
161
+ async function loadText(versionId: Id<'skillVersions'>, path: string) {
162
+ const cacheKey = `${versionId}:${path}`
163
+ const cached = cacheRef.current.get(cacheKey)
164
+ if (cached !== undefined) return cached
165
+ const result = await getFileText({ versionId, path })
166
+ cacheRef.current.set(cacheKey, result.text)
167
+ return result.text
168
+ }
169
+
170
+ async function load() {
171
+ if (!selectedItem || !leftVersionId || !rightVersionId) {
172
+ setLeftText(EMPTY_DIFF_TEXT)
173
+ setRightText(EMPTY_DIFF_TEXT)
174
+ return
175
+ }
176
+
177
+ setIsLoading(true)
178
+ setError(null)
179
+ setSizeWarning(null)
180
+
181
+ const leftFile = selectedItem.left
182
+ const rightFile = selectedItem.right
183
+ const warnings: SizeWarning[] = []
184
+
185
+ if (leftFile && leftFile.size > MAX_DIFF_FILE_BYTES) {
186
+ warnings.push({ side: 'left', path: leftFile.path })
187
+ }
188
+ if (rightFile && rightFile.size > MAX_DIFF_FILE_BYTES) {
189
+ warnings.push({ side: 'right', path: rightFile.path })
190
+ }
191
+
192
+ if (warnings.length) {
193
+ if (!cancelled) {
194
+ setSizeWarning(warnings[0])
195
+ setLeftText(EMPTY_DIFF_TEXT)
196
+ setRightText(EMPTY_DIFF_TEXT)
197
+ setIsLoading(false)
198
+ }
199
+ return
200
+ }
201
+
202
+ try {
203
+ const [nextLeft, nextRight] = await Promise.all([
204
+ leftFile ? loadText(leftVersionId, leftFile.path) : Promise.resolve(''),
205
+ rightFile ? loadText(rightVersionId, rightFile.path) : Promise.resolve(''),
206
+ ])
207
+ if (cancelled) return
208
+ setLeftText(nextLeft ?? EMPTY_DIFF_TEXT)
209
+ setRightText(nextRight ?? EMPTY_DIFF_TEXT)
210
+ } catch (err) {
211
+ if (cancelled) return
212
+ setError(err instanceof Error ? err.message : 'Failed to load diff')
213
+ } finally {
214
+ if (!cancelled) setIsLoading(false)
215
+ }
216
+ }
217
+
218
+ void load()
219
+
220
+ return () => {
221
+ cancelled = true
222
+ }
223
+ }, [getFileText, leftVersionId, rightVersionId, selectedItem])
224
+
225
+ useEffect(() => {
226
+ if (!monaco || typeof document === 'undefined') return
227
+ const observer = new MutationObserver(() => {
228
+ applyMonacoTheme(monaco)
229
+ })
230
+ observer.observe(document.documentElement, {
231
+ attributes: true,
232
+ attributeFilter: ['data-theme'],
233
+ })
234
+ applyMonacoTheme(monaco)
235
+ return () => observer.disconnect()
236
+ }, [monaco])
237
+
238
+ const leftLabel = leftVersion ? `v${leftVersion.version}` : '—'
239
+ const rightLabel = rightVersion ? `v${rightVersion.version}` : '—'
240
+ const diffUnavailable = versions.length < 2
241
+ const selectionReady = Boolean(leftVersionId && rightVersionId)
242
+ const fileSelected = Boolean(selectedItem)
243
+
244
+ const containerClass = variant === 'card' ? 'card diff-card' : 'diff-card diff-card-embedded'
245
+
246
+ return (
247
+ <div className={containerClass}>
248
+ <div className="diff-header">
249
+ <div>
250
+ <h2 className="section-title" style={{ fontSize: '1.2rem', margin: 0 }}>
251
+ Compare versions
252
+ </h2>
253
+ <p className="section-subtitle" style={{ margin: 0 }}>
254
+ Inline or side-by-side diff for any file.
255
+ </p>
256
+ </div>
257
+ <fieldset className="diff-toggle-group">
258
+ <legend className="sr-only">Diff layout</legend>
259
+ <button
260
+ className={`diff-toggle${viewMode === 'split' ? ' is-active' : ''}`}
261
+ type="button"
262
+ onClick={() => setViewMode('split')}
263
+ >
264
+ Side-by-side
265
+ </button>
266
+ <button
267
+ className={`diff-toggle${viewMode === 'inline' ? ' is-active' : ''}`}
268
+ type="button"
269
+ onClick={() => setViewMode('inline')}
270
+ >
271
+ Inline
272
+ </button>
273
+ </fieldset>
274
+ </div>
275
+
276
+ <div className="diff-controls">
277
+ <div className="diff-select">
278
+ <label htmlFor="diff-left">Left</label>
279
+ <select
280
+ id="diff-left"
281
+ className="search-input"
282
+ value={leftVersionId ?? ''}
283
+ onChange={(event) => setLeftVersionId(event.target.value as Id<'skillVersions'>)}
284
+ >
285
+ <option value="" disabled>
286
+ Select version
287
+ </option>
288
+ {renderOptions(versionOptions)}
289
+ </select>
290
+ </div>
291
+ <button
292
+ className="btn diff-swap"
293
+ type="button"
294
+ onClick={() => {
295
+ setLeftVersionId(rightVersionId)
296
+ setRightVersionId(leftVersionId)
297
+ }}
298
+ disabled={!leftVersionId || !rightVersionId}
299
+ >
300
+ Swap
301
+ </button>
302
+ <div className="diff-select">
303
+ <label htmlFor="diff-right">Right</label>
304
+ <select
305
+ id="diff-right"
306
+ className="search-input"
307
+ value={rightVersionId ?? ''}
308
+ onChange={(event) => setRightVersionId(event.target.value as Id<'skillVersions'>)}
309
+ >
310
+ <option value="" disabled>
311
+ Select version
312
+ </option>
313
+ {renderOptions(versionOptions)}
314
+ </select>
315
+ </div>
316
+ </div>
317
+
318
+ <div className="diff-meta">
319
+ <span>
320
+ Left {leftLabel} • Right {rightLabel}
321
+ </span>
322
+ {diffUnavailable ? <span>Need at least 2 versions.</span> : null}
323
+ </div>
324
+
325
+ <div className="diff-layout">
326
+ <div className="diff-files">
327
+ {fileDiffItems.length === 0 ? (
328
+ <div className="diff-empty">No files to compare.</div>
329
+ ) : (
330
+ fileDiffItems.map((item) => (
331
+ <button
332
+ key={item.path}
333
+ type="button"
334
+ className={`diff-file${item.path === selectedPath ? ' is-active' : ''}`}
335
+ onClick={() => setSelectedPath(item.path)}
336
+ >
337
+ <span className={`diff-pill diff-pill-${item.status}`}>{item.status}</span>
338
+ <span className="diff-file-name">{item.path}</span>
339
+ </button>
340
+ ))
341
+ )}
342
+ </div>
343
+ <div className="diff-view">
344
+ {error ? (
345
+ <div className="diff-empty">{error}</div>
346
+ ) : sizeWarning ? (
347
+ <div className="diff-empty">
348
+ {sizeWarning.side === 'left' ? 'Left' : 'Right'} file exceeds 200KB:{' '}
349
+ {sizeWarning.path}
350
+ </div>
351
+ ) : diffUnavailable ? (
352
+ <div className="diff-empty">Publish another version to compare.</div>
353
+ ) : !selectionReady ? (
354
+ <div className="diff-empty">Select two versions to compare.</div>
355
+ ) : !fileSelected ? (
356
+ <div className="diff-empty">Select a file to compare.</div>
357
+ ) : (
358
+ <ClientOnly fallback={<div className="diff-empty">Preparing diff…</div>}>
359
+ <DiffEditor
360
+ className="diff-monaco"
361
+ original={leftText}
362
+ modified={rightText}
363
+ theme={getMonacoThemeName()}
364
+ loading={<div className="diff-empty">Loading diff…</div>}
365
+ options={buildDiffOptions(viewMode)}
366
+ />
367
+ {isLoading ? <div className="diff-loading">Loading…</div> : null}
368
+ </ClientOnly>
369
+ )}
370
+ </div>
371
+ </div>
372
+ </div>
373
+ )
374
+ }
375
+
376
+ function renderOptions(options: VersionOption[]) {
377
+ const groups: Record<VersionOption['group'], VersionOption[]> = {
378
+ Special: [],
379
+ Tags: [],
380
+ Versions: [],
381
+ }
382
+ for (const option of options) {
383
+ groups[option.group].push(option)
384
+ }
385
+ return (['Special', 'Tags', 'Versions'] as const)
386
+ .filter((group) => groups[group].length > 0)
387
+ .map((group) => (
388
+ <optgroup key={group} label={group}>
389
+ {groups[group].map((option) => (
390
+ <option key={`${group}-${option.value}`} value={option.value} disabled={option.disabled}>
391
+ {option.label}
392
+ </option>
393
+ ))}
394
+ </optgroup>
395
+ ))
396
+ }
397
+
398
+ function getMonacoThemeName() {
399
+ if (typeof document === 'undefined') return 'pilothub-light'
400
+ return document.documentElement.dataset.theme === 'dark' ? 'pilothub-dark' : 'pilothub-light'
401
+ }
402
+
403
+ function buildDiffOptions(viewMode: 'split' | 'inline'): DiffEditorProps['options'] {
404
+ return {
405
+ readOnly: true,
406
+ renderSideBySide: viewMode === 'split',
407
+ renderSideBySideInlineBreakpoint: 860,
408
+ wordWrap: 'on',
409
+ minimap: { enabled: false },
410
+ scrollBeyondLastLine: false,
411
+ overviewRulerBorder: false,
412
+ renderIndicators: true,
413
+ diffAlgorithm: 'advanced',
414
+ fontFamily: 'var(--font-mono)',
415
+ fontSize: 13,
416
+ }
417
+ }
418
+
419
+ function applyMonacoTheme(monaco: NonNullable<ReturnType<typeof useMonaco>>) {
420
+ const styles = getComputedStyle(document.documentElement)
421
+ const surface = normalizeHex(styles.getPropertyValue('--surface').trim() || '#ffffff')
422
+ const surfaceMuted = styles.getPropertyValue('--surface-muted').trim() || '#f6f1ec'
423
+ const ink = styles.getPropertyValue('--ink').trim() || '#1d1a17'
424
+ const inkSoft = styles.getPropertyValue('--ink-soft').trim() || '#4c463f'
425
+ const line = styles.getPropertyValue('--line').trim() || 'rgba(29, 26, 23, 0.12)'
426
+ const accent = styles.getPropertyValue('--accent').trim() || '#ff6b4a'
427
+ const seafoam = styles.getPropertyValue('--seafoam').trim() || '#2bc6a4'
428
+ const diffAdded = styles.getPropertyValue('--diff-added').trim() || seafoam
429
+ const diffRemoved = styles.getPropertyValue('--diff-removed').trim() || accent
430
+ const background = surface
431
+ const gutter = surfaceMuted
432
+ const isDark = document.documentElement.dataset.theme === 'dark'
433
+ const base = isDark ? 'vs-dark' : 'vs'
434
+
435
+ const diffInserted = toRgba(diffAdded, isDark ? 0.12 : 0.16)
436
+ const diffRemovedBg = toRgba(diffRemoved, isDark ? 0.12 : 0.16)
437
+
438
+ monaco.editor.defineTheme(`pilothub-${isDark ? 'dark' : 'light'}`, {
439
+ base,
440
+ inherit: true,
441
+ rules: [
442
+ { token: '', foreground: normalizeHex(ink) },
443
+ { token: 'comment', foreground: normalizeHex(inkSoft) },
444
+ ],
445
+ colors: {
446
+ 'editor.background': background,
447
+ 'editor.foreground': ink,
448
+ 'editorLineNumber.foreground': inkSoft,
449
+ 'editorLineNumber.activeForeground': ink,
450
+ 'editorGutter.background': gutter,
451
+ 'editor.selectionBackground': toRgba(accent, 0.18),
452
+ 'editor.inactiveSelectionBackground': toRgba(accent, 0.12),
453
+ 'editorWidget.background': surface,
454
+ 'editorWidget.border': line,
455
+ 'editorWidget.foreground': ink,
456
+ 'diffEditor.insertedTextBackground': diffInserted,
457
+ 'diffEditor.removedTextBackground': diffRemovedBg,
458
+ 'diffEditor.insertedLineBackground': diffInserted,
459
+ 'diffEditor.removedLineBackground': diffRemovedBg,
460
+ 'diffEditor.border': line,
461
+ 'scrollbarSlider.background': toRgba(inkSoft, 0.15),
462
+ 'scrollbarSlider.hoverBackground': toRgba(inkSoft, 0.28),
463
+ 'scrollbarSlider.activeBackground': toRgba(inkSoft, 0.4),
464
+ },
465
+ })
466
+
467
+ monaco.editor.setTheme(`pilothub-${isDark ? 'dark' : 'light'}`)
468
+ }
469
+
470
+ function normalizeHex(value: string) {
471
+ if (!value.startsWith('#')) return value
472
+ if (value.length === 4) {
473
+ return `#${value[1]}${value[1]}${value[2]}${value[2]}${value[3]}${value[3]}`
474
+ }
475
+ return value
476
+ }
477
+
478
+ function toRgba(color: string, alpha: number) {
479
+ const hex = normalizeHex(color).replace('#', '')
480
+ if (hex.length !== 6) return color
481
+ const r = Number.parseInt(hex.slice(0, 2), 16)
482
+ const g = Number.parseInt(hex.slice(2, 4), 16)
483
+ const b = Number.parseInt(hex.slice(4, 6), 16)
484
+ return `rgba(${r}, ${g}, ${b}, ${alpha})`
485
+ }
@@ -0,0 +1,19 @@
1
+ import { Link } from '@tanstack/react-router'
2
+ import type { ReactNode } from 'react'
3
+ import type { PublicSoul } from '../lib/publicUser'
4
+
5
+ type SoulCardProps = {
6
+ soul: PublicSoul
7
+ summaryFallback: string
8
+ meta: ReactNode
9
+ }
10
+
11
+ export function SoulCard({ soul, summaryFallback, meta }: SoulCardProps) {
12
+ return (
13
+ <Link to="/souls/$slug" params={{ slug: soul.slug }} className="card skill-card">
14
+ <h3 className="skill-card-title">{soul.displayName}</h3>
15
+ <p className="skill-card-summary">{soul.summary ?? summaryFallback}</p>
16
+ <div className="skill-card-footer">{meta}</div>
17
+ </Link>
18
+ )
19
+ }