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,5 @@
1
+ export type { ArkValidator } from './ark.js'
2
+ export { formatArkErrors, parseArk } from './ark.js'
3
+ export { ApiRoutes, LegacyApiRoutes } from './routes.js'
4
+ export * from './schemas.js'
5
+ export * from './textFiles.js'
@@ -0,0 +1,22 @@
1
+ export const LegacyApiRoutes = {
2
+ download: '/api/download',
3
+ search: '/api/search',
4
+ skill: '/api/skill',
5
+ skillResolve: '/api/skill/resolve',
6
+ cliWhoami: '/api/cli/whoami',
7
+ cliUploadUrl: '/api/cli/upload-url',
8
+ cliPublish: '/api/cli/publish',
9
+ cliTelemetrySync: '/api/cli/telemetry/sync',
10
+ cliSkillDelete: '/api/cli/skill/delete',
11
+ cliSkillUndelete: '/api/cli/skill/undelete',
12
+ } as const
13
+
14
+ export const ApiRoutes = {
15
+ search: '/api/v1/search',
16
+ resolve: '/api/v1/resolve',
17
+ download: '/api/v1/download',
18
+ skills: '/api/v1/skills',
19
+ stars: '/api/v1/stars',
20
+ souls: '/api/v1/souls',
21
+ whoami: '/api/v1/whoami',
22
+ } as const
@@ -0,0 +1,123 @@
1
+ /* @vitest-environment node */
2
+
3
+ import { describe, expect, it } from 'vitest'
4
+ import { parseArk } from './ark'
5
+ import {
6
+ ApiSearchResponseSchema,
7
+ CliPublishRequestSchema,
8
+ CliSkillDeleteRequestSchema,
9
+ LockfileSchema,
10
+ WellKnownConfigSchema,
11
+ } from './schemas'
12
+
13
+ describe('pilothub-schema', () => {
14
+ it('parses lockfile records', () => {
15
+ const lock = parseArk(
16
+ LockfileSchema,
17
+ { version: 1, skills: { demo: { version: '1.0.0', installedAt: 123 } } },
18
+ 'Lockfile',
19
+ )
20
+ expect(lock.skills.demo?.version).toBe('1.0.0')
21
+ })
22
+
23
+ it('allows publish payload without tags', () => {
24
+ const payload = parseArk(
25
+ CliPublishRequestSchema,
26
+ {
27
+ slug: 'demo',
28
+ displayName: 'Demo',
29
+ version: '1.0.0',
30
+ changelog: '',
31
+ files: [{ path: 'SKILL.md', size: 1, storageId: 's', sha256: 'x' }],
32
+ },
33
+ 'Publish payload',
34
+ )
35
+ expect(payload.tags).toBeUndefined()
36
+ expect(payload.files[0]?.path).toBe('SKILL.md')
37
+ })
38
+
39
+ it('accepts publish payload with github source', () => {
40
+ const payload = parseArk(
41
+ CliPublishRequestSchema,
42
+ {
43
+ slug: 'demo',
44
+ displayName: 'Demo',
45
+ version: '1.0.0',
46
+ changelog: '',
47
+ source: {
48
+ kind: 'github',
49
+ url: 'https://github.com/example/demo',
50
+ repo: 'example/demo',
51
+ ref: 'main',
52
+ commit: 'abc123',
53
+ path: '.',
54
+ importedAt: 123,
55
+ },
56
+ files: [{ path: 'SKILL.md', size: 1, storageId: 's', sha256: 'x' }],
57
+ },
58
+ 'Publish payload',
59
+ )
60
+ expect(payload.source?.repo).toBe('example/demo')
61
+ })
62
+
63
+ it('parses well-known config', () => {
64
+ expect(
65
+ parseArk(WellKnownConfigSchema, { registry: 'https://example.convex.site' }, 'WellKnown'),
66
+ ).toEqual({ registry: 'https://example.convex.site' })
67
+
68
+ expect(
69
+ parseArk(
70
+ WellKnownConfigSchema,
71
+ { registry: 'https://example.convex.site', authBase: 'https://pilothub.com' },
72
+ 'WellKnown',
73
+ ),
74
+ ).toEqual({ registry: 'https://example.convex.site', authBase: 'https://pilothub.com' })
75
+
76
+ expect(
77
+ parseArk(
78
+ WellKnownConfigSchema,
79
+ { apiBase: 'https://example.convex.site', minCliVersion: '0.1.0' },
80
+ 'WellKnown',
81
+ ),
82
+ ).toEqual({ apiBase: 'https://example.convex.site', minCliVersion: '0.1.0' })
83
+
84
+ const combined = parseArk(
85
+ WellKnownConfigSchema,
86
+ {
87
+ apiBase: 'https://pilothub.com',
88
+ registry: 'https://pilothub.com',
89
+ authBase: 'https://pilothub.com',
90
+ },
91
+ 'WellKnown',
92
+ ) as unknown as Record<string, unknown>
93
+ expect(combined.apiBase).toBe('https://pilothub.com')
94
+ expect(combined.registry).toBe('https://pilothub.com')
95
+ })
96
+
97
+ it('throws labeled errors', () => {
98
+ expect(() => parseArk(LockfileSchema, null, 'Lockfile')).toThrow(/Lockfile:/)
99
+ })
100
+
101
+ it('parses search results arrays', () => {
102
+ expect(parseArk(ApiSearchResponseSchema, { results: [] }, 'Search')).toEqual({ results: [] })
103
+
104
+ const parsed = parseArk(
105
+ ApiSearchResponseSchema,
106
+ {
107
+ results: [
108
+ { slug: 'a', displayName: 'A', version: '1.0.0', score: 0.9 },
109
+ { slug: 'b', displayName: 'B', version: null, score: 0.1 },
110
+ ],
111
+ },
112
+ 'Search',
113
+ )
114
+ expect(parsed.results).toHaveLength(2)
115
+ expect(parsed.results[0]?.slug).toBe('a')
116
+ })
117
+
118
+ it('parses delete request payload', () => {
119
+ expect(parseArk(CliSkillDeleteRequestSchema, { slug: 'demo' }, 'Delete')).toEqual({
120
+ slug: 'demo',
121
+ })
122
+ })
123
+ })
@@ -0,0 +1,287 @@
1
+ import { type inferred, type } from 'arktype'
2
+
3
+ export const GlobalConfigSchema = type({
4
+ registry: 'string',
5
+ token: 'string?',
6
+ })
7
+ export type GlobalConfig = (typeof GlobalConfigSchema)[inferred]
8
+
9
+ export const WellKnownConfigSchema = type({
10
+ apiBase: 'string',
11
+ authBase: 'string?',
12
+ minCliVersion: 'string?',
13
+ }).or({
14
+ registry: 'string',
15
+ authBase: 'string?',
16
+ minCliVersion: 'string?',
17
+ })
18
+ export type WellKnownConfig = (typeof WellKnownConfigSchema)[inferred]
19
+
20
+ export const LockfileSchema = type({
21
+ version: '1',
22
+ skills: {
23
+ '[string]': {
24
+ version: 'string|null',
25
+ installedAt: 'number',
26
+ },
27
+ },
28
+ })
29
+ export type Lockfile = (typeof LockfileSchema)[inferred]
30
+
31
+ export const ApiCliWhoamiResponseSchema = type({
32
+ user: {
33
+ handle: 'string|null',
34
+ },
35
+ })
36
+
37
+ export const ApiSearchResponseSchema = type({
38
+ results: type({
39
+ slug: 'string?',
40
+ displayName: 'string?',
41
+ version: 'string|null?',
42
+ score: 'number',
43
+ }).array(),
44
+ })
45
+
46
+ export const ApiSkillMetaResponseSchema = type({
47
+ latestVersion: type({
48
+ version: 'string',
49
+ }).optional(),
50
+ skill: 'unknown|null?',
51
+ })
52
+
53
+ export const ApiCliUploadUrlResponseSchema = type({
54
+ uploadUrl: 'string',
55
+ })
56
+
57
+ export const ApiUploadFileResponseSchema = type({
58
+ storageId: 'string',
59
+ })
60
+
61
+ export const CliPublishFileSchema = type({
62
+ path: 'string',
63
+ size: 'number',
64
+ storageId: 'string',
65
+ sha256: 'string',
66
+ contentType: 'string?',
67
+ })
68
+ export type CliPublishFile = (typeof CliPublishFileSchema)[inferred]
69
+
70
+ export const PublishSourceSchema = type({
71
+ kind: '"github"',
72
+ url: 'string',
73
+ repo: 'string',
74
+ ref: 'string',
75
+ commit: 'string',
76
+ path: 'string',
77
+ importedAt: 'number',
78
+ })
79
+
80
+ export const CliPublishRequestSchema = type({
81
+ slug: 'string',
82
+ displayName: 'string',
83
+ version: 'string',
84
+ changelog: 'string',
85
+ tags: 'string[]?',
86
+ source: PublishSourceSchema.optional(),
87
+ forkOf: type({
88
+ slug: 'string',
89
+ version: 'string?',
90
+ }).optional(),
91
+ files: CliPublishFileSchema.array(),
92
+ })
93
+ export type CliPublishRequest = (typeof CliPublishRequestSchema)[inferred]
94
+
95
+ export const ApiCliPublishResponseSchema = type({
96
+ ok: 'true',
97
+ skillId: 'string',
98
+ versionId: 'string',
99
+ })
100
+
101
+ export const CliSkillDeleteRequestSchema = type({
102
+ slug: 'string',
103
+ })
104
+ export type CliSkillDeleteRequest = (typeof CliSkillDeleteRequestSchema)[inferred]
105
+
106
+ export const ApiCliSkillDeleteResponseSchema = type({
107
+ ok: 'true',
108
+ })
109
+
110
+ export const ApiSkillResolveResponseSchema = type({
111
+ match: type({ version: 'string' }).or('null'),
112
+ latestVersion: type({ version: 'string' }).or('null'),
113
+ })
114
+
115
+ export const CliTelemetrySyncRequestSchema = type({
116
+ roots: type({
117
+ rootId: 'string',
118
+ label: 'string',
119
+ skills: type({
120
+ slug: 'string',
121
+ version: 'string|null?',
122
+ }).array(),
123
+ }).array(),
124
+ })
125
+ export type CliTelemetrySyncRequest = (typeof CliTelemetrySyncRequestSchema)[inferred]
126
+
127
+ export const ApiCliTelemetrySyncResponseSchema = type({
128
+ ok: 'true',
129
+ })
130
+
131
+ export const ApiV1WhoamiResponseSchema = type({
132
+ user: {
133
+ handle: 'string|null',
134
+ displayName: 'string|null?',
135
+ image: 'string|null?',
136
+ },
137
+ })
138
+
139
+ export const ApiV1SearchResponseSchema = type({
140
+ results: type({
141
+ slug: 'string?',
142
+ displayName: 'string?',
143
+ summary: 'string|null?',
144
+ version: 'string|null?',
145
+ score: 'number',
146
+ updatedAt: 'number?',
147
+ }).array(),
148
+ })
149
+
150
+ export const ApiV1SkillListResponseSchema = type({
151
+ items: type({
152
+ slug: 'string',
153
+ displayName: 'string',
154
+ summary: 'string|null?',
155
+ tags: 'unknown',
156
+ stats: 'unknown',
157
+ createdAt: 'number',
158
+ updatedAt: 'number',
159
+ latestVersion: type({
160
+ version: 'string',
161
+ createdAt: 'number',
162
+ changelog: 'string',
163
+ }).optional(),
164
+ }).array(),
165
+ nextCursor: 'string|null',
166
+ })
167
+
168
+ export const ApiV1SkillResponseSchema = type({
169
+ skill: type({
170
+ slug: 'string',
171
+ displayName: 'string',
172
+ summary: 'string|null?',
173
+ tags: 'unknown',
174
+ stats: 'unknown',
175
+ createdAt: 'number',
176
+ updatedAt: 'number',
177
+ }).or('null'),
178
+ latestVersion: type({
179
+ version: 'string',
180
+ createdAt: 'number',
181
+ changelog: 'string',
182
+ }).or('null'),
183
+ owner: type({
184
+ handle: 'string|null',
185
+ displayName: 'string|null?',
186
+ image: 'string|null?',
187
+ }).or('null'),
188
+ })
189
+
190
+ export const ApiV1SkillVersionListResponseSchema = type({
191
+ items: type({
192
+ version: 'string',
193
+ createdAt: 'number',
194
+ changelog: 'string',
195
+ changelogSource: '"auto"|"user"|null?',
196
+ }).array(),
197
+ nextCursor: 'string|null',
198
+ })
199
+
200
+ export const ApiV1SkillVersionResponseSchema = type({
201
+ version: type({
202
+ version: 'string',
203
+ createdAt: 'number',
204
+ changelog: 'string',
205
+ changelogSource: '"auto"|"user"|null?',
206
+ files: 'unknown?',
207
+ }).or('null'),
208
+ skill: type({
209
+ slug: 'string',
210
+ displayName: 'string',
211
+ }).or('null'),
212
+ })
213
+
214
+ export const ApiV1SkillResolveResponseSchema = type({
215
+ match: type({ version: 'string' }).or('null'),
216
+ latestVersion: type({ version: 'string' }).or('null'),
217
+ })
218
+
219
+ export const ApiV1PublishResponseSchema = type({
220
+ ok: 'true',
221
+ skillId: 'string',
222
+ versionId: 'string',
223
+ })
224
+
225
+ export const ApiV1DeleteResponseSchema = type({
226
+ ok: 'true',
227
+ })
228
+
229
+ export const ApiV1StarResponseSchema = type({
230
+ ok: 'true',
231
+ starred: 'boolean',
232
+ alreadyStarred: 'boolean',
233
+ })
234
+
235
+ export const ApiV1UnstarResponseSchema = type({
236
+ ok: 'true',
237
+ unstarred: 'boolean',
238
+ alreadyUnstarred: 'boolean',
239
+ })
240
+
241
+ export const SkillInstallSpecSchema = type({
242
+ id: 'string?',
243
+ kind: '"brew"|"node"|"go"|"uv"',
244
+ label: 'string?',
245
+ bins: 'string[]?',
246
+ formula: 'string?',
247
+ tap: 'string?',
248
+ package: 'string?',
249
+ module: 'string?',
250
+ })
251
+ export type SkillInstallSpec = (typeof SkillInstallSpecSchema)[inferred]
252
+
253
+ export const NixPluginSpecSchema = type({
254
+ plugin: 'string',
255
+ systems: 'string[]?',
256
+ })
257
+ export type NixPluginSpec = (typeof NixPluginSpecSchema)[inferred]
258
+
259
+ export const PilotbotConfigSpecSchema = type({
260
+ requiredEnv: 'string[]?',
261
+ stateDirs: 'string[]?',
262
+ example: 'string?',
263
+ })
264
+ export type PilotbotConfigSpec = (typeof PilotbotConfigSpecSchema)[inferred]
265
+
266
+ export const PilotbotRequiresSchema = type({
267
+ bins: 'string[]?',
268
+ anyBins: 'string[]?',
269
+ env: 'string[]?',
270
+ config: 'string[]?',
271
+ })
272
+ export type PilotbotRequires = (typeof PilotbotRequiresSchema)[inferred]
273
+
274
+ export const PilotbotSkillMetadataSchema = type({
275
+ always: 'boolean?',
276
+ skillKey: 'string?',
277
+ primaryEnv: 'string?',
278
+ emoji: 'string?',
279
+ homepage: 'string?',
280
+ os: 'string[]?',
281
+ cliHelp: 'string?',
282
+ requires: PilotbotRequiresSchema.optional(),
283
+ install: SkillInstallSpecSchema.array().optional(),
284
+ nix: NixPluginSpecSchema.optional(),
285
+ config: PilotbotConfigSpecSchema.optional(),
286
+ })
287
+ export type PilotbotSkillMetadata = (typeof PilotbotSkillMetadataSchema)[inferred]
@@ -0,0 +1,23 @@
1
+ /* @vitest-environment node */
2
+
3
+ import { describe, expect, it } from 'vitest'
4
+ import * as schema from '.'
5
+ import { isTextContentType, TEXT_FILE_EXTENSION_SET } from './textFiles'
6
+
7
+ describe('pilothub-schema textFiles', () => {
8
+ it('exports text-file extension set', () => {
9
+ expect(TEXT_FILE_EXTENSION_SET.has('md')).toBe(true)
10
+ expect(TEXT_FILE_EXTENSION_SET.has('exe')).toBe(false)
11
+ })
12
+
13
+ it('detects text content types with parameters', () => {
14
+ expect(isTextContentType('text/plain; charset=utf-8')).toBe(true)
15
+ expect(isTextContentType('application/json; charset=utf-8')).toBe(true)
16
+ expect(isTextContentType('application/octet-stream')).toBe(false)
17
+ })
18
+
19
+ it('re-exports helpers from index', () => {
20
+ expect(typeof schema.isTextContentType).toBe('function')
21
+ expect(schema.isTextContentType('application/markdown')).toBe(true)
22
+ })
23
+ })
@@ -0,0 +1,66 @@
1
+ const RAW_TEXT_FILE_EXTENSIONS = [
2
+ 'md',
3
+ 'mdx',
4
+ 'txt',
5
+ 'json',
6
+ 'json5',
7
+ 'yaml',
8
+ 'yml',
9
+ 'toml',
10
+ 'js',
11
+ 'cjs',
12
+ 'mjs',
13
+ 'ts',
14
+ 'tsx',
15
+ 'jsx',
16
+ 'py',
17
+ 'sh',
18
+ 'rb',
19
+ 'go',
20
+ 'rs',
21
+ 'swift',
22
+ 'kt',
23
+ 'java',
24
+ 'cs',
25
+ 'cpp',
26
+ 'c',
27
+ 'h',
28
+ 'hpp',
29
+ 'sql',
30
+ 'csv',
31
+ 'ini',
32
+ 'cfg',
33
+ 'env',
34
+ 'xml',
35
+ 'html',
36
+ 'css',
37
+ 'scss',
38
+ 'sass',
39
+ 'svg',
40
+ ] as const
41
+
42
+ export const TEXT_FILE_EXTENSIONS = RAW_TEXT_FILE_EXTENSIONS
43
+ export const TEXT_FILE_EXTENSION_SET = new Set<string>(TEXT_FILE_EXTENSIONS)
44
+
45
+ const RAW_TEXT_CONTENT_TYPES = [
46
+ 'application/json',
47
+ 'application/xml',
48
+ 'application/yaml',
49
+ 'application/x-yaml',
50
+ 'application/toml',
51
+ 'application/javascript',
52
+ 'application/typescript',
53
+ 'application/markdown',
54
+ 'image/svg+xml',
55
+ ] as const
56
+
57
+ export const TEXT_CONTENT_TYPES = RAW_TEXT_CONTENT_TYPES
58
+ export const TEXT_CONTENT_TYPE_SET = new Set<string>(TEXT_CONTENT_TYPES)
59
+
60
+ export function isTextContentType(contentType: string) {
61
+ if (!contentType) return false
62
+ const normalized = contentType.split(';', 1)[0]?.trim().toLowerCase() ?? ''
63
+ if (!normalized) return false
64
+ if (normalized.startsWith('text/')) return true
65
+ return TEXT_CONTENT_TYPE_SET.has(normalized)
66
+ }
@@ -0,0 +1,15 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "ES2022",
5
+ "moduleResolution": "Bundler",
6
+ "outDir": "dist",
7
+ "rootDir": "src",
8
+ "strict": true,
9
+ "declaration": true,
10
+ "sourceMap": true,
11
+ "skipLibCheck": true
12
+ },
13
+ "include": ["src/**/*.ts"],
14
+ "exclude": ["src/**/*.test.ts"]
15
+ }
package/pilothub ADDED
@@ -0,0 +1,46 @@
1
+ #!/usr/bin/env bun
2
+ import { existsSync } from 'node:fs'
3
+ import { stat } from 'node:fs/promises'
4
+ import { fileURLToPath } from 'node:url'
5
+
6
+ const distCliUrl = new URL('./packages/pilothub/dist/cli.js', import.meta.url)
7
+ const distCliPath = fileURLToPath(distCliUrl)
8
+ const srcRootPath = fileURLToPath(new URL('./packages/pilothub/src/', import.meta.url))
9
+
10
+ const shouldBuild = await (async () => {
11
+ if (!existsSync(distCliPath)) return true
12
+ try {
13
+ const dist = await stat(distCliPath)
14
+ const latestSrcMtime = await getLatestMtime(srcRootPath)
15
+ return latestSrcMtime > dist.mtimeMs
16
+ } catch {
17
+ return true
18
+ }
19
+ })()
20
+
21
+ if (shouldBuild) {
22
+ const proc = Bun.spawn(['bunx', 'tsc', '-p', 'packages/pilothub/tsconfig.json'], {
23
+ stdin: 'inherit',
24
+ stdout: 'inherit',
25
+ stderr: 'inherit',
26
+ })
27
+ const code = await proc.exited
28
+ if (code !== 0) process.exit(code)
29
+ }
30
+
31
+ await import(distCliUrl.href)
32
+
33
+ async function getLatestMtime(root: string) {
34
+ let latest = 0
35
+ const glob = new Bun.Glob('**/*.ts')
36
+ for await (const rel of glob.scan({ cwd: root, onlyFiles: true })) {
37
+ const path = `${root}${root.endsWith('/') ? '' : '/'}${rel}`
38
+ try {
39
+ const entry = await stat(path)
40
+ latest = Math.max(latest, entry.mtimeMs)
41
+ } catch {
42
+ // ignore
43
+ }
44
+ }
45
+ return latest
46
+ }
@@ -0,0 +1,33 @@
1
+ import { defineConfig, devices } from '@playwright/test'
2
+
3
+ const port = Number(process.env.PLAYWRIGHT_PORT || 4173)
4
+ const baseURL = process.env.PLAYWRIGHT_BASE_URL || `http://127.0.0.1:${port}`
5
+
6
+ export default defineConfig({
7
+ testDir: './e2e',
8
+ testMatch: /.*\.pw\.test\.ts/,
9
+ timeout: 60_000,
10
+ expect: { timeout: 10_000 },
11
+ fullyParallel: true,
12
+ retries: process.env.CI ? 2 : 0,
13
+ reporter: process.env.CI ? [['list'], ['html', { open: 'never' }]] : [['list']],
14
+ use: {
15
+ baseURL,
16
+ trace: 'retain-on-failure',
17
+ },
18
+ webServer: process.env.PLAYWRIGHT_BASE_URL
19
+ ? undefined
20
+ : {
21
+ command: 'bun run preview -- --host 127.0.0.1 --port 4173',
22
+ url: baseURL,
23
+ reuseExistingServer: !process.env.CI,
24
+ stdout: 'ignore',
25
+ stderr: 'pipe',
26
+ },
27
+ projects: [
28
+ {
29
+ name: 'chromium',
30
+ use: { ...devices['Desktop Chrome'] },
31
+ },
32
+ ],
33
+ })
@@ -0,0 +1,6 @@
1
+ {
2
+ "apiBase": "https://pilothub.com",
3
+ "authBase": "https://pilothub.com",
4
+ "minCliVersion": "0.1.0",
5
+ "registry": "https://pilothub.com"
6
+ }