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,311 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import { stat } from 'node:fs/promises'
|
|
3
|
-
import { join, resolve } from 'node:path'
|
|
4
|
-
import { Command } from 'commander'
|
|
5
|
-
import { getCliBuildLabel, getCliVersion } from './cli/buildInfo.js'
|
|
6
|
-
import { cmdLoginFlow, cmdLogout, cmdWhoami } from './cli/commands/auth.js'
|
|
7
|
-
import { cmdDeleteSkill, cmdUndeleteSkill } from './cli/commands/delete.js'
|
|
8
|
-
import { cmdPublish } from './cli/commands/publish.js'
|
|
9
|
-
import { cmdExplore, cmdInstall, cmdList, cmdSearch, cmdUpdate } from './cli/commands/skills.js'
|
|
10
|
-
import { cmdStarSkill } from './cli/commands/star.js'
|
|
11
|
-
import { cmdSync } from './cli/commands/sync.js'
|
|
12
|
-
import { cmdUnstarSkill } from './cli/commands/unstar.js'
|
|
13
|
-
import { configureCommanderHelp, styleEnvBlock, styleTitle } from './cli/helpStyle.js'
|
|
14
|
-
import { resolvePilotbotDefaultWorkspace } from './cli/pilotbotConfig.js'
|
|
15
|
-
import { DEFAULT_REGISTRY, DEFAULT_SITE } from './cli/registry.js'
|
|
16
|
-
import type { GlobalOpts } from './cli/types.js'
|
|
17
|
-
import { fail } from './cli/ui.js'
|
|
18
|
-
import { readGlobalConfig } from './config.js'
|
|
19
|
-
|
|
20
|
-
const program = new Command()
|
|
21
|
-
.name('pilothub')
|
|
22
|
-
.description(
|
|
23
|
-
`${styleTitle(`PilotHub CLI ${getCliBuildLabel()}`)}\n${styleEnvBlock(
|
|
24
|
-
'install, update, search, and publish agent skills.',
|
|
25
|
-
)}`,
|
|
26
|
-
)
|
|
27
|
-
.version(getCliVersion(), '-V, --cli-version', 'Show CLI version')
|
|
28
|
-
.option('--workdir <dir>', 'Working directory (default: cwd)')
|
|
29
|
-
.option('--dir <dir>', 'Skills directory (relative to workdir, default: skills)')
|
|
30
|
-
.option('--site <url>', 'Site base URL (for browser login)')
|
|
31
|
-
.option('--registry <url>', 'Registry API base URL')
|
|
32
|
-
.option('--no-input', 'Disable prompts')
|
|
33
|
-
.showHelpAfterError()
|
|
34
|
-
.showSuggestionAfterError()
|
|
35
|
-
.addHelpText(
|
|
36
|
-
'after',
|
|
37
|
-
styleEnvBlock('\nEnv:\n PILOTHUB_SITE\n PILOTHUB_REGISTRY\n PILOTHUB_WORKDIR\n'),
|
|
38
|
-
)
|
|
39
|
-
|
|
40
|
-
configureCommanderHelp(program)
|
|
41
|
-
|
|
42
|
-
async function resolveGlobalOpts(): Promise<GlobalOpts> {
|
|
43
|
-
const raw = program.opts<{ workdir?: string; dir?: string; site?: string; registry?: string }>()
|
|
44
|
-
const workdir = await resolveWorkdir(raw.workdir)
|
|
45
|
-
const dir = resolve(workdir, raw.dir ?? 'skills')
|
|
46
|
-
const site = raw.site ?? process.env.PILOTHUB_SITE ?? DEFAULT_SITE
|
|
47
|
-
const registrySource = raw.registry ? 'cli' : process.env.PILOTHUB_REGISTRY ? 'env' : 'default'
|
|
48
|
-
const registry = raw.registry ?? process.env.PILOTHUB_REGISTRY ?? DEFAULT_REGISTRY
|
|
49
|
-
return { workdir, dir, site, registry, registrySource }
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
function isInputAllowed() {
|
|
53
|
-
const globalFlags = program.opts<{ input?: boolean }>()
|
|
54
|
-
return globalFlags.input !== false
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
async function resolveWorkdir(explicit?: string) {
|
|
58
|
-
if (explicit?.trim()) return resolve(explicit.trim())
|
|
59
|
-
const envWorkdir = process.env.PILOTHUB_WORKDIR?.trim()
|
|
60
|
-
if (envWorkdir) return resolve(envWorkdir)
|
|
61
|
-
|
|
62
|
-
const cwd = resolve(process.cwd())
|
|
63
|
-
const hasMarker = await hasPilothubMarker(cwd)
|
|
64
|
-
if (hasMarker) return cwd
|
|
65
|
-
|
|
66
|
-
const pilotbotWorkspace = await resolvePilotbotDefaultWorkspace()
|
|
67
|
-
return pilotbotWorkspace ? resolve(pilotbotWorkspace) : cwd
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
async function hasPilothubMarker(workdir: string) {
|
|
71
|
-
const lockfile = join(workdir, '.pilothub', 'lock.json')
|
|
72
|
-
if (await pathExists(lockfile)) return true
|
|
73
|
-
const markerDir = join(workdir, '.pilothub')
|
|
74
|
-
return pathExists(markerDir)
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
async function pathExists(path: string) {
|
|
78
|
-
try {
|
|
79
|
-
await stat(path)
|
|
80
|
-
return true
|
|
81
|
-
} catch {
|
|
82
|
-
return false
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
program
|
|
87
|
-
.command('login')
|
|
88
|
-
.description('Log in (opens browser or stores token)')
|
|
89
|
-
.option('--token <token>', 'API token')
|
|
90
|
-
.option('--label <label>', 'Token label (browser flow only)', 'CLI token')
|
|
91
|
-
.option('--no-browser', 'Do not open browser (requires --token)')
|
|
92
|
-
.action(async (options) => {
|
|
93
|
-
const opts = await resolveGlobalOpts()
|
|
94
|
-
await cmdLoginFlow(opts, options, isInputAllowed())
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
program
|
|
98
|
-
.command('logout')
|
|
99
|
-
.description('Remove stored token')
|
|
100
|
-
.action(async () => {
|
|
101
|
-
const opts = await resolveGlobalOpts()
|
|
102
|
-
await cmdLogout(opts)
|
|
103
|
-
})
|
|
104
|
-
|
|
105
|
-
program
|
|
106
|
-
.command('whoami')
|
|
107
|
-
.description('Validate token')
|
|
108
|
-
.action(async () => {
|
|
109
|
-
const opts = await resolveGlobalOpts()
|
|
110
|
-
await cmdWhoami(opts)
|
|
111
|
-
})
|
|
112
|
-
|
|
113
|
-
const auth = program
|
|
114
|
-
.command('auth')
|
|
115
|
-
.description('Authentication commands')
|
|
116
|
-
.showHelpAfterError()
|
|
117
|
-
.showSuggestionAfterError()
|
|
118
|
-
|
|
119
|
-
auth
|
|
120
|
-
.command('login')
|
|
121
|
-
.description('Log in (opens browser or stores token)')
|
|
122
|
-
.option('--token <token>', 'API token')
|
|
123
|
-
.option('--label <label>', 'Token label (browser flow only)', 'CLI token')
|
|
124
|
-
.option('--no-browser', 'Do not open browser (requires --token)')
|
|
125
|
-
.action(async (options) => {
|
|
126
|
-
const opts = await resolveGlobalOpts()
|
|
127
|
-
await cmdLoginFlow(opts, options, isInputAllowed())
|
|
128
|
-
})
|
|
129
|
-
|
|
130
|
-
auth
|
|
131
|
-
.command('logout')
|
|
132
|
-
.description('Remove stored token')
|
|
133
|
-
.action(async () => {
|
|
134
|
-
const opts = await resolveGlobalOpts()
|
|
135
|
-
await cmdLogout(opts)
|
|
136
|
-
})
|
|
137
|
-
|
|
138
|
-
auth
|
|
139
|
-
.command('whoami')
|
|
140
|
-
.description('Validate token')
|
|
141
|
-
.action(async () => {
|
|
142
|
-
const opts = await resolveGlobalOpts()
|
|
143
|
-
await cmdWhoami(opts)
|
|
144
|
-
})
|
|
145
|
-
|
|
146
|
-
program
|
|
147
|
-
.command('search')
|
|
148
|
-
.description('Vector search skills')
|
|
149
|
-
.argument('<query...>', 'Query string')
|
|
150
|
-
.option('--limit <n>', 'Max results', (value) => Number.parseInt(value, 10))
|
|
151
|
-
.action(async (queryParts, options) => {
|
|
152
|
-
const opts = await resolveGlobalOpts()
|
|
153
|
-
const query = queryParts.join(' ').trim()
|
|
154
|
-
await cmdSearch(opts, query, options.limit)
|
|
155
|
-
})
|
|
156
|
-
|
|
157
|
-
program
|
|
158
|
-
.command('install')
|
|
159
|
-
.description('Install into <dir>/<slug>')
|
|
160
|
-
.argument('<slug>', 'Skill slug')
|
|
161
|
-
.option('--version <version>', 'Version to install')
|
|
162
|
-
.option('--force', 'Overwrite existing folder')
|
|
163
|
-
.action(async (slug, options) => {
|
|
164
|
-
const opts = await resolveGlobalOpts()
|
|
165
|
-
await cmdInstall(opts, slug, options.version, options.force)
|
|
166
|
-
})
|
|
167
|
-
|
|
168
|
-
program
|
|
169
|
-
.command('update')
|
|
170
|
-
.description('Update installed skills')
|
|
171
|
-
.argument('[slug]', 'Skill slug')
|
|
172
|
-
.option('--all', 'Update all installed skills')
|
|
173
|
-
.option('--version <version>', 'Update to specific version (single slug only)')
|
|
174
|
-
.option('--force', 'Overwrite when local files do not match any version')
|
|
175
|
-
.action(async (slug, options) => {
|
|
176
|
-
const opts = await resolveGlobalOpts()
|
|
177
|
-
await cmdUpdate(opts, slug, options, isInputAllowed())
|
|
178
|
-
})
|
|
179
|
-
|
|
180
|
-
program
|
|
181
|
-
.command('list')
|
|
182
|
-
.description('List installed skills (from lockfile)')
|
|
183
|
-
.action(async () => {
|
|
184
|
-
const opts = await resolveGlobalOpts()
|
|
185
|
-
await cmdList(opts)
|
|
186
|
-
})
|
|
187
|
-
|
|
188
|
-
program
|
|
189
|
-
.command('explore')
|
|
190
|
-
.description('Browse latest updated skills from the registry')
|
|
191
|
-
.option(
|
|
192
|
-
'--limit <n>',
|
|
193
|
-
'Number of skills to show (max 200)',
|
|
194
|
-
(value) => Number.parseInt(value, 10),
|
|
195
|
-
25,
|
|
196
|
-
)
|
|
197
|
-
.option(
|
|
198
|
-
'--sort <order>',
|
|
199
|
-
'Sort by newest, downloads, rating, installs, installsAllTime, or trending',
|
|
200
|
-
'newest',
|
|
201
|
-
)
|
|
202
|
-
.option('--json', 'Output JSON')
|
|
203
|
-
.action(async (options) => {
|
|
204
|
-
const opts = await resolveGlobalOpts()
|
|
205
|
-
const limit =
|
|
206
|
-
typeof options.limit === 'number' && Number.isFinite(options.limit) ? options.limit : 25
|
|
207
|
-
await cmdExplore(opts, { limit, sort: options.sort, json: options.json })
|
|
208
|
-
})
|
|
209
|
-
|
|
210
|
-
program
|
|
211
|
-
.command('publish')
|
|
212
|
-
.description('Publish skill from folder')
|
|
213
|
-
.argument('<path>', 'Skill folder path')
|
|
214
|
-
.option('--slug <slug>', 'Skill slug')
|
|
215
|
-
.option('--name <name>', 'Display name')
|
|
216
|
-
.option('--version <version>', 'Version (semver)')
|
|
217
|
-
.option('--fork-of <slug[@version]>', 'Mark as a fork of an existing skill')
|
|
218
|
-
.option('--changelog <text>', 'Changelog text')
|
|
219
|
-
.option('--tags <tags>', 'Comma-separated tags', 'latest')
|
|
220
|
-
.action(async (folder, options) => {
|
|
221
|
-
const opts = await resolveGlobalOpts()
|
|
222
|
-
await cmdPublish(opts, folder, options)
|
|
223
|
-
})
|
|
224
|
-
|
|
225
|
-
program
|
|
226
|
-
.command('delete')
|
|
227
|
-
.description('Soft-delete a skill (owner/admin only)')
|
|
228
|
-
.argument('<slug>', 'Skill slug')
|
|
229
|
-
.option('--yes', 'Skip confirmation')
|
|
230
|
-
.action(async (slug, options) => {
|
|
231
|
-
const opts = await resolveGlobalOpts()
|
|
232
|
-
await cmdDeleteSkill(opts, slug, options, isInputAllowed())
|
|
233
|
-
})
|
|
234
|
-
|
|
235
|
-
program
|
|
236
|
-
.command('undelete')
|
|
237
|
-
.description('Restore a soft-deleted skill (owner/admin only)')
|
|
238
|
-
.argument('<slug>', 'Skill slug')
|
|
239
|
-
.option('--yes', 'Skip confirmation')
|
|
240
|
-
.action(async (slug, options) => {
|
|
241
|
-
const opts = await resolveGlobalOpts()
|
|
242
|
-
await cmdUndeleteSkill(opts, slug, options, isInputAllowed())
|
|
243
|
-
})
|
|
244
|
-
|
|
245
|
-
program
|
|
246
|
-
.command('star')
|
|
247
|
-
.description('Add a skill to your highlights')
|
|
248
|
-
.argument('<slug>', 'Skill slug')
|
|
249
|
-
.option('--yes', 'Skip confirmation')
|
|
250
|
-
.action(async (slug, options) => {
|
|
251
|
-
const opts = await resolveGlobalOpts()
|
|
252
|
-
await cmdStarSkill(opts, slug, options, isInputAllowed())
|
|
253
|
-
})
|
|
254
|
-
|
|
255
|
-
program
|
|
256
|
-
.command('unstar')
|
|
257
|
-
.description('Remove a skill from your highlights')
|
|
258
|
-
.argument('<slug>', 'Skill slug')
|
|
259
|
-
.option('--yes', 'Skip confirmation')
|
|
260
|
-
.action(async (slug, options) => {
|
|
261
|
-
const opts = await resolveGlobalOpts()
|
|
262
|
-
await cmdUnstarSkill(opts, slug, options, isInputAllowed())
|
|
263
|
-
})
|
|
264
|
-
|
|
265
|
-
program
|
|
266
|
-
.command('sync')
|
|
267
|
-
.description('Scan local skills and publish new/updated ones')
|
|
268
|
-
.option('--root <dir...>', 'Extra scan roots (one or more)')
|
|
269
|
-
.option('--all', 'Upload all new/updated skills without prompting')
|
|
270
|
-
.option('--dry-run', 'Show what would be uploaded')
|
|
271
|
-
.option('--bump <type>', 'Version bump for updates (patch|minor|major)', 'patch')
|
|
272
|
-
.option('--changelog <text>', 'Changelog to use for updates (non-interactive)')
|
|
273
|
-
.option('--tags <tags>', 'Comma-separated tags', 'latest')
|
|
274
|
-
.option('--concurrency <n>', 'Concurrent registry checks (default: 4)', '4')
|
|
275
|
-
.action(async (options) => {
|
|
276
|
-
const opts = await resolveGlobalOpts()
|
|
277
|
-
const bump = String(options.bump ?? 'patch') as 'patch' | 'minor' | 'major'
|
|
278
|
-
if (!['patch', 'minor', 'major'].includes(bump)) fail('--bump must be patch|minor|major')
|
|
279
|
-
const concurrencyRaw = Number(options.concurrency ?? 4)
|
|
280
|
-
const concurrency = Number.isFinite(concurrencyRaw) ? Math.round(concurrencyRaw) : 4
|
|
281
|
-
if (concurrency < 1 || concurrency > 32) fail('--concurrency must be between 1 and 32')
|
|
282
|
-
await cmdSync(
|
|
283
|
-
opts,
|
|
284
|
-
{
|
|
285
|
-
root: options.root,
|
|
286
|
-
all: options.all,
|
|
287
|
-
dryRun: options.dryRun,
|
|
288
|
-
bump,
|
|
289
|
-
changelog: options.changelog,
|
|
290
|
-
tags: options.tags,
|
|
291
|
-
concurrency,
|
|
292
|
-
},
|
|
293
|
-
isInputAllowed(),
|
|
294
|
-
)
|
|
295
|
-
})
|
|
296
|
-
|
|
297
|
-
program.action(async () => {
|
|
298
|
-
const opts = await resolveGlobalOpts()
|
|
299
|
-
const cfg = await readGlobalConfig()
|
|
300
|
-
if (cfg?.token) {
|
|
301
|
-
await cmdSync(opts, {}, isInputAllowed())
|
|
302
|
-
return
|
|
303
|
-
}
|
|
304
|
-
program.outputHelp()
|
|
305
|
-
process.exitCode = 0
|
|
306
|
-
})
|
|
307
|
-
|
|
308
|
-
void program.parseAsync(process.argv).catch((error) => {
|
|
309
|
-
const message = error instanceof Error ? error.message : String(error)
|
|
310
|
-
fail(message)
|
|
311
|
-
})
|
|
@@ -1,36 +0,0 @@
|
|
|
1
|
-
import { mkdir, readFile, writeFile } from 'node:fs/promises'
|
|
2
|
-
import { homedir } from 'node:os'
|
|
3
|
-
import { dirname, join, resolve } from 'node:path'
|
|
4
|
-
import { type GlobalConfig, GlobalConfigSchema, parseArk } from './schema/index.js'
|
|
5
|
-
|
|
6
|
-
export function getGlobalConfigPath() {
|
|
7
|
-
const override = process.env.PILOTHUB_CONFIG_PATH?.trim()
|
|
8
|
-
if (override) return resolve(override)
|
|
9
|
-
const home = homedir()
|
|
10
|
-
if (process.platform === 'darwin') {
|
|
11
|
-
return join(home, 'Library', 'Application Support', 'pilothub', 'config.json')
|
|
12
|
-
}
|
|
13
|
-
const xdg = process.env.XDG_CONFIG_HOME
|
|
14
|
-
if (xdg) return join(xdg, 'pilothub', 'config.json')
|
|
15
|
-
if (process.platform === 'win32') {
|
|
16
|
-
const appData = process.env.APPDATA
|
|
17
|
-
if (appData) return join(appData, 'pilothub', 'config.json')
|
|
18
|
-
}
|
|
19
|
-
return join(home, '.config', 'pilothub', 'config.json')
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export async function readGlobalConfig(): Promise<GlobalConfig | null> {
|
|
23
|
-
try {
|
|
24
|
-
const raw = await readFile(getGlobalConfigPath(), 'utf8')
|
|
25
|
-
const parsed = JSON.parse(raw) as unknown
|
|
26
|
-
return parseArk(GlobalConfigSchema, parsed, 'Global config')
|
|
27
|
-
} catch {
|
|
28
|
-
return null
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
export async function writeGlobalConfig(config: GlobalConfig) {
|
|
33
|
-
const path = getGlobalConfigPath()
|
|
34
|
-
await mkdir(dirname(path), { recursive: true })
|
|
35
|
-
await writeFile(path, `${JSON.stringify(config, null, 2)}\n`, 'utf8')
|
|
36
|
-
}
|
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
4
|
-
import { discoverRegistryFromSite } from './discovery'
|
|
5
|
-
|
|
6
|
-
describe('discovery', () => {
|
|
7
|
-
afterEach(() => {
|
|
8
|
-
vi.unstubAllGlobals()
|
|
9
|
-
})
|
|
10
|
-
|
|
11
|
-
it('returns null on non-ok response', async () => {
|
|
12
|
-
vi.stubGlobal(
|
|
13
|
-
'fetch',
|
|
14
|
-
vi.fn(async () => new Response('nope', { status: 404 })) as unknown as typeof fetch,
|
|
15
|
-
)
|
|
16
|
-
await expect(discoverRegistryFromSite('https://example.com')).resolves.toBeNull()
|
|
17
|
-
})
|
|
18
|
-
|
|
19
|
-
it('parses registry config', async () => {
|
|
20
|
-
vi.stubGlobal(
|
|
21
|
-
'fetch',
|
|
22
|
-
vi.fn(
|
|
23
|
-
async () =>
|
|
24
|
-
new Response(JSON.stringify({ registry: 'https://example.convex.site' }), {
|
|
25
|
-
status: 200,
|
|
26
|
-
headers: { 'Content-Type': 'application/json' },
|
|
27
|
-
}),
|
|
28
|
-
) as unknown as typeof fetch,
|
|
29
|
-
)
|
|
30
|
-
await expect(discoverRegistryFromSite('https://example.com')).resolves.toEqual({
|
|
31
|
-
apiBase: 'https://example.convex.site',
|
|
32
|
-
authBase: undefined,
|
|
33
|
-
minCliVersion: undefined,
|
|
34
|
-
})
|
|
35
|
-
})
|
|
36
|
-
|
|
37
|
-
it('parses apiBase config', async () => {
|
|
38
|
-
vi.stubGlobal(
|
|
39
|
-
'fetch',
|
|
40
|
-
vi.fn(
|
|
41
|
-
async () =>
|
|
42
|
-
new Response(
|
|
43
|
-
JSON.stringify({
|
|
44
|
-
apiBase: 'https://api.example.com',
|
|
45
|
-
authBase: 'https://auth.example.com',
|
|
46
|
-
minCliVersion: '1.2.3',
|
|
47
|
-
}),
|
|
48
|
-
{
|
|
49
|
-
status: 200,
|
|
50
|
-
headers: { 'Content-Type': 'application/json' },
|
|
51
|
-
},
|
|
52
|
-
),
|
|
53
|
-
) as unknown as typeof fetch,
|
|
54
|
-
)
|
|
55
|
-
await expect(discoverRegistryFromSite('https://example.com')).resolves.toEqual({
|
|
56
|
-
apiBase: 'https://api.example.com',
|
|
57
|
-
authBase: 'https://auth.example.com',
|
|
58
|
-
minCliVersion: '1.2.3',
|
|
59
|
-
})
|
|
60
|
-
})
|
|
61
|
-
|
|
62
|
-
it('returns null when apiBase is empty', async () => {
|
|
63
|
-
vi.stubGlobal(
|
|
64
|
-
'fetch',
|
|
65
|
-
vi.fn(
|
|
66
|
-
async () =>
|
|
67
|
-
new Response(JSON.stringify({ apiBase: '' }), {
|
|
68
|
-
status: 200,
|
|
69
|
-
headers: { 'Content-Type': 'application/json' },
|
|
70
|
-
}),
|
|
71
|
-
) as unknown as typeof fetch,
|
|
72
|
-
)
|
|
73
|
-
await expect(discoverRegistryFromSite('https://example.com')).resolves.toBeNull()
|
|
74
|
-
})
|
|
75
|
-
})
|
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
import { parseArk, WellKnownConfigSchema } from './schema/index.js'
|
|
2
|
-
|
|
3
|
-
export async function discoverRegistryFromSite(siteUrl: string) {
|
|
4
|
-
const url = new URL('/.well-known/pilothub.json', siteUrl)
|
|
5
|
-
const response = await fetch(url.toString(), {
|
|
6
|
-
method: 'GET',
|
|
7
|
-
headers: { Accept: 'application/json' },
|
|
8
|
-
})
|
|
9
|
-
if (!response.ok) return null
|
|
10
|
-
const raw = (await response.json()) as unknown
|
|
11
|
-
const parsed = parseArk(WellKnownConfigSchema, raw, 'WellKnown config')
|
|
12
|
-
const apiBase = 'apiBase' in parsed ? parsed.apiBase : parsed.registry
|
|
13
|
-
if (!apiBase) return null
|
|
14
|
-
return {
|
|
15
|
-
apiBase,
|
|
16
|
-
authBase: parsed.authBase,
|
|
17
|
-
minCliVersion: parsed.minCliVersion,
|
|
18
|
-
}
|
|
19
|
-
}
|
|
@@ -1,156 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { describe, expect, it, vi } from 'vitest'
|
|
4
|
-
import { apiRequest, apiRequestForm, downloadZip } from './http.js'
|
|
5
|
-
import { ApiV1WhoamiResponseSchema } from './schema/index.js'
|
|
6
|
-
|
|
7
|
-
describe('apiRequest', () => {
|
|
8
|
-
it('adds bearer token and parses json', async () => {
|
|
9
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
10
|
-
ok: true,
|
|
11
|
-
json: async () => ({ user: { handle: null } }),
|
|
12
|
-
})
|
|
13
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
14
|
-
const result = await apiRequest(
|
|
15
|
-
'https://example.com',
|
|
16
|
-
{ method: 'GET', path: '/x', token: 'clh_token' },
|
|
17
|
-
ApiV1WhoamiResponseSchema,
|
|
18
|
-
)
|
|
19
|
-
expect(result.user.handle).toBeNull()
|
|
20
|
-
expect(fetchMock).toHaveBeenCalledTimes(1)
|
|
21
|
-
const [, init] = fetchMock.mock.calls[0] as [string, RequestInit]
|
|
22
|
-
expect((init.headers as Record<string, string>).Authorization).toBe('Bearer clh_token')
|
|
23
|
-
vi.unstubAllGlobals()
|
|
24
|
-
})
|
|
25
|
-
|
|
26
|
-
it('posts json body', async () => {
|
|
27
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
28
|
-
ok: true,
|
|
29
|
-
json: async () => ({ ok: true }),
|
|
30
|
-
})
|
|
31
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
32
|
-
await apiRequest('https://example.com', {
|
|
33
|
-
method: 'POST',
|
|
34
|
-
path: '/x',
|
|
35
|
-
body: { a: 1 },
|
|
36
|
-
})
|
|
37
|
-
const [url, init] = fetchMock.mock.calls[0] as [string, RequestInit]
|
|
38
|
-
expect(url).toBe('https://example.com/x')
|
|
39
|
-
expect(init.body).toBe(JSON.stringify({ a: 1 }))
|
|
40
|
-
expect((init.headers as Record<string, string>)['Content-Type']).toBe('application/json')
|
|
41
|
-
vi.unstubAllGlobals()
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
it('throws text body on non-200', async () => {
|
|
45
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
46
|
-
ok: false,
|
|
47
|
-
status: 400,
|
|
48
|
-
text: async () => 'bad',
|
|
49
|
-
})
|
|
50
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
51
|
-
await expect(apiRequest('https://example.com', { method: 'GET', path: '/x' })).rejects.toThrow(
|
|
52
|
-
'bad',
|
|
53
|
-
)
|
|
54
|
-
vi.unstubAllGlobals()
|
|
55
|
-
})
|
|
56
|
-
|
|
57
|
-
it('falls back to HTTP status when body is empty', async () => {
|
|
58
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
59
|
-
ok: false,
|
|
60
|
-
status: 500,
|
|
61
|
-
text: async () => '',
|
|
62
|
-
})
|
|
63
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
64
|
-
await expect(
|
|
65
|
-
apiRequest('https://example.com', { method: 'GET', url: 'https://example.com/x' }),
|
|
66
|
-
).rejects.toThrow('HTTP 500')
|
|
67
|
-
vi.unstubAllGlobals()
|
|
68
|
-
})
|
|
69
|
-
|
|
70
|
-
it('downloads zip bytes', async () => {
|
|
71
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
72
|
-
ok: true,
|
|
73
|
-
arrayBuffer: async () => new Uint8Array([1, 2, 3]).buffer,
|
|
74
|
-
})
|
|
75
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
76
|
-
const bytes = await downloadZip('https://example.com', { slug: 'demo', version: '1.0.0' })
|
|
77
|
-
expect(Array.from(bytes)).toEqual([1, 2, 3])
|
|
78
|
-
const [url] = fetchMock.mock.calls[0] as [string]
|
|
79
|
-
expect(url).toContain('slug=demo')
|
|
80
|
-
expect(url).toContain('version=1.0.0')
|
|
81
|
-
vi.unstubAllGlobals()
|
|
82
|
-
})
|
|
83
|
-
|
|
84
|
-
it('does not retry on non-retryable errors', async () => {
|
|
85
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
86
|
-
ok: false,
|
|
87
|
-
status: 404,
|
|
88
|
-
text: async () => 'nope',
|
|
89
|
-
})
|
|
90
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
91
|
-
await expect(downloadZip('https://example.com', { slug: 'demo' })).rejects.toThrow('nope')
|
|
92
|
-
expect(fetchMock).toHaveBeenCalledTimes(1)
|
|
93
|
-
vi.unstubAllGlobals()
|
|
94
|
-
})
|
|
95
|
-
})
|
|
96
|
-
|
|
97
|
-
describe('apiRequestForm', () => {
|
|
98
|
-
it('posts form data and returns json', async () => {
|
|
99
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
100
|
-
ok: true,
|
|
101
|
-
json: async () => ({ ok: true }),
|
|
102
|
-
})
|
|
103
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
104
|
-
const form = new FormData()
|
|
105
|
-
form.append('x', '1')
|
|
106
|
-
const result = await apiRequestForm('https://example.com', {
|
|
107
|
-
method: 'POST',
|
|
108
|
-
path: '/upload',
|
|
109
|
-
token: 'clh_token',
|
|
110
|
-
form,
|
|
111
|
-
})
|
|
112
|
-
expect(result).toEqual({ ok: true })
|
|
113
|
-
const [, init] = fetchMock.mock.calls[0] as [string, RequestInit]
|
|
114
|
-
expect(init.body).toBe(form)
|
|
115
|
-
expect((init.headers as Record<string, string>).Authorization).toBe('Bearer clh_token')
|
|
116
|
-
vi.unstubAllGlobals()
|
|
117
|
-
})
|
|
118
|
-
|
|
119
|
-
it('retries on 429', async () => {
|
|
120
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
121
|
-
ok: false,
|
|
122
|
-
status: 429,
|
|
123
|
-
text: async () => 'rate limited',
|
|
124
|
-
})
|
|
125
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
126
|
-
await expect(
|
|
127
|
-
apiRequestForm('https://example.com', {
|
|
128
|
-
method: 'POST',
|
|
129
|
-
path: '/upload',
|
|
130
|
-
form: new FormData(),
|
|
131
|
-
}),
|
|
132
|
-
).rejects.toThrow('rate limited')
|
|
133
|
-
expect(fetchMock).toHaveBeenCalledTimes(3)
|
|
134
|
-
vi.unstubAllGlobals()
|
|
135
|
-
})
|
|
136
|
-
|
|
137
|
-
it('falls back to HTTP status when body cannot be read', async () => {
|
|
138
|
-
const fetchMock = vi.fn().mockResolvedValue({
|
|
139
|
-
ok: false,
|
|
140
|
-
status: 400,
|
|
141
|
-
text: async () => {
|
|
142
|
-
throw new Error('boom')
|
|
143
|
-
},
|
|
144
|
-
})
|
|
145
|
-
vi.stubGlobal('fetch', fetchMock)
|
|
146
|
-
await expect(
|
|
147
|
-
apiRequestForm('https://example.com', {
|
|
148
|
-
method: 'POST',
|
|
149
|
-
path: '/upload',
|
|
150
|
-
form: new FormData(),
|
|
151
|
-
}),
|
|
152
|
-
).rejects.toThrow('HTTP 400')
|
|
153
|
-
expect(fetchMock).toHaveBeenCalledTimes(1)
|
|
154
|
-
vi.unstubAllGlobals()
|
|
155
|
-
})
|
|
156
|
-
})
|