pilothub 0.0.1 → 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +1 -0
- package/README.md +36 -129
- package/dist/browserAuth.d.ts +20 -0
- package/dist/browserAuth.js +156 -0
- package/dist/browserAuth.js.map +1 -0
- package/dist/browserAuth.test.d.ts +1 -0
- package/dist/browserAuth.test.js +83 -0
- package/dist/browserAuth.test.js.map +1 -0
- package/dist/cli/buildInfo.d.ts +3 -0
- package/dist/cli/buildInfo.js +103 -0
- package/dist/cli/buildInfo.js.map +1 -0
- package/dist/cli/commands/auth.d.ts +9 -0
- package/dist/cli/commands/auth.js +75 -0
- package/dist/cli/commands/auth.js.map +1 -0
- package/dist/cli/commands/delete.d.ts +11 -0
- package/dist/cli/commands/delete.js +67 -0
- package/dist/cli/commands/delete.js.map +1 -0
- package/dist/cli/commands/delete.test.d.ts +1 -0
- package/dist/cli/commands/delete.test.js +52 -0
- package/dist/cli/commands/delete.test.js.map +1 -0
- package/dist/cli/commands/publish.d.ts +9 -0
- package/dist/cli/commands/publish.js +87 -0
- package/dist/cli/commands/publish.js.map +1 -0
- package/dist/cli/commands/publish.test.d.ts +1 -0
- package/dist/cli/commands/publish.test.js +104 -0
- package/dist/cli/commands/publish.test.js.map +1 -0
- package/dist/cli/commands/skills.d.ts +23 -0
- package/dist/cli/commands/skills.js +298 -0
- package/dist/cli/commands/skills.js.map +1 -0
- package/dist/cli/commands/skills.test.d.ts +1 -0
- package/dist/cli/commands/skills.test.js +156 -0
- package/dist/cli/commands/skills.test.js.map +1 -0
- package/dist/cli/commands/star.d.ts +8 -0
- package/dist/cli/commands/star.js +38 -0
- package/dist/cli/commands/star.js.map +1 -0
- package/dist/cli/commands/sync.d.ts +3 -0
- package/dist/cli/commands/sync.js +160 -0
- package/dist/cli/commands/sync.js.map +1 -0
- package/dist/cli/commands/sync.test.d.ts +1 -0
- package/dist/cli/commands/sync.test.js +277 -0
- package/dist/cli/commands/sync.test.js.map +1 -0
- package/dist/cli/commands/syncHelpers.d.ts +76 -0
- package/dist/cli/commands/syncHelpers.js +349 -0
- package/dist/cli/commands/syncHelpers.js.map +1 -0
- package/dist/cli/commands/syncHelpers.test.d.ts +1 -0
- package/dist/cli/commands/syncHelpers.test.js +22 -0
- package/dist/cli/commands/syncHelpers.test.js.map +1 -0
- package/dist/cli/commands/syncTypes.d.ts +24 -0
- package/dist/cli/commands/syncTypes.js +2 -0
- package/dist/cli/commands/syncTypes.js.map +1 -0
- package/dist/cli/commands/unstar.d.ts +8 -0
- package/dist/cli/commands/unstar.js +38 -0
- package/dist/cli/commands/unstar.js.map +1 -0
- package/dist/cli/helpStyle.d.ts +13 -0
- package/dist/cli/helpStyle.js +38 -0
- package/dist/cli/helpStyle.js.map +1 -0
- package/dist/cli/pilotbotConfig.d.ts +6 -0
- package/dist/cli/pilotbotConfig.js +110 -0
- package/dist/cli/pilotbotConfig.js.map +1 -0
- package/dist/cli/pilotbotConfig.test.d.ts +1 -0
- package/dist/cli/pilotbotConfig.test.js +133 -0
- package/dist/cli/pilotbotConfig.test.js.map +1 -0
- package/dist/cli/registry.d.ts +7 -0
- package/dist/cli/registry.js +42 -0
- package/dist/cli/registry.js.map +1 -0
- package/dist/cli/registry.test.d.ts +1 -0
- package/dist/cli/registry.test.js +48 -0
- package/dist/cli/registry.test.js.map +1 -0
- package/dist/cli/scanSkills.d.ts +7 -0
- package/dist/cli/scanSkills.js +75 -0
- package/dist/cli/scanSkills.js.map +1 -0
- package/dist/cli/scanSkills.test.d.ts +1 -0
- package/dist/cli/scanSkills.test.js +60 -0
- package/dist/cli/scanSkills.test.js.map +1 -0
- package/dist/cli/slug.d.ts +2 -0
- package/dist/cli/slug.js +16 -0
- package/dist/cli/slug.js.map +1 -0
- package/dist/cli/types.d.ts +15 -0
- package/dist/cli/types.js +2 -0
- package/dist/cli/types.js.map +1 -0
- package/dist/cli/ui.d.ts +7 -0
- package/dist/cli/ui.js +72 -0
- package/dist/cli/ui.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +268 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +4 -0
- package/dist/config.js +38 -0
- package/dist/config.js.map +1 -0
- package/dist/discovery.d.ts +5 -0
- package/dist/discovery.js +21 -0
- package/dist/discovery.js.map +1 -0
- package/dist/discovery.test.d.ts +1 -0
- package/dist/discovery.test.js +46 -0
- package/dist/discovery.test.js.map +1 -0
- package/dist/http.d.ts +32 -0
- package/dist/http.js +261 -0
- package/dist/http.js.map +1 -0
- package/dist/http.test.d.ts +1 -0
- package/dist/http.test.js +135 -0
- package/dist/http.test.js.map +1 -0
- package/dist/schema/ark.js.map +1 -0
- package/dist/schema/index.js.map +1 -0
- package/dist/schema/routes.js.map +1 -0
- package/{packages/schema/dist → dist/schema}/schemas.d.ts +0 -39
- package/{packages/schema/dist → dist/schema}/schemas.js +0 -22
- package/dist/schema/schemas.js.map +1 -0
- package/dist/schema/textFiles.js.map +1 -0
- package/dist/schema/textFiles.test.d.ts +1 -0
- package/dist/schema/textFiles.test.js +20 -0
- package/dist/schema/textFiles.test.js.map +1 -0
- package/dist/skills.d.ts +43 -0
- package/dist/skills.js +163 -0
- package/dist/skills.js.map +1 -0
- package/dist/skills.test.d.ts +1 -0
- package/dist/skills.test.js +144 -0
- package/dist/skills.test.js.map +1 -0
- package/dist/types.d.ts +7 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +27 -70
- package/.env.local.example +0 -19
- package/.github/workflows/ci.yml +0 -40
- package/.oxlintrc.json +0 -3
- package/AGENTS.md +0 -45
- package/CHANGELOG.md +0 -138
- package/DEPRECATIONS.md +0 -7
- package/biome.json +0 -41
- package/convex/_generated/api.d.ts +0 -153
- package/convex/_generated/api.js +0 -23
- package/convex/_generated/dataModel.d.ts +0 -60
- package/convex/_generated/server.d.ts +0 -143
- package/convex/_generated/server.js +0 -93
- package/convex/auth.config.ts +0 -8
- package/convex/auth.ts +0 -19
- package/convex/comments.ts +0 -88
- package/convex/crons.ts +0 -34
- package/convex/devSeed.ts +0 -459
- package/convex/devSeedExtra.ts +0 -541
- package/convex/downloads.ts +0 -78
- package/convex/githubBackups.ts +0 -170
- package/convex/githubBackupsNode.ts +0 -183
- package/convex/githubImport.ts +0 -317
- package/convex/githubSoulBackups.ts +0 -170
- package/convex/githubSoulBackupsNode.ts +0 -186
- package/convex/http.ts +0 -194
- package/convex/httpApi.handlers.test.ts +0 -488
- package/convex/httpApi.test.ts +0 -70
- package/convex/httpApi.ts +0 -305
- package/convex/httpApiV1.handlers.test.ts +0 -584
- package/convex/httpApiV1.ts +0 -1172
- package/convex/leaderboards.ts +0 -39
- package/convex/lib/access.ts +0 -36
- package/convex/lib/apiTokenAuth.ts +0 -36
- package/convex/lib/badges.ts +0 -50
- package/convex/lib/changelog.test.ts +0 -34
- package/convex/lib/changelog.ts +0 -278
- package/convex/lib/embeddings.ts +0 -38
- package/convex/lib/githubBackup.ts +0 -443
- package/convex/lib/githubImport.test.ts +0 -247
- package/convex/lib/githubImport.ts +0 -425
- package/convex/lib/githubSoulBackup.ts +0 -443
- package/convex/lib/leaderboards.ts +0 -103
- package/convex/lib/moderation.ts +0 -42
- package/convex/lib/public.ts +0 -89
- package/convex/lib/searchText.test.ts +0 -46
- package/convex/lib/searchText.ts +0 -27
- package/convex/lib/skillBackfill.test.ts +0 -34
- package/convex/lib/skillBackfill.ts +0 -67
- package/convex/lib/skillPublish.test.ts +0 -28
- package/convex/lib/skillPublish.ts +0 -284
- package/convex/lib/skillStats.ts +0 -80
- package/convex/lib/skills.test.ts +0 -197
- package/convex/lib/skills.ts +0 -273
- package/convex/lib/soulChangelog.ts +0 -273
- package/convex/lib/soulPublish.ts +0 -236
- package/convex/lib/tokens.test.ts +0 -33
- package/convex/lib/tokens.ts +0 -51
- package/convex/lib/webhooks.test.ts +0 -91
- package/convex/lib/webhooks.ts +0 -112
- package/convex/maintenance.test.ts +0 -270
- package/convex/maintenance.ts +0 -840
- package/convex/rateLimits.ts +0 -50
- package/convex/schema.ts +0 -472
- package/convex/search.test.ts +0 -12
- package/convex/search.ts +0 -254
- package/convex/seed.test.ts +0 -37
- package/convex/seed.ts +0 -254
- package/convex/seedSouls.ts +0 -111
- package/convex/skillStatEvents.ts +0 -568
- package/convex/skills.ts +0 -1606
- package/convex/soulComments.ts +0 -88
- package/convex/soulDownloads.ts +0 -14
- package/convex/soulStars.ts +0 -71
- package/convex/souls.ts +0 -570
- package/convex/stars.ts +0 -108
- package/convex/statsMaintenance.ts +0 -205
- package/convex/telemetry.ts +0 -434
- package/convex/tokens.ts +0 -88
- package/convex/tsconfig.json +0 -7
- package/convex/uploads.ts +0 -20
- package/convex/users.ts +0 -122
- package/convex/webhooks.ts +0 -50
- package/convex.json +0 -3
- package/docs/README.md +0 -32
- package/docs/api.md +0 -51
- package/docs/architecture.md +0 -61
- package/docs/auth.md +0 -54
- package/docs/cli.md +0 -117
- package/docs/deploy.md +0 -78
- package/docs/diffing.md +0 -84
- package/docs/github-import.md +0 -171
- package/docs/http-api.md +0 -187
- package/docs/manual-testing.md +0 -64
- package/docs/mintlify.md +0 -43
- package/docs/quickstart.md +0 -120
- package/docs/skill-format.md +0 -58
- package/docs/soul-format.md +0 -37
- package/docs/spec.md +0 -177
- package/docs/telemetry.md +0 -91
- package/docs/troubleshooting.md +0 -49
- package/docs/webhook.md +0 -51
- package/e2e/menu-smoke.pw.test.ts +0 -49
- package/e2e/pilothub.e2e.test.ts +0 -494
- package/e2e/search-exact.pw.test.ts +0 -97
- package/packages/pilothub/LICENSE +0 -22
- package/packages/pilothub/README.md +0 -57
- package/packages/pilothub/package.json +0 -41
- package/packages/pilothub/src/browserAuth.test.ts +0 -96
- package/packages/pilothub/src/browserAuth.ts +0 -174
- package/packages/pilothub/src/cli/buildInfo.ts +0 -94
- package/packages/pilothub/src/cli/commands/auth.ts +0 -97
- package/packages/pilothub/src/cli/commands/delete.test.ts +0 -73
- package/packages/pilothub/src/cli/commands/delete.ts +0 -83
- package/packages/pilothub/src/cli/commands/publish.test.ts +0 -122
- package/packages/pilothub/src/cli/commands/publish.ts +0 -108
- package/packages/pilothub/src/cli/commands/skills.test.ts +0 -191
- package/packages/pilothub/src/cli/commands/skills.ts +0 -380
- package/packages/pilothub/src/cli/commands/star.ts +0 -46
- package/packages/pilothub/src/cli/commands/sync.test.ts +0 -310
- package/packages/pilothub/src/cli/commands/sync.ts +0 -200
- package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +0 -26
- package/packages/pilothub/src/cli/commands/syncHelpers.ts +0 -427
- package/packages/pilothub/src/cli/commands/syncTypes.ts +0 -27
- package/packages/pilothub/src/cli/commands/unstar.ts +0 -48
- package/packages/pilothub/src/cli/helpStyle.ts +0 -45
- package/packages/pilothub/src/cli/pilotbotConfig.test.ts +0 -159
- package/packages/pilothub/src/cli/pilotbotConfig.ts +0 -147
- package/packages/pilothub/src/cli/registry.test.ts +0 -63
- package/packages/pilothub/src/cli/registry.ts +0 -43
- package/packages/pilothub/src/cli/scanSkills.test.ts +0 -64
- package/packages/pilothub/src/cli/scanSkills.ts +0 -84
- package/packages/pilothub/src/cli/slug.ts +0 -16
- package/packages/pilothub/src/cli/types.ts +0 -12
- package/packages/pilothub/src/cli/ui.ts +0 -75
- package/packages/pilothub/src/cli.ts +0 -311
- package/packages/pilothub/src/config.ts +0 -36
- package/packages/pilothub/src/discovery.test.ts +0 -75
- package/packages/pilothub/src/discovery.ts +0 -19
- package/packages/pilothub/src/http.test.ts +0 -156
- package/packages/pilothub/src/http.ts +0 -301
- package/packages/pilothub/src/schema/ark.ts +0 -29
- package/packages/pilothub/src/schema/index.ts +0 -5
- package/packages/pilothub/src/schema/routes.ts +0 -22
- package/packages/pilothub/src/schema/schemas.ts +0 -260
- package/packages/pilothub/src/schema/textFiles.test.ts +0 -23
- package/packages/pilothub/src/schema/textFiles.ts +0 -66
- package/packages/pilothub/src/skills.test.ts +0 -191
- package/packages/pilothub/src/skills.ts +0 -172
- package/packages/pilothub/src/types.ts +0 -10
- package/packages/pilothub/tsconfig.json +0 -14
- package/packages/schema/README.md +0 -3
- package/packages/schema/dist/ark.js.map +0 -1
- package/packages/schema/dist/index.js.map +0 -1
- package/packages/schema/dist/routes.js.map +0 -1
- package/packages/schema/dist/schemas.js.map +0 -1
- package/packages/schema/dist/textFiles.js.map +0 -1
- package/packages/schema/package.json +0 -26
- package/packages/schema/src/ark.ts +0 -29
- package/packages/schema/src/index.ts +0 -5
- package/packages/schema/src/routes.ts +0 -22
- package/packages/schema/src/schemas.test.ts +0 -123
- package/packages/schema/src/schemas.ts +0 -287
- package/packages/schema/src/textFiles.test.ts +0 -23
- package/packages/schema/src/textFiles.ts +0 -66
- package/packages/schema/tsconfig.json +0 -15
- package/pilothub +0 -46
- package/playwright.config.ts +0 -33
- package/public/.well-known/pilothub.json +0 -6
- package/public/api/v1/openapi.json +0 -379
- package/public/favicon.ico +0 -0
- package/public/logo192.png +0 -0
- package/public/logo512.png +0 -0
- package/public/manifest.json +0 -25
- package/public/og.png +0 -0
- package/public/og.svg +0 -98
- package/public/pilot-logo.png +0 -0
- package/public/pilot-mark.png +0 -0
- package/public/robots.txt +0 -3
- package/public/tanstack-circle-logo.png +0 -0
- package/public/tanstack-word-logo-white.svg +0 -1
- package/scripts/check-peer-deps.ts +0 -56
- package/scripts/docs-list.ts +0 -148
- package/scripts/run-playwright-local.sh +0 -14
- package/server/og/fetchSkillOgMeta.ts +0 -27
- package/server/og/fetchSoulOgMeta.ts +0 -27
- package/server/og/ogAssets.ts +0 -80
- package/server/og/skillOgSvg.test.ts +0 -59
- package/server/og/skillOgSvg.ts +0 -258
- package/server/og/soulOgSvg.ts +0 -209
- package/server/routes/og/skill.png.ts +0 -103
- package/server/routes/og/soul.png.ts +0 -111
- package/src/__tests__/skill-detail-page.test.tsx +0 -86
- package/src/__tests__/skills-index.test.tsx +0 -145
- package/src/__tests__/upload.route.test.tsx +0 -228
- package/src/components/AppProviders.tsx +0 -19
- package/src/components/ClientOnly.tsx +0 -18
- package/src/components/Footer.tsx +0 -29
- package/src/components/Header.tsx +0 -295
- package/src/components/InstallSwitcher.tsx +0 -53
- package/src/components/SkillCard.tsx +0 -36
- package/src/components/SkillDetailPage.tsx +0 -817
- package/src/components/SkillDiffCard.tsx +0 -485
- package/src/components/SoulCard.tsx +0 -19
- package/src/components/SoulDetailPage.tsx +0 -263
- package/src/components/UserBootstrap.tsx +0 -18
- package/src/components/ui/dropdown-menu.tsx +0 -67
- package/src/components/ui/toggle-group.tsx +0 -35
- package/src/convex/client.ts +0 -3
- package/src/lib/badges.ts +0 -29
- package/src/lib/diffing.test.ts +0 -163
- package/src/lib/diffing.ts +0 -106
- package/src/lib/gravatar.test.ts +0 -9
- package/src/lib/gravatar.ts +0 -158
- package/src/lib/og.test.ts +0 -142
- package/src/lib/og.ts +0 -156
- package/src/lib/publicUser.ts +0 -39
- package/src/lib/roles.ts +0 -19
- package/src/lib/site.test.ts +0 -130
- package/src/lib/site.ts +0 -84
- package/src/lib/theme-transition.test.ts +0 -134
- package/src/lib/theme-transition.ts +0 -134
- package/src/lib/theme.test.tsx +0 -88
- package/src/lib/theme.ts +0 -43
- package/src/lib/uploadFiles.jsdom.test.ts +0 -33
- package/src/lib/uploadFiles.test.ts +0 -123
- package/src/lib/uploadFiles.ts +0 -245
- package/src/lib/uploadUtils.test.ts +0 -78
- package/src/lib/uploadUtils.ts +0 -93
- package/src/lib/useAuthStatus.ts +0 -12
- package/src/lib/utils.test.ts +0 -9
- package/src/lib/utils.ts +0 -6
- package/src/logo.svg +0 -12
- package/src/routeTree.gen.ts +0 -345
- package/src/router.tsx +0 -17
- package/src/routes/$owner/$slug.tsx +0 -55
- package/src/routes/__root.tsx +0 -136
- package/src/routes/admin.tsx +0 -11
- package/src/routes/cli/auth.tsx +0 -168
- package/src/routes/dashboard.tsx +0 -97
- package/src/routes/import.tsx +0 -415
- package/src/routes/index.tsx +0 -252
- package/src/routes/management.tsx +0 -529
- package/src/routes/settings.tsx +0 -203
- package/src/routes/skills/index.tsx +0 -422
- package/src/routes/souls/$slug.tsx +0 -55
- package/src/routes/souls/index.tsx +0 -243
- package/src/routes/stars.tsx +0 -68
- package/src/routes/u/$handle.tsx +0 -307
- package/src/routes/upload/utils.ts +0 -81
- package/src/routes/upload.tsx +0 -499
- package/src/styles.css +0 -2718
- package/tsconfig.json +0 -24
- package/tsconfig.oxlint.json +0 -16
- package/vercel.json +0 -8
- package/vite.config.ts +0 -48
- package/vitest.config.ts +0 -47
- package/vitest.e2e.config.ts +0 -11
- package/vitest.setup.ts +0 -1
- /package/{packages/pilothub/bin → bin}/pilothub.js +0 -0
- /package/{packages/schema/dist → dist/schema}/ark.d.ts +0 -0
- /package/{packages/schema/dist → dist/schema}/ark.js +0 -0
- /package/{packages/schema/dist → dist/schema}/index.d.ts +0 -0
- /package/{packages/schema/dist → dist/schema}/index.js +0 -0
- /package/{packages/schema/dist → dist/schema}/routes.d.ts +0 -0
- /package/{packages/schema/dist → dist/schema}/routes.js +0 -0
- /package/{packages/schema/dist → dist/schema}/textFiles.d.ts +0 -0
- /package/{packages/schema/dist → dist/schema}/textFiles.js +0 -0
|
@@ -1,147 +0,0 @@
|
|
|
1
|
-
import { readFile } from 'node:fs/promises'
|
|
2
|
-
import { homedir } from 'node:os'
|
|
3
|
-
import { basename, join, resolve } from 'node:path'
|
|
4
|
-
import JSON5 from 'json5'
|
|
5
|
-
|
|
6
|
-
type PilotbotConfig = {
|
|
7
|
-
agent?: { workspace?: string }
|
|
8
|
-
agents?: {
|
|
9
|
-
defaults?: { workspace?: string }
|
|
10
|
-
list?: Array<{
|
|
11
|
-
id?: string
|
|
12
|
-
name?: string
|
|
13
|
-
workspace?: string
|
|
14
|
-
default?: boolean
|
|
15
|
-
}>
|
|
16
|
-
}
|
|
17
|
-
routing?: {
|
|
18
|
-
agents?: Record<
|
|
19
|
-
string,
|
|
20
|
-
{
|
|
21
|
-
name?: string
|
|
22
|
-
workspace?: string
|
|
23
|
-
}
|
|
24
|
-
>
|
|
25
|
-
}
|
|
26
|
-
skills?: {
|
|
27
|
-
load?: {
|
|
28
|
-
extraDirs?: string[]
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export type PilotbotSkillRoots = {
|
|
34
|
-
roots: string[]
|
|
35
|
-
labels: Record<string, string>
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
export async function resolvePilotbotSkillRoots(): Promise<PilotbotSkillRoots> {
|
|
39
|
-
const roots: string[] = []
|
|
40
|
-
const labels: Record<string, string> = {}
|
|
41
|
-
|
|
42
|
-
const stateDir = resolvePilotbotStateDir()
|
|
43
|
-
const sharedSkills = resolveUserPath(join(stateDir, 'skills'))
|
|
44
|
-
pushRoot(roots, labels, sharedSkills, 'Shared skills')
|
|
45
|
-
|
|
46
|
-
const config = await readPilotbotConfig()
|
|
47
|
-
if (!config) return { roots, labels }
|
|
48
|
-
|
|
49
|
-
const mainWorkspace = resolveUserPath(
|
|
50
|
-
config.agents?.defaults?.workspace ?? config.agent?.workspace ?? '',
|
|
51
|
-
)
|
|
52
|
-
if (mainWorkspace) {
|
|
53
|
-
pushRoot(roots, labels, join(mainWorkspace, 'skills'), 'Agent: main')
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
const listedAgents = config.agents?.list ?? []
|
|
57
|
-
for (const entry of listedAgents) {
|
|
58
|
-
const workspace = resolveUserPath(entry?.workspace ?? '')
|
|
59
|
-
if (!workspace) continue
|
|
60
|
-
const name = entry?.name?.trim() || entry?.id?.trim() || 'agent'
|
|
61
|
-
pushRoot(roots, labels, join(workspace, 'skills'), `Agent: ${name}`)
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
const agents = config.routing?.agents ?? {}
|
|
65
|
-
for (const [agentId, entry] of Object.entries(agents)) {
|
|
66
|
-
const workspace = resolveUserPath(entry?.workspace ?? '')
|
|
67
|
-
if (!workspace) continue
|
|
68
|
-
const name = entry?.name?.trim() || agentId
|
|
69
|
-
pushRoot(roots, labels, join(workspace, 'skills'), `Agent: ${name}`)
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
const extraDirs = config.skills?.load?.extraDirs ?? []
|
|
73
|
-
for (const dir of extraDirs) {
|
|
74
|
-
const resolved = resolveUserPath(String(dir))
|
|
75
|
-
if (!resolved) continue
|
|
76
|
-
const label = `Extra: ${basename(resolved) || resolved}`
|
|
77
|
-
pushRoot(roots, labels, resolved, label)
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
return { roots, labels }
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
export async function resolvePilotbotDefaultWorkspace(): Promise<string | null> {
|
|
84
|
-
const config = await readPilotbotConfig()
|
|
85
|
-
if (!config) return null
|
|
86
|
-
|
|
87
|
-
const defaultsWorkspace = resolveUserPath(
|
|
88
|
-
config.agents?.defaults?.workspace ?? config.agent?.workspace ?? '',
|
|
89
|
-
)
|
|
90
|
-
if (defaultsWorkspace) return defaultsWorkspace
|
|
91
|
-
|
|
92
|
-
const listedAgents = config.agents?.list ?? []
|
|
93
|
-
const defaultAgent =
|
|
94
|
-
listedAgents.find((entry) => entry.default) ?? listedAgents.find((entry) => entry.id === 'main')
|
|
95
|
-
const listWorkspace = resolveUserPath(defaultAgent?.workspace ?? '')
|
|
96
|
-
return listWorkspace || null
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
function resolvePilotbotStateDir() {
|
|
100
|
-
const override = process.env.PILOTBOT_STATE_DIR?.trim()
|
|
101
|
-
if (override) return resolveUserPath(override)
|
|
102
|
-
return join(homedir(), '.pilotbot')
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
function resolvePilotbotConfigPath() {
|
|
106
|
-
const override = process.env.PILOTBOT_CONFIG_PATH?.trim()
|
|
107
|
-
if (override) return resolveUserPath(override)
|
|
108
|
-
return join(resolvePilotbotStateDir(), 'pilotbot.json')
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
function resolveUserPath(input: string) {
|
|
112
|
-
const trimmed = input.trim()
|
|
113
|
-
if (!trimmed) return ''
|
|
114
|
-
if (trimmed.startsWith('~')) {
|
|
115
|
-
return resolve(trimmed.replace(/^~(?=$|[\\/])/, homedir()))
|
|
116
|
-
}
|
|
117
|
-
return resolve(trimmed)
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
async function readPilotbotConfig(): Promise<PilotbotConfig | null> {
|
|
121
|
-
try {
|
|
122
|
-
const raw = await readFile(resolvePilotbotConfigPath(), 'utf8')
|
|
123
|
-
const parsed = JSON5.parse(raw)
|
|
124
|
-
if (!parsed || typeof parsed !== 'object') return null
|
|
125
|
-
return parsed as PilotbotConfig
|
|
126
|
-
} catch {
|
|
127
|
-
return null
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
function pushRoot(roots: string[], labels: Record<string, string>, root: string, label?: string) {
|
|
132
|
-
const resolved = resolveUserPath(root)
|
|
133
|
-
if (!resolved) return
|
|
134
|
-
if (!roots.includes(resolved)) roots.push(resolved)
|
|
135
|
-
if (!label) return
|
|
136
|
-
const existing = labels[resolved]
|
|
137
|
-
if (!existing) {
|
|
138
|
-
labels[resolved] = label
|
|
139
|
-
return
|
|
140
|
-
}
|
|
141
|
-
const parts = existing
|
|
142
|
-
.split(', ')
|
|
143
|
-
.map((part) => part.trim())
|
|
144
|
-
.filter(Boolean)
|
|
145
|
-
if (parts.includes(label)) return
|
|
146
|
-
labels[resolved] = `${existing}, ${label}`
|
|
147
|
-
}
|
|
@@ -1,63 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { beforeEach, describe, expect, it, vi } from 'vitest'
|
|
4
|
-
import type { GlobalOpts } from './types'
|
|
5
|
-
|
|
6
|
-
const readGlobalConfig = vi.fn()
|
|
7
|
-
const writeGlobalConfig = vi.fn()
|
|
8
|
-
const discoverRegistryFromSite = vi.fn()
|
|
9
|
-
|
|
10
|
-
vi.mock('../config.js', () => ({
|
|
11
|
-
readGlobalConfig: (...args: unknown[]) => readGlobalConfig(...args),
|
|
12
|
-
writeGlobalConfig: (...args: unknown[]) => writeGlobalConfig(...args),
|
|
13
|
-
}))
|
|
14
|
-
|
|
15
|
-
vi.mock('../discovery.js', () => ({
|
|
16
|
-
discoverRegistryFromSite: (...args: unknown[]) => discoverRegistryFromSite(...args),
|
|
17
|
-
}))
|
|
18
|
-
|
|
19
|
-
const { DEFAULT_REGISTRY, getRegistry, resolveRegistry } = await import('./registry')
|
|
20
|
-
|
|
21
|
-
function makeOpts(overrides: Partial<GlobalOpts> = {}): GlobalOpts {
|
|
22
|
-
return {
|
|
23
|
-
workdir: '/work',
|
|
24
|
-
dir: '/work/skills',
|
|
25
|
-
site: 'https://pilothub.com',
|
|
26
|
-
registry: DEFAULT_REGISTRY,
|
|
27
|
-
registrySource: 'default',
|
|
28
|
-
...overrides,
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
beforeEach(() => {
|
|
33
|
-
readGlobalConfig.mockReset()
|
|
34
|
-
writeGlobalConfig.mockReset()
|
|
35
|
-
discoverRegistryFromSite.mockReset()
|
|
36
|
-
})
|
|
37
|
-
|
|
38
|
-
describe('registry resolution', () => {
|
|
39
|
-
it('prefers explicit registry over discovery/cache', async () => {
|
|
40
|
-
readGlobalConfig.mockResolvedValue({ registry: 'https://auth.pilothub.com' })
|
|
41
|
-
discoverRegistryFromSite.mockResolvedValue({ apiBase: 'https://pilothub.com' })
|
|
42
|
-
|
|
43
|
-
const registry = await resolveRegistry(
|
|
44
|
-
makeOpts({ registry: 'https://custom.example', registrySource: 'cli' }),
|
|
45
|
-
)
|
|
46
|
-
|
|
47
|
-
expect(registry).toBe('https://custom.example')
|
|
48
|
-
expect(discoverRegistryFromSite).not.toHaveBeenCalled()
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
it('ignores legacy registry and updates cache from discovery', async () => {
|
|
52
|
-
readGlobalConfig.mockResolvedValue({ registry: 'https://auth.pilothub.com', token: 'tkn' })
|
|
53
|
-
discoverRegistryFromSite.mockResolvedValue({ apiBase: 'https://pilothub.com' })
|
|
54
|
-
|
|
55
|
-
const registry = await getRegistry(makeOpts(), { cache: true })
|
|
56
|
-
|
|
57
|
-
expect(registry).toBe('https://pilothub.com')
|
|
58
|
-
expect(writeGlobalConfig).toHaveBeenCalledWith({
|
|
59
|
-
registry: 'https://pilothub.com',
|
|
60
|
-
token: 'tkn',
|
|
61
|
-
})
|
|
62
|
-
})
|
|
63
|
-
})
|
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
import { readGlobalConfig, writeGlobalConfig } from '../config.js'
|
|
2
|
-
import { discoverRegistryFromSite } from '../discovery.js'
|
|
3
|
-
import type { GlobalOpts } from './types.js'
|
|
4
|
-
|
|
5
|
-
export const DEFAULT_SITE = 'https://pilothub.com'
|
|
6
|
-
export const DEFAULT_REGISTRY = 'https://pilothub.com'
|
|
7
|
-
const LEGACY_REGISTRY_HOSTS = new Set(['auth.pilothub.com'])
|
|
8
|
-
|
|
9
|
-
export async function resolveRegistry(opts: GlobalOpts) {
|
|
10
|
-
const explicit = opts.registrySource !== 'default' ? opts.registry.trim() : ''
|
|
11
|
-
if (explicit) return explicit
|
|
12
|
-
|
|
13
|
-
const discovery = await discoverRegistryFromSite(opts.site).catch(() => null)
|
|
14
|
-
const discovered = discovery?.apiBase?.trim()
|
|
15
|
-
if (discovered) return discovered
|
|
16
|
-
|
|
17
|
-
const cfg = await readGlobalConfig()
|
|
18
|
-
const cached = cfg?.registry?.trim()
|
|
19
|
-
if (cached && !isLegacyRegistry(cached)) return cached
|
|
20
|
-
return DEFAULT_REGISTRY
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
export async function getRegistry(opts: GlobalOpts, params?: { cache?: boolean }) {
|
|
24
|
-
const cache = params?.cache !== false
|
|
25
|
-
const registry = await resolveRegistry(opts)
|
|
26
|
-
if (!cache) return registry
|
|
27
|
-
const cfg = await readGlobalConfig()
|
|
28
|
-
const cached = cfg?.registry?.trim()
|
|
29
|
-
const shouldUpdate =
|
|
30
|
-
!cached ||
|
|
31
|
-
isLegacyRegistry(cached) ||
|
|
32
|
-
(cached === DEFAULT_REGISTRY && registry !== DEFAULT_REGISTRY)
|
|
33
|
-
if (shouldUpdate) await writeGlobalConfig({ registry, token: cfg?.token })
|
|
34
|
-
return registry
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function isLegacyRegistry(registry: string) {
|
|
38
|
-
try {
|
|
39
|
-
return LEGACY_REGISTRY_HOSTS.has(new URL(registry).hostname)
|
|
40
|
-
} catch {
|
|
41
|
-
return false
|
|
42
|
-
}
|
|
43
|
-
}
|
|
@@ -1,64 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { mkdir, mkdtemp, rm, writeFile } from 'node:fs/promises'
|
|
4
|
-
import { tmpdir } from 'node:os'
|
|
5
|
-
import { join, resolve } from 'node:path'
|
|
6
|
-
import { describe, expect, it } from 'vitest'
|
|
7
|
-
import { findSkillFolders, getFallbackSkillRoots } from './scanSkills'
|
|
8
|
-
|
|
9
|
-
async function makeTmpDir() {
|
|
10
|
-
return mkdtemp(join(tmpdir(), 'pilothub-scan-'))
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
describe('scanSkills', () => {
|
|
14
|
-
it('detects a single skill folder (root contains SKILL.md)', async () => {
|
|
15
|
-
const root = await makeTmpDir()
|
|
16
|
-
try {
|
|
17
|
-
await writeFile(join(root, 'SKILL.md'), '# Skill\n', 'utf8')
|
|
18
|
-
const found = await findSkillFolders(root)
|
|
19
|
-
expect(found).toHaveLength(1)
|
|
20
|
-
expect(found[0]?.folder).toBe(resolve(root))
|
|
21
|
-
expect(found[0]?.slug).toBeTruthy()
|
|
22
|
-
} finally {
|
|
23
|
-
await rm(root, { recursive: true, force: true })
|
|
24
|
-
}
|
|
25
|
-
})
|
|
26
|
-
|
|
27
|
-
it('detects skills in a skills directory (subfolders)', async () => {
|
|
28
|
-
const root = await makeTmpDir()
|
|
29
|
-
try {
|
|
30
|
-
const skillsDir = join(root, 'skills')
|
|
31
|
-
const folder = join(skillsDir, 'cool-skill')
|
|
32
|
-
await mkdir(folder, { recursive: true })
|
|
33
|
-
await writeFile(join(folder, 'SKILL.md'), '# Skill\n', 'utf8')
|
|
34
|
-
|
|
35
|
-
const found = await findSkillFolders(skillsDir)
|
|
36
|
-
expect(found).toHaveLength(1)
|
|
37
|
-
expect(found[0]?.slug).toBe('cool-skill')
|
|
38
|
-
expect(found[0]?.folder).toBe(resolve(folder))
|
|
39
|
-
} finally {
|
|
40
|
-
await rm(root, { recursive: true, force: true })
|
|
41
|
-
}
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
it('ignores plural skills.md marker files', async () => {
|
|
45
|
-
const root = await makeTmpDir()
|
|
46
|
-
try {
|
|
47
|
-
const folder = join(root, 'docs')
|
|
48
|
-
await mkdir(folder, { recursive: true })
|
|
49
|
-
await writeFile(join(folder, 'skills.md'), '# Docs\n', 'utf8')
|
|
50
|
-
|
|
51
|
-
const found = await findSkillFolders(root)
|
|
52
|
-
expect(found).toHaveLength(0)
|
|
53
|
-
} finally {
|
|
54
|
-
await rm(root, { recursive: true, force: true })
|
|
55
|
-
}
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
it('includes known legacy roots', () => {
|
|
59
|
-
const roots = getFallbackSkillRoots('/tmp/anywhere')
|
|
60
|
-
expect(roots.some((p) => p.endsWith('/pilotbot/skills'))).toBe(true)
|
|
61
|
-
expect(roots.some((p) => p.endsWith('/pilot/skills'))).toBe(true)
|
|
62
|
-
expect(roots.some((p) => p.endsWith('/pilotbot/skills'))).toBe(true)
|
|
63
|
-
})
|
|
64
|
-
})
|
|
@@ -1,84 +0,0 @@
|
|
|
1
|
-
import { readdir, stat } from 'node:fs/promises'
|
|
2
|
-
import { homedir } from 'node:os'
|
|
3
|
-
import { basename, join, resolve } from 'node:path'
|
|
4
|
-
import { sanitizeSlug, titleCase } from './slug.js'
|
|
5
|
-
|
|
6
|
-
export type SkillFolder = {
|
|
7
|
-
folder: string
|
|
8
|
-
slug: string
|
|
9
|
-
displayName: string
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
export async function findSkillFolders(root: string): Promise<SkillFolder[]> {
|
|
13
|
-
const absRoot = resolve(root)
|
|
14
|
-
const rootStat = await stat(absRoot).catch(() => null)
|
|
15
|
-
if (!rootStat || !rootStat.isDirectory()) return []
|
|
16
|
-
|
|
17
|
-
const direct = await isSkillFolder(absRoot)
|
|
18
|
-
if (direct) return [direct]
|
|
19
|
-
|
|
20
|
-
const entries = await readdir(absRoot, { withFileTypes: true }).catch(() => [])
|
|
21
|
-
const folders = entries
|
|
22
|
-
.filter((entry) => entry.isDirectory())
|
|
23
|
-
.map((entry) => join(absRoot, entry.name))
|
|
24
|
-
const results: SkillFolder[] = []
|
|
25
|
-
for (const folder of folders) {
|
|
26
|
-
const found = await isSkillFolder(folder)
|
|
27
|
-
if (found) results.push(found)
|
|
28
|
-
}
|
|
29
|
-
return results.sort((a, b) => a.slug.localeCompare(b.slug))
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
export function getFallbackSkillRoots(workdir: string) {
|
|
33
|
-
const home = homedir()
|
|
34
|
-
const roots = [
|
|
35
|
-
// adjacent repo installs
|
|
36
|
-
resolve(workdir, '..', 'pilotbot', 'skills'),
|
|
37
|
-
resolve(workdir, '..', 'pilotbot', 'Skills'),
|
|
38
|
-
resolve(workdir, '..', 'pilotbot', 'skills'),
|
|
39
|
-
resolve(workdir, '..', 'pilotbot', 'Skills'),
|
|
40
|
-
|
|
41
|
-
// legacy locations
|
|
42
|
-
resolve(home, 'pilot', 'skills'),
|
|
43
|
-
resolve(home, 'pilot', 'Skills'),
|
|
44
|
-
resolve(home, '.pilot', 'skills'),
|
|
45
|
-
resolve(home, '.pilot', 'Skills'),
|
|
46
|
-
|
|
47
|
-
resolve(home, 'pilotbot', 'skills'),
|
|
48
|
-
resolve(home, 'pilotbot', 'Skills'),
|
|
49
|
-
resolve(home, '.pilotbot', 'skills'),
|
|
50
|
-
resolve(home, '.pilotbot', 'Skills'),
|
|
51
|
-
|
|
52
|
-
resolve(home, 'pilotbot', 'skills'),
|
|
53
|
-
resolve(home, 'pilotbot', 'Skills'),
|
|
54
|
-
resolve(home, '.pilotbot', 'skills'),
|
|
55
|
-
resolve(home, '.pilotbot', 'Skills'),
|
|
56
|
-
|
|
57
|
-
// macOS App Support legacy
|
|
58
|
-
resolve(home, 'Library', 'Application Support', 'pilotbot', 'skills'),
|
|
59
|
-
resolve(home, 'Library', 'Application Support', 'pilotbot', 'Skills'),
|
|
60
|
-
resolve(home, 'Library', 'Application Support', 'pilotbot', 'skills'),
|
|
61
|
-
resolve(home, 'Library', 'Application Support', 'pilotbot', 'Skills'),
|
|
62
|
-
]
|
|
63
|
-
return Array.from(new Set(roots))
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
async function isSkillFolder(folder: string): Promise<SkillFolder | null> {
|
|
67
|
-
const marker = await findSkillMarker(folder)
|
|
68
|
-
if (!marker) return null
|
|
69
|
-
const base = basename(folder)
|
|
70
|
-
const slug = sanitizeSlug(base)
|
|
71
|
-
if (!slug) return null
|
|
72
|
-
const displayName = titleCase(base)
|
|
73
|
-
return { folder, slug, displayName }
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
async function findSkillMarker(folder: string) {
|
|
77
|
-
const candidates = ['SKILL.md', 'skill.md']
|
|
78
|
-
for (const name of candidates) {
|
|
79
|
-
const path = join(folder, name)
|
|
80
|
-
const st = await stat(path).catch(() => null)
|
|
81
|
-
if (st?.isFile()) return path
|
|
82
|
-
}
|
|
83
|
-
return null
|
|
84
|
-
}
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
export function sanitizeSlug(value: string) {
|
|
2
|
-
const raw = value
|
|
3
|
-
.trim()
|
|
4
|
-
.toLowerCase()
|
|
5
|
-
.replace(/[^a-z0-9-]+/g, '-')
|
|
6
|
-
const cleaned = raw.replace(/^-+/, '').replace(/-+$/, '').replace(/--+/g, '-')
|
|
7
|
-
return cleaned
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
export function titleCase(value: string) {
|
|
11
|
-
return value
|
|
12
|
-
.trim()
|
|
13
|
-
.replace(/[-_]+/g, ' ')
|
|
14
|
-
.replace(/\s+/g, ' ')
|
|
15
|
-
.replace(/\b\w/g, (char) => char.toUpperCase())
|
|
16
|
-
}
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
export type GlobalOpts = {
|
|
2
|
-
workdir: string
|
|
3
|
-
dir: string
|
|
4
|
-
site: string
|
|
5
|
-
registry: string
|
|
6
|
-
registrySource: 'cli' | 'env' | 'default'
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
export type ResolveResult = {
|
|
10
|
-
match: { version: string } | null
|
|
11
|
-
latestVersion: { version: string } | null
|
|
12
|
-
}
|
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
import { spawn } from 'node:child_process'
|
|
2
|
-
import { stdin } from 'node:process'
|
|
3
|
-
import { confirm, isCancel } from '@clack/prompts'
|
|
4
|
-
import ora from 'ora'
|
|
5
|
-
|
|
6
|
-
export async function promptHidden(prompt: string) {
|
|
7
|
-
if (!stdin.isTTY) return ''
|
|
8
|
-
process.stdout.write(prompt)
|
|
9
|
-
const chunks: Buffer[] = []
|
|
10
|
-
stdin.setRawMode(true)
|
|
11
|
-
stdin.resume()
|
|
12
|
-
return new Promise<string>((resolvePromise) => {
|
|
13
|
-
function onData(data: Buffer) {
|
|
14
|
-
const text = data.toString('utf8')
|
|
15
|
-
if (text === '\r' || text === '\n') {
|
|
16
|
-
stdin.setRawMode(false)
|
|
17
|
-
stdin.pause()
|
|
18
|
-
stdin.off('data', onData)
|
|
19
|
-
process.stdout.write('\n')
|
|
20
|
-
resolvePromise(Buffer.concat(chunks).toString('utf8').trim())
|
|
21
|
-
return
|
|
22
|
-
}
|
|
23
|
-
if (text === '\u0003') {
|
|
24
|
-
stdin.setRawMode(false)
|
|
25
|
-
stdin.pause()
|
|
26
|
-
stdin.off('data', onData)
|
|
27
|
-
process.stdout.write('\n')
|
|
28
|
-
fail('Canceled')
|
|
29
|
-
}
|
|
30
|
-
if (text === '\u007f') {
|
|
31
|
-
chunks.pop()
|
|
32
|
-
return
|
|
33
|
-
}
|
|
34
|
-
chunks.push(data)
|
|
35
|
-
}
|
|
36
|
-
stdin.on('data', onData)
|
|
37
|
-
})
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
export async function promptConfirm(prompt: string) {
|
|
41
|
-
const answer = await confirm({ message: prompt })
|
|
42
|
-
if (isCancel(answer)) return false
|
|
43
|
-
return Boolean(answer)
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
export function openInBrowser(url: string) {
|
|
47
|
-
const args =
|
|
48
|
-
process.platform === 'darwin'
|
|
49
|
-
? ['open', url]
|
|
50
|
-
: process.platform === 'win32'
|
|
51
|
-
? ['cmd', '/c', 'start', '', url]
|
|
52
|
-
: ['xdg-open', url]
|
|
53
|
-
const [command, ...commandArgs] = args
|
|
54
|
-
if (!command) return
|
|
55
|
-
const child = spawn(command, commandArgs, { stdio: 'ignore', detached: true })
|
|
56
|
-
child.unref()
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
export function isInteractive() {
|
|
60
|
-
return Boolean(process.stdout.isTTY && stdin.isTTY)
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
export function createSpinner(text: string) {
|
|
64
|
-
return ora({ text, spinner: 'dots', isEnabled: isInteractive() }).start()
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
export function formatError(error: unknown) {
|
|
68
|
-
if (error instanceof Error) return error.message
|
|
69
|
-
return String(error)
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
export function fail(message: string): never {
|
|
73
|
-
console.error(`Error: ${message}`)
|
|
74
|
-
process.exit(1)
|
|
75
|
-
}
|