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.
- package/.env.local.example +19 -0
- package/.github/workflows/ci.yml +40 -0
- package/.oxlintrc.json +3 -0
- package/AGENTS.md +45 -0
- package/CHANGELOG.md +138 -0
- package/DEPRECATIONS.md +7 -0
- package/LICENSE +21 -0
- package/README.md +150 -0
- package/biome.json +41 -0
- package/convex/_generated/api.d.ts +153 -0
- package/convex/_generated/api.js +23 -0
- package/convex/_generated/dataModel.d.ts +60 -0
- package/convex/_generated/server.d.ts +143 -0
- package/convex/_generated/server.js +93 -0
- package/convex/auth.config.ts +8 -0
- package/convex/auth.ts +19 -0
- package/convex/comments.ts +88 -0
- package/convex/crons.ts +34 -0
- package/convex/devSeed.ts +459 -0
- package/convex/devSeedExtra.ts +541 -0
- package/convex/downloads.ts +78 -0
- package/convex/githubBackups.ts +170 -0
- package/convex/githubBackupsNode.ts +183 -0
- package/convex/githubImport.ts +317 -0
- package/convex/githubSoulBackups.ts +170 -0
- package/convex/githubSoulBackupsNode.ts +186 -0
- package/convex/http.ts +194 -0
- package/convex/httpApi.handlers.test.ts +488 -0
- package/convex/httpApi.test.ts +70 -0
- package/convex/httpApi.ts +305 -0
- package/convex/httpApiV1.handlers.test.ts +584 -0
- package/convex/httpApiV1.ts +1172 -0
- package/convex/leaderboards.ts +39 -0
- package/convex/lib/access.ts +36 -0
- package/convex/lib/apiTokenAuth.ts +36 -0
- package/convex/lib/badges.ts +50 -0
- package/convex/lib/changelog.test.ts +34 -0
- package/convex/lib/changelog.ts +278 -0
- package/convex/lib/embeddings.ts +38 -0
- package/convex/lib/githubBackup.ts +443 -0
- package/convex/lib/githubImport.test.ts +247 -0
- package/convex/lib/githubImport.ts +425 -0
- package/convex/lib/githubSoulBackup.ts +443 -0
- package/convex/lib/leaderboards.ts +103 -0
- package/convex/lib/moderation.ts +42 -0
- package/convex/lib/public.ts +89 -0
- package/convex/lib/searchText.test.ts +46 -0
- package/convex/lib/searchText.ts +27 -0
- package/convex/lib/skillBackfill.test.ts +34 -0
- package/convex/lib/skillBackfill.ts +67 -0
- package/convex/lib/skillPublish.test.ts +28 -0
- package/convex/lib/skillPublish.ts +284 -0
- package/convex/lib/skillStats.ts +80 -0
- package/convex/lib/skills.test.ts +197 -0
- package/convex/lib/skills.ts +273 -0
- package/convex/lib/soulChangelog.ts +273 -0
- package/convex/lib/soulPublish.ts +236 -0
- package/convex/lib/tokens.test.ts +33 -0
- package/convex/lib/tokens.ts +51 -0
- package/convex/lib/webhooks.test.ts +91 -0
- package/convex/lib/webhooks.ts +112 -0
- package/convex/maintenance.test.ts +270 -0
- package/convex/maintenance.ts +840 -0
- package/convex/rateLimits.ts +50 -0
- package/convex/schema.ts +472 -0
- package/convex/search.test.ts +12 -0
- package/convex/search.ts +254 -0
- package/convex/seed.test.ts +37 -0
- package/convex/seed.ts +254 -0
- package/convex/seedSouls.ts +111 -0
- package/convex/skillStatEvents.ts +568 -0
- package/convex/skills.ts +1606 -0
- package/convex/soulComments.ts +88 -0
- package/convex/soulDownloads.ts +14 -0
- package/convex/soulStars.ts +71 -0
- package/convex/souls.ts +570 -0
- package/convex/stars.ts +108 -0
- package/convex/statsMaintenance.ts +205 -0
- package/convex/telemetry.ts +434 -0
- package/convex/tokens.ts +88 -0
- package/convex/tsconfig.json +7 -0
- package/convex/uploads.ts +20 -0
- package/convex/users.ts +122 -0
- package/convex/webhooks.ts +50 -0
- package/convex.json +3 -0
- package/docs/README.md +32 -0
- package/docs/api.md +51 -0
- package/docs/architecture.md +61 -0
- package/docs/auth.md +54 -0
- package/docs/cli.md +117 -0
- package/docs/deploy.md +78 -0
- package/docs/diffing.md +84 -0
- package/docs/github-import.md +171 -0
- package/docs/http-api.md +187 -0
- package/docs/manual-testing.md +64 -0
- package/docs/mintlify.md +43 -0
- package/docs/quickstart.md +120 -0
- package/docs/skill-format.md +58 -0
- package/docs/soul-format.md +37 -0
- package/docs/spec.md +177 -0
- package/docs/telemetry.md +91 -0
- package/docs/troubleshooting.md +49 -0
- package/docs/webhook.md +51 -0
- package/e2e/menu-smoke.pw.test.ts +49 -0
- package/e2e/pilothub.e2e.test.ts +494 -0
- package/e2e/search-exact.pw.test.ts +97 -0
- package/package.json +84 -0
- package/packages/pilothub/LICENSE +22 -0
- package/packages/pilothub/README.md +57 -0
- package/packages/pilothub/bin/pilothub.js +2 -0
- package/packages/pilothub/package.json +41 -0
- package/packages/pilothub/src/browserAuth.test.ts +96 -0
- package/packages/pilothub/src/browserAuth.ts +174 -0
- package/packages/pilothub/src/cli/buildInfo.ts +94 -0
- package/packages/pilothub/src/cli/commands/auth.ts +97 -0
- package/packages/pilothub/src/cli/commands/delete.test.ts +73 -0
- package/packages/pilothub/src/cli/commands/delete.ts +83 -0
- package/packages/pilothub/src/cli/commands/publish.test.ts +122 -0
- package/packages/pilothub/src/cli/commands/publish.ts +108 -0
- package/packages/pilothub/src/cli/commands/skills.test.ts +191 -0
- package/packages/pilothub/src/cli/commands/skills.ts +380 -0
- package/packages/pilothub/src/cli/commands/star.ts +46 -0
- package/packages/pilothub/src/cli/commands/sync.test.ts +310 -0
- package/packages/pilothub/src/cli/commands/sync.ts +200 -0
- package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +26 -0
- package/packages/pilothub/src/cli/commands/syncHelpers.ts +427 -0
- package/packages/pilothub/src/cli/commands/syncTypes.ts +27 -0
- package/packages/pilothub/src/cli/commands/unstar.ts +48 -0
- package/packages/pilothub/src/cli/helpStyle.ts +45 -0
- package/packages/pilothub/src/cli/pilotbotConfig.test.ts +159 -0
- package/packages/pilothub/src/cli/pilotbotConfig.ts +147 -0
- package/packages/pilothub/src/cli/registry.test.ts +63 -0
- package/packages/pilothub/src/cli/registry.ts +43 -0
- package/packages/pilothub/src/cli/scanSkills.test.ts +64 -0
- package/packages/pilothub/src/cli/scanSkills.ts +84 -0
- package/packages/pilothub/src/cli/slug.ts +16 -0
- package/packages/pilothub/src/cli/types.ts +12 -0
- package/packages/pilothub/src/cli/ui.ts +75 -0
- package/packages/pilothub/src/cli.ts +311 -0
- package/packages/pilothub/src/config.ts +36 -0
- package/packages/pilothub/src/discovery.test.ts +75 -0
- package/packages/pilothub/src/discovery.ts +19 -0
- package/packages/pilothub/src/http.test.ts +156 -0
- package/packages/pilothub/src/http.ts +301 -0
- package/packages/pilothub/src/schema/ark.ts +29 -0
- package/packages/pilothub/src/schema/index.ts +5 -0
- package/packages/pilothub/src/schema/routes.ts +22 -0
- package/packages/pilothub/src/schema/schemas.ts +260 -0
- package/packages/pilothub/src/schema/textFiles.test.ts +23 -0
- package/packages/pilothub/src/schema/textFiles.ts +66 -0
- package/packages/pilothub/src/skills.test.ts +191 -0
- package/packages/pilothub/src/skills.ts +172 -0
- package/packages/pilothub/src/types.ts +10 -0
- package/packages/pilothub/tsconfig.json +14 -0
- package/packages/schema/README.md +3 -0
- package/packages/schema/dist/ark.d.ts +4 -0
- package/packages/schema/dist/ark.js +26 -0
- package/packages/schema/dist/ark.js.map +1 -0
- package/packages/schema/dist/index.d.ts +5 -0
- package/packages/schema/dist/index.js +5 -0
- package/packages/schema/dist/index.js.map +1 -0
- package/packages/schema/dist/routes.d.ts +21 -0
- package/packages/schema/dist/routes.js +22 -0
- package/packages/schema/dist/routes.js.map +1 -0
- package/packages/schema/dist/schemas.d.ts +297 -0
- package/packages/schema/dist/schemas.js +243 -0
- package/packages/schema/dist/schemas.js.map +1 -0
- package/packages/schema/dist/textFiles.d.ts +5 -0
- package/packages/schema/dist/textFiles.js +66 -0
- package/packages/schema/dist/textFiles.js.map +1 -0
- package/packages/schema/package.json +26 -0
- package/packages/schema/src/ark.ts +29 -0
- package/packages/schema/src/index.ts +5 -0
- package/packages/schema/src/routes.ts +22 -0
- package/packages/schema/src/schemas.test.ts +123 -0
- package/packages/schema/src/schemas.ts +287 -0
- package/packages/schema/src/textFiles.test.ts +23 -0
- package/packages/schema/src/textFiles.ts +66 -0
- package/packages/schema/tsconfig.json +15 -0
- package/pilothub +46 -0
- package/playwright.config.ts +33 -0
- package/public/.well-known/pilothub.json +6 -0
- package/public/api/v1/openapi.json +379 -0
- package/public/favicon.ico +0 -0
- package/public/logo192.png +0 -0
- package/public/logo512.png +0 -0
- package/public/manifest.json +25 -0
- package/public/og.png +0 -0
- package/public/og.svg +98 -0
- package/public/pilot-logo.png +0 -0
- package/public/pilot-mark.png +0 -0
- package/public/robots.txt +3 -0
- package/public/tanstack-circle-logo.png +0 -0
- package/public/tanstack-word-logo-white.svg +1 -0
- package/scripts/check-peer-deps.ts +56 -0
- package/scripts/docs-list.ts +148 -0
- package/scripts/run-playwright-local.sh +14 -0
- package/server/og/fetchSkillOgMeta.ts +27 -0
- package/server/og/fetchSoulOgMeta.ts +27 -0
- package/server/og/ogAssets.ts +80 -0
- package/server/og/skillOgSvg.test.ts +59 -0
- package/server/og/skillOgSvg.ts +258 -0
- package/server/og/soulOgSvg.ts +209 -0
- package/server/routes/og/skill.png.ts +103 -0
- package/server/routes/og/soul.png.ts +111 -0
- package/src/__tests__/skill-detail-page.test.tsx +86 -0
- package/src/__tests__/skills-index.test.tsx +145 -0
- package/src/__tests__/upload.route.test.tsx +228 -0
- package/src/components/AppProviders.tsx +19 -0
- package/src/components/ClientOnly.tsx +18 -0
- package/src/components/Footer.tsx +29 -0
- package/src/components/Header.tsx +295 -0
- package/src/components/InstallSwitcher.tsx +53 -0
- package/src/components/SkillCard.tsx +36 -0
- package/src/components/SkillDetailPage.tsx +817 -0
- package/src/components/SkillDiffCard.tsx +485 -0
- package/src/components/SoulCard.tsx +19 -0
- package/src/components/SoulDetailPage.tsx +263 -0
- package/src/components/UserBootstrap.tsx +18 -0
- package/src/components/ui/dropdown-menu.tsx +67 -0
- package/src/components/ui/toggle-group.tsx +35 -0
- package/src/convex/client.ts +3 -0
- package/src/lib/badges.ts +29 -0
- package/src/lib/diffing.test.ts +163 -0
- package/src/lib/diffing.ts +106 -0
- package/src/lib/gravatar.test.ts +9 -0
- package/src/lib/gravatar.ts +158 -0
- package/src/lib/og.test.ts +142 -0
- package/src/lib/og.ts +156 -0
- package/src/lib/publicUser.ts +39 -0
- package/src/lib/roles.ts +19 -0
- package/src/lib/site.test.ts +130 -0
- package/src/lib/site.ts +84 -0
- package/src/lib/theme-transition.test.ts +134 -0
- package/src/lib/theme-transition.ts +134 -0
- package/src/lib/theme.test.tsx +88 -0
- package/src/lib/theme.ts +43 -0
- package/src/lib/uploadFiles.jsdom.test.ts +33 -0
- package/src/lib/uploadFiles.test.ts +123 -0
- package/src/lib/uploadFiles.ts +245 -0
- package/src/lib/uploadUtils.test.ts +78 -0
- package/src/lib/uploadUtils.ts +93 -0
- package/src/lib/useAuthStatus.ts +12 -0
- package/src/lib/utils.test.ts +9 -0
- package/src/lib/utils.ts +6 -0
- package/src/logo.svg +12 -0
- package/src/routeTree.gen.ts +345 -0
- package/src/router.tsx +17 -0
- package/src/routes/$owner/$slug.tsx +55 -0
- package/src/routes/__root.tsx +136 -0
- package/src/routes/admin.tsx +11 -0
- package/src/routes/cli/auth.tsx +168 -0
- package/src/routes/dashboard.tsx +97 -0
- package/src/routes/import.tsx +415 -0
- package/src/routes/index.tsx +252 -0
- package/src/routes/management.tsx +529 -0
- package/src/routes/settings.tsx +203 -0
- package/src/routes/skills/index.tsx +422 -0
- package/src/routes/souls/$slug.tsx +55 -0
- package/src/routes/souls/index.tsx +243 -0
- package/src/routes/stars.tsx +68 -0
- package/src/routes/u/$handle.tsx +307 -0
- package/src/routes/upload/utils.ts +81 -0
- package/src/routes/upload.tsx +499 -0
- package/src/styles.css +2718 -0
- package/tsconfig.json +24 -0
- package/tsconfig.oxlint.json +16 -0
- package/vercel.json +8 -0
- package/vite.config.ts +48 -0
- package/vitest.config.ts +47 -0
- package/vitest.e2e.config.ts +11 -0
- package/vitest.setup.ts +1 -0
|
@@ -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
|
+
})
|