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,41 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "pilothub",
|
|
3
|
-
"version": "0.3.0",
|
|
4
|
-
"description": "PilotHub CLI \\u2014 install, update, search, and publish agent skills.",
|
|
5
|
-
"license": "MIT",
|
|
6
|
-
"type": "module",
|
|
7
|
-
"bin": {
|
|
8
|
-
"pilothub": "bin/pilothub.js"
|
|
9
|
-
},
|
|
10
|
-
"files": [
|
|
11
|
-
"bin",
|
|
12
|
-
"dist",
|
|
13
|
-
"README.md",
|
|
14
|
-
"LICENSE"
|
|
15
|
-
],
|
|
16
|
-
"scripts": {
|
|
17
|
-
"build": "tsc -p tsconfig.json",
|
|
18
|
-
"dev": "node --enable-source-maps dist/cli.js",
|
|
19
|
-
"prepublishOnly": "npm run build"
|
|
20
|
-
},
|
|
21
|
-
"dependencies": {
|
|
22
|
-
"@clack/prompts": "^0.11.0",
|
|
23
|
-
"arktype": "^2.1.29",
|
|
24
|
-
"commander": "^14.0.2",
|
|
25
|
-
"fflate": "^0.8.2",
|
|
26
|
-
"ignore": "^7.0.5",
|
|
27
|
-
"json5": "^2.2.3",
|
|
28
|
-
"mime": "^4.1.0",
|
|
29
|
-
"ora": "^9.0.0",
|
|
30
|
-
"p-retry": "^7.1.1",
|
|
31
|
-
"semver": "^7.7.3",
|
|
32
|
-
"undici": "^7.16.0"
|
|
33
|
-
},
|
|
34
|
-
"devDependencies": {
|
|
35
|
-
"@types/node": "^25.0.9",
|
|
36
|
-
"typescript": "^5.9.3"
|
|
37
|
-
},
|
|
38
|
-
"engines": {
|
|
39
|
-
"node": ">=20"
|
|
40
|
-
}
|
|
41
|
-
}
|
|
@@ -1,96 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { describe, expect, it } from 'vitest'
|
|
4
|
-
import {
|
|
5
|
-
buildCliAuthUrl,
|
|
6
|
-
isAllowedLoopbackRedirectUri,
|
|
7
|
-
startLoopbackAuthServer,
|
|
8
|
-
} from './browserAuth'
|
|
9
|
-
|
|
10
|
-
describe('browserAuth', () => {
|
|
11
|
-
it('builds auth url', () => {
|
|
12
|
-
const url = buildCliAuthUrl({
|
|
13
|
-
siteUrl: 'https://example.com',
|
|
14
|
-
redirectUri: 'http://127.0.0.1:1234/callback',
|
|
15
|
-
label: 'CLI token',
|
|
16
|
-
state: 'state123',
|
|
17
|
-
})
|
|
18
|
-
expect(url).toContain('https://example.com/cli/auth?')
|
|
19
|
-
expect(url).toContain('redirect_uri=')
|
|
20
|
-
expect(url).toContain('label_b64=')
|
|
21
|
-
expect(url).toContain('state=')
|
|
22
|
-
})
|
|
23
|
-
|
|
24
|
-
it('builds auth url without label', () => {
|
|
25
|
-
const url = buildCliAuthUrl({
|
|
26
|
-
siteUrl: 'https://example.com',
|
|
27
|
-
redirectUri: 'http://127.0.0.1:1234/callback',
|
|
28
|
-
state: 'state123',
|
|
29
|
-
})
|
|
30
|
-
expect(url).toContain('https://example.com/cli/auth?')
|
|
31
|
-
expect(url).not.toContain('label_b64=')
|
|
32
|
-
})
|
|
33
|
-
|
|
34
|
-
it('accepts only loopback http redirect uris', () => {
|
|
35
|
-
expect(isAllowedLoopbackRedirectUri('http://127.0.0.1:1234/callback')).toBe(true)
|
|
36
|
-
expect(isAllowedLoopbackRedirectUri('http://localhost:1234/callback')).toBe(true)
|
|
37
|
-
expect(isAllowedLoopbackRedirectUri('http://[::1]:1234/callback')).toBe(true)
|
|
38
|
-
expect(isAllowedLoopbackRedirectUri('https://127.0.0.1:1234/callback')).toBe(false)
|
|
39
|
-
expect(isAllowedLoopbackRedirectUri('http://evil.com/callback')).toBe(false)
|
|
40
|
-
expect(isAllowedLoopbackRedirectUri('not a url')).toBe(false)
|
|
41
|
-
})
|
|
42
|
-
|
|
43
|
-
it('receives token via loopback server', async () => {
|
|
44
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
|
|
45
|
-
const payload = {
|
|
46
|
-
token: 'clh_test',
|
|
47
|
-
registry: 'https://example.convex.site',
|
|
48
|
-
state: server.state,
|
|
49
|
-
}
|
|
50
|
-
await fetch(server.redirectUri.replace('/callback', '/token'), {
|
|
51
|
-
method: 'POST',
|
|
52
|
-
headers: { 'Content-Type': 'application/json' },
|
|
53
|
-
body: JSON.stringify(payload),
|
|
54
|
-
})
|
|
55
|
-
await expect(server.waitForResult()).resolves.toEqual(payload)
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
it('serves callback html', async () => {
|
|
59
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
|
|
60
|
-
const response = await fetch(server.redirectUri)
|
|
61
|
-
expect(response.status).toBe(200)
|
|
62
|
-
const text = await response.text()
|
|
63
|
-
expect(text).toContain('PilotHub CLI Login')
|
|
64
|
-
server.close()
|
|
65
|
-
})
|
|
66
|
-
|
|
67
|
-
it('returns 404 for unknown routes', async () => {
|
|
68
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
|
|
69
|
-
const response = await fetch(server.redirectUri.replace('/callback', '/nope'))
|
|
70
|
-
expect(response.status).toBe(404)
|
|
71
|
-
server.close()
|
|
72
|
-
})
|
|
73
|
-
|
|
74
|
-
it('rejects invalid json payloads', async () => {
|
|
75
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
|
|
76
|
-
const tokenUrl = server.redirectUri.replace('/callback', '/token')
|
|
77
|
-
const response = await fetch(tokenUrl, { method: 'POST', body: '{' })
|
|
78
|
-
expect(response.status).toBe(400)
|
|
79
|
-
await expect(server.waitForResult()).rejects.toThrow()
|
|
80
|
-
})
|
|
81
|
-
|
|
82
|
-
it('rejects state mismatches', async () => {
|
|
83
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 2000 })
|
|
84
|
-
await fetch(server.redirectUri.replace('/callback', '/token'), {
|
|
85
|
-
method: 'POST',
|
|
86
|
-
headers: { 'Content-Type': 'application/json' },
|
|
87
|
-
body: JSON.stringify({ token: 'clh_test', registry: 'https://example.com', state: 'nope' }),
|
|
88
|
-
})
|
|
89
|
-
await expect(server.waitForResult()).rejects.toThrow(/state mismatch/i)
|
|
90
|
-
})
|
|
91
|
-
|
|
92
|
-
it('times out waiting for login', async () => {
|
|
93
|
-
const server = await startLoopbackAuthServer({ timeoutMs: 25 })
|
|
94
|
-
await expect(server.waitForResult()).rejects.toThrow(/timed out waiting for browser login/i)
|
|
95
|
-
})
|
|
96
|
-
})
|
|
@@ -1,174 +0,0 @@
|
|
|
1
|
-
import { createServer } from 'node:http'
|
|
2
|
-
import type { AddressInfo } from 'node:net'
|
|
3
|
-
|
|
4
|
-
export type LoopbackAuthResult = {
|
|
5
|
-
token: string
|
|
6
|
-
registry?: string
|
|
7
|
-
state?: string
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
export function buildCliAuthUrl(params: {
|
|
11
|
-
siteUrl: string
|
|
12
|
-
redirectUri: string
|
|
13
|
-
label?: string
|
|
14
|
-
state: string
|
|
15
|
-
}) {
|
|
16
|
-
const url = new URL('/cli/auth', params.siteUrl)
|
|
17
|
-
url.searchParams.set('redirect_uri', params.redirectUri)
|
|
18
|
-
if (params.label) url.searchParams.set('label_b64', encodeBase64Url(params.label))
|
|
19
|
-
url.searchParams.set('state', params.state)
|
|
20
|
-
return url.toString()
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
export function isAllowedLoopbackRedirectUri(value: string) {
|
|
24
|
-
let url: URL
|
|
25
|
-
try {
|
|
26
|
-
url = new URL(value)
|
|
27
|
-
} catch {
|
|
28
|
-
return false
|
|
29
|
-
}
|
|
30
|
-
if (url.protocol !== 'http:') return false
|
|
31
|
-
const host = url.hostname.toLowerCase()
|
|
32
|
-
if (host !== '127.0.0.1' && host !== 'localhost' && host !== '::1' && host !== '[::1]') {
|
|
33
|
-
return false
|
|
34
|
-
}
|
|
35
|
-
return true
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
export async function startLoopbackAuthServer(params?: { timeoutMs?: number }) {
|
|
39
|
-
const timeoutMs = params?.timeoutMs ?? 5 * 60_000
|
|
40
|
-
const expectedState = generateState()
|
|
41
|
-
|
|
42
|
-
let resolveToken: ((value: LoopbackAuthResult) => void) | null = null
|
|
43
|
-
let rejectToken: ((error: Error) => void) | null = null
|
|
44
|
-
const tokenPromise = new Promise<LoopbackAuthResult>((resolve, reject) => {
|
|
45
|
-
resolveToken = resolve
|
|
46
|
-
rejectToken = reject
|
|
47
|
-
})
|
|
48
|
-
|
|
49
|
-
const server = createServer((req, res) => {
|
|
50
|
-
const method = req.method ?? 'GET'
|
|
51
|
-
const url = req.url ?? '/'
|
|
52
|
-
|
|
53
|
-
if (method === 'GET' && (url === '/' || url.startsWith('/callback'))) {
|
|
54
|
-
res.statusCode = 200
|
|
55
|
-
res.setHeader('Content-Type', 'text/html; charset=utf-8')
|
|
56
|
-
res.end(CALLBACK_HTML)
|
|
57
|
-
return
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
if (method === 'POST' && url === '/token') {
|
|
61
|
-
const chunks: Uint8Array[] = []
|
|
62
|
-
req.on('data', (chunk) => chunks.push(chunk as Uint8Array))
|
|
63
|
-
req.on('end', () => {
|
|
64
|
-
try {
|
|
65
|
-
const raw = Buffer.concat(chunks).toString('utf8')
|
|
66
|
-
const parsed = JSON.parse(raw) as unknown
|
|
67
|
-
if (!parsed || typeof parsed !== 'object') throw new Error('invalid payload')
|
|
68
|
-
const token = (parsed as { token?: unknown }).token
|
|
69
|
-
const registry = (parsed as { registry?: unknown }).registry
|
|
70
|
-
const state = (parsed as { state?: unknown }).state
|
|
71
|
-
if (typeof token !== 'string' || !token.trim()) throw new Error('token required')
|
|
72
|
-
if (typeof state !== 'string' || state !== expectedState) {
|
|
73
|
-
throw new Error('state mismatch')
|
|
74
|
-
}
|
|
75
|
-
res.statusCode = 200
|
|
76
|
-
res.setHeader('Content-Type', 'application/json')
|
|
77
|
-
res.end(JSON.stringify({ ok: true }))
|
|
78
|
-
resolveToken?.({
|
|
79
|
-
token: token.trim(),
|
|
80
|
-
registry: typeof registry === 'string' ? registry : undefined,
|
|
81
|
-
state,
|
|
82
|
-
})
|
|
83
|
-
} catch (error) {
|
|
84
|
-
res.statusCode = 400
|
|
85
|
-
res.setHeader('Content-Type', 'application/json')
|
|
86
|
-
res.end(JSON.stringify({ ok: false }))
|
|
87
|
-
const message = error instanceof Error ? error.message : 'invalid payload'
|
|
88
|
-
rejectToken?.(new Error(message))
|
|
89
|
-
} finally {
|
|
90
|
-
server.close()
|
|
91
|
-
}
|
|
92
|
-
})
|
|
93
|
-
return
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
res.statusCode = 404
|
|
97
|
-
res.setHeader('Content-Type', 'text/plain; charset=utf-8')
|
|
98
|
-
res.end('Not found')
|
|
99
|
-
})
|
|
100
|
-
|
|
101
|
-
await new Promise<void>((resolve, reject) => {
|
|
102
|
-
server.once('error', reject)
|
|
103
|
-
server.listen(0, '127.0.0.1', () => resolve())
|
|
104
|
-
})
|
|
105
|
-
const address = server.address() as AddressInfo | null
|
|
106
|
-
if (!address) {
|
|
107
|
-
server.close()
|
|
108
|
-
throw new Error('Failed to bind loopback server')
|
|
109
|
-
}
|
|
110
|
-
const redirectUri = `http://127.0.0.1:${address.port}/callback`
|
|
111
|
-
|
|
112
|
-
const timeout = setTimeout(() => {
|
|
113
|
-
server.close()
|
|
114
|
-
rejectToken?.(new Error('Timed out waiting for browser login'))
|
|
115
|
-
}, timeoutMs)
|
|
116
|
-
tokenPromise.finally(() => clearTimeout(timeout)).catch(() => {})
|
|
117
|
-
|
|
118
|
-
return {
|
|
119
|
-
redirectUri,
|
|
120
|
-
state: expectedState,
|
|
121
|
-
waitForResult: () => tokenPromise,
|
|
122
|
-
close: () => server.close(),
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
const CALLBACK_HTML = `<!doctype html>
|
|
127
|
-
<html lang="en">
|
|
128
|
-
<meta charset="utf-8" />
|
|
129
|
-
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
130
|
-
<title>PilotHub CLI Login</title>
|
|
131
|
-
<style>
|
|
132
|
-
:root { color-scheme: light dark; }
|
|
133
|
-
body { font-family: ui-sans-serif, system-ui, -apple-system, Segoe UI, sans-serif; padding: 24px; }
|
|
134
|
-
.card { max-width: 560px; margin: 40px auto; padding: 18px 16px; border: 1px solid rgba(127,127,127,.35); border-radius: 12px; }
|
|
135
|
-
code { font-family: ui-monospace, SFMono-Regular, Menlo, monospace; }
|
|
136
|
-
</style>
|
|
137
|
-
<body>
|
|
138
|
-
<div class="card">
|
|
139
|
-
<h1 style="margin: 0 0 10px; font-size: 18px;">Completing login…</h1>
|
|
140
|
-
<p id="status" style="margin: 0; opacity: .8;">Waiting for token.</p>
|
|
141
|
-
</div>
|
|
142
|
-
<script>
|
|
143
|
-
const statusEl = document.getElementById('status')
|
|
144
|
-
const params = new URLSearchParams(location.hash.replace(/^#/, ''))
|
|
145
|
-
const token = params.get('token')
|
|
146
|
-
const registry = params.get('registry')
|
|
147
|
-
const state = params.get('state')
|
|
148
|
-
if (!token) {
|
|
149
|
-
statusEl.textContent = 'Missing token in URL. You can close this tab and try again.'
|
|
150
|
-
} else if (!state) {
|
|
151
|
-
statusEl.textContent = 'Missing state in URL. You can close this tab and try again.'
|
|
152
|
-
} else {
|
|
153
|
-
fetch('/token', {
|
|
154
|
-
method: 'POST',
|
|
155
|
-
headers: { 'Content-Type': 'application/json' },
|
|
156
|
-
body: JSON.stringify({ token, registry, state }),
|
|
157
|
-
}).then(() => {
|
|
158
|
-
statusEl.textContent = 'Logged in. You can close this tab.'
|
|
159
|
-
setTimeout(() => window.close(), 250)
|
|
160
|
-
}).catch(() => {
|
|
161
|
-
statusEl.textContent = 'Failed to send token to CLI. You can close this tab and try again.'
|
|
162
|
-
})
|
|
163
|
-
}
|
|
164
|
-
</script>
|
|
165
|
-
</body>
|
|
166
|
-
</html>`
|
|
167
|
-
|
|
168
|
-
function encodeBase64Url(value: string) {
|
|
169
|
-
return Buffer.from(value, 'utf8').toString('base64url')
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
function generateState() {
|
|
173
|
-
return Buffer.from(crypto.getRandomValues(new Uint8Array(16))).toString('hex')
|
|
174
|
-
}
|
|
@@ -1,94 +0,0 @@
|
|
|
1
|
-
import { existsSync, readFileSync, statSync } from 'node:fs'
|
|
2
|
-
import { dirname, join, resolve } from 'node:path'
|
|
3
|
-
import { fileURLToPath } from 'node:url'
|
|
4
|
-
|
|
5
|
-
type PackageJson = { version?: string }
|
|
6
|
-
|
|
7
|
-
function readPackageVersion() {
|
|
8
|
-
try {
|
|
9
|
-
const path = join(dirname(fileURLToPath(import.meta.url)), '../../package.json')
|
|
10
|
-
const raw = readFileSync(path, 'utf8')
|
|
11
|
-
const pkg = JSON.parse(raw) as PackageJson
|
|
12
|
-
return typeof pkg.version === 'string' ? pkg.version : '0.0.0'
|
|
13
|
-
} catch {
|
|
14
|
-
return '0.0.0'
|
|
15
|
-
}
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
function shortCommit(value: string) {
|
|
19
|
-
const trimmed = value.trim()
|
|
20
|
-
if (!trimmed) return null
|
|
21
|
-
if (trimmed.length <= 8) return trimmed
|
|
22
|
-
return trimmed.slice(0, 8)
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
export function getCliCommit() {
|
|
26
|
-
const candidates = [
|
|
27
|
-
process.env.PILOTHUB_COMMIT,
|
|
28
|
-
process.env.VERCEL_GIT_COMMIT_SHA,
|
|
29
|
-
process.env.GITHUB_SHA,
|
|
30
|
-
process.env.COMMIT_SHA,
|
|
31
|
-
]
|
|
32
|
-
for (const candidate of candidates) {
|
|
33
|
-
if (!candidate) continue
|
|
34
|
-
const short = shortCommit(candidate)
|
|
35
|
-
if (short) return short
|
|
36
|
-
}
|
|
37
|
-
return readGitCommitFromCwd()
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
export function getCliVersion() {
|
|
41
|
-
return readPackageVersion()
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
export function getCliBuildLabel() {
|
|
45
|
-
const version = getCliVersion()
|
|
46
|
-
const commit = getCliCommit()
|
|
47
|
-
return commit ? `v${version} (${commit})` : `v${version}`
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function readGitCommitFromCwd() {
|
|
51
|
-
try {
|
|
52
|
-
const gitDir = findGitDir(process.cwd())
|
|
53
|
-
if (!gitDir) return null
|
|
54
|
-
const headPath = join(gitDir, 'HEAD')
|
|
55
|
-
if (!existsSync(headPath)) return null
|
|
56
|
-
const head = readFileSync(headPath, 'utf8').trim()
|
|
57
|
-
if (!head) return null
|
|
58
|
-
if (!head.startsWith('ref:')) return shortCommit(head)
|
|
59
|
-
const ref = head.replace(/^ref:\s*/, '').trim()
|
|
60
|
-
if (!ref) return null
|
|
61
|
-
const refPath = join(gitDir, ref)
|
|
62
|
-
if (!existsSync(refPath)) return null
|
|
63
|
-
const sha = readFileSync(refPath, 'utf8').trim()
|
|
64
|
-
return shortCommit(sha)
|
|
65
|
-
} catch {
|
|
66
|
-
return null
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
function findGitDir(start: string) {
|
|
71
|
-
let current = resolve(start)
|
|
72
|
-
for (;;) {
|
|
73
|
-
const dotGit = join(current, '.git')
|
|
74
|
-
if (existsSync(dotGit)) {
|
|
75
|
-
try {
|
|
76
|
-
const stat = statSync(dotGit)
|
|
77
|
-
if (stat.isDirectory()) return dotGit
|
|
78
|
-
} catch {
|
|
79
|
-
// ignore
|
|
80
|
-
}
|
|
81
|
-
try {
|
|
82
|
-
const content = readFileSync(dotGit, 'utf8').trim()
|
|
83
|
-
const match = content.match(/^gitdir:\s*(.+)$/)
|
|
84
|
-
if (match?.[1]) return resolve(current, match[1])
|
|
85
|
-
} catch {
|
|
86
|
-
return dotGit
|
|
87
|
-
}
|
|
88
|
-
return dotGit
|
|
89
|
-
}
|
|
90
|
-
const parent = resolve(current, '..')
|
|
91
|
-
if (parent === current) return null
|
|
92
|
-
current = parent
|
|
93
|
-
}
|
|
94
|
-
}
|
|
@@ -1,97 +0,0 @@
|
|
|
1
|
-
import { buildCliAuthUrl, startLoopbackAuthServer } from '../../browserAuth.js'
|
|
2
|
-
import { readGlobalConfig, writeGlobalConfig } from '../../config.js'
|
|
3
|
-
import { discoverRegistryFromSite } from '../../discovery.js'
|
|
4
|
-
import { apiRequest } from '../../http.js'
|
|
5
|
-
import { ApiRoutes, ApiV1WhoamiResponseSchema } from '../../schema/index.js'
|
|
6
|
-
import { getRegistry } from '../registry.js'
|
|
7
|
-
import type { GlobalOpts } from '../types.js'
|
|
8
|
-
import { createSpinner, fail, formatError, openInBrowser, promptHidden } from '../ui.js'
|
|
9
|
-
|
|
10
|
-
export async function cmdLoginFlow(
|
|
11
|
-
opts: GlobalOpts,
|
|
12
|
-
options: { token?: string; label?: string; browser?: boolean },
|
|
13
|
-
inputAllowed: boolean,
|
|
14
|
-
) {
|
|
15
|
-
if (options.token) {
|
|
16
|
-
await cmdLogin(opts, options.token, inputAllowed)
|
|
17
|
-
return
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
if (options.browser === false) {
|
|
21
|
-
fail('Token required (use --token or remove --no-browser)')
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
const label = String(options.label ?? 'CLI token').trim() || 'CLI token'
|
|
25
|
-
const receiver = await startLoopbackAuthServer()
|
|
26
|
-
const discovery = await discoverRegistryFromSite(opts.site).catch(() => null)
|
|
27
|
-
const authBase = discovery?.authBase?.trim() || opts.site
|
|
28
|
-
const authUrl = buildCliAuthUrl({
|
|
29
|
-
siteUrl: authBase,
|
|
30
|
-
redirectUri: receiver.redirectUri,
|
|
31
|
-
label,
|
|
32
|
-
state: receiver.state,
|
|
33
|
-
})
|
|
34
|
-
|
|
35
|
-
console.log(`Opening browser: ${authUrl}`)
|
|
36
|
-
openInBrowser(authUrl)
|
|
37
|
-
|
|
38
|
-
const result = await receiver.waitForResult()
|
|
39
|
-
const registry = result.registry?.trim() || opts.registry
|
|
40
|
-
await cmdLogin({ ...opts, registry }, result.token, inputAllowed)
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export async function cmdLogin(
|
|
44
|
-
opts: GlobalOpts,
|
|
45
|
-
tokenFlag: string | undefined,
|
|
46
|
-
inputAllowed: boolean,
|
|
47
|
-
) {
|
|
48
|
-
if (!tokenFlag && !inputAllowed) fail('Token required (use --token or remove --no-input)')
|
|
49
|
-
|
|
50
|
-
const token = tokenFlag || (await promptHidden('PilotHub token: '))
|
|
51
|
-
if (!token) fail('Token required')
|
|
52
|
-
|
|
53
|
-
const registry = await getRegistry(opts, { cache: true })
|
|
54
|
-
const spinner = createSpinner('Verifying token')
|
|
55
|
-
try {
|
|
56
|
-
const whoami = await apiRequest(
|
|
57
|
-
registry,
|
|
58
|
-
{ method: 'GET', path: ApiRoutes.whoami, token },
|
|
59
|
-
ApiV1WhoamiResponseSchema,
|
|
60
|
-
)
|
|
61
|
-
if (!whoami.user) fail('Login failed')
|
|
62
|
-
|
|
63
|
-
await writeGlobalConfig({ registry, token })
|
|
64
|
-
const handle = whoami.user.handle ? `@${whoami.user.handle}` : 'unknown user'
|
|
65
|
-
spinner.succeed(`OK. Logged in as ${handle}.`)
|
|
66
|
-
} catch (error) {
|
|
67
|
-
spinner.fail(formatError(error))
|
|
68
|
-
throw error
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
export async function cmdLogout(opts: GlobalOpts) {
|
|
73
|
-
const cfg = await readGlobalConfig()
|
|
74
|
-
const registry = cfg?.registry || (await getRegistry(opts, { cache: true }))
|
|
75
|
-
await writeGlobalConfig({ registry, token: undefined })
|
|
76
|
-
console.log('OK. Logged out.')
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
export async function cmdWhoami(opts: GlobalOpts) {
|
|
80
|
-
const cfg = await readGlobalConfig()
|
|
81
|
-
const token = cfg?.token
|
|
82
|
-
if (!token) fail('Not logged in. Run: pilothub login')
|
|
83
|
-
const registry = await getRegistry(opts, { cache: true })
|
|
84
|
-
|
|
85
|
-
const spinner = createSpinner('Checking token')
|
|
86
|
-
try {
|
|
87
|
-
const whoami = await apiRequest(
|
|
88
|
-
registry,
|
|
89
|
-
{ method: 'GET', path: ApiRoutes.whoami, token },
|
|
90
|
-
ApiV1WhoamiResponseSchema,
|
|
91
|
-
)
|
|
92
|
-
spinner.succeed(whoami.user.handle ?? 'unknown')
|
|
93
|
-
} catch (error) {
|
|
94
|
-
spinner.fail(formatError(error))
|
|
95
|
-
throw error
|
|
96
|
-
}
|
|
97
|
-
}
|
|
@@ -1,73 +0,0 @@
|
|
|
1
|
-
/* @vitest-environment node */
|
|
2
|
-
|
|
3
|
-
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
4
|
-
import type { GlobalOpts } from '../types'
|
|
5
|
-
|
|
6
|
-
vi.mock('../../config.js', () => ({
|
|
7
|
-
readGlobalConfig: vi.fn(async () => ({ registry: 'https://pilothub.com', token: 'tkn' })),
|
|
8
|
-
}))
|
|
9
|
-
|
|
10
|
-
vi.mock('../registry.js', () => ({
|
|
11
|
-
getRegistry: vi.fn(async () => 'https://pilothub.com'),
|
|
12
|
-
}))
|
|
13
|
-
|
|
14
|
-
const mockApiRequest = vi.fn()
|
|
15
|
-
vi.mock('../../http.js', () => ({
|
|
16
|
-
apiRequest: (registry: unknown, args: unknown, schema?: unknown) =>
|
|
17
|
-
mockApiRequest(registry, args, schema),
|
|
18
|
-
}))
|
|
19
|
-
|
|
20
|
-
const mockFail = vi.fn((message: string) => {
|
|
21
|
-
throw new Error(message)
|
|
22
|
-
})
|
|
23
|
-
|
|
24
|
-
vi.mock('../ui.js', () => ({
|
|
25
|
-
createSpinner: vi.fn(() => ({ succeed: vi.fn(), fail: vi.fn() })),
|
|
26
|
-
fail: (message: string) => mockFail(message),
|
|
27
|
-
formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
|
|
28
|
-
isInteractive: () => false,
|
|
29
|
-
promptConfirm: vi.fn(async () => true),
|
|
30
|
-
}))
|
|
31
|
-
|
|
32
|
-
const { cmdDeleteSkill, cmdUndeleteSkill } = await import('./delete')
|
|
33
|
-
|
|
34
|
-
function makeOpts(): GlobalOpts {
|
|
35
|
-
return {
|
|
36
|
-
workdir: '/work',
|
|
37
|
-
dir: '/work/skills',
|
|
38
|
-
site: 'https://pilothub.com',
|
|
39
|
-
registry: 'https://pilothub.com',
|
|
40
|
-
registrySource: 'default',
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
afterEach(() => {
|
|
45
|
-
vi.clearAllMocks()
|
|
46
|
-
})
|
|
47
|
-
|
|
48
|
-
describe('delete/undelete', () => {
|
|
49
|
-
it('requires --yes when input is disabled', async () => {
|
|
50
|
-
await expect(cmdDeleteSkill(makeOpts(), 'demo', {}, false)).rejects.toThrow(/--yes/i)
|
|
51
|
-
await expect(cmdUndeleteSkill(makeOpts(), 'demo', {}, false)).rejects.toThrow(/--yes/i)
|
|
52
|
-
})
|
|
53
|
-
|
|
54
|
-
it('calls delete endpoint with --yes', async () => {
|
|
55
|
-
mockApiRequest.mockResolvedValueOnce({ ok: true })
|
|
56
|
-
await cmdDeleteSkill(makeOpts(), 'demo', { yes: true }, false)
|
|
57
|
-
expect(mockApiRequest).toHaveBeenCalledWith(
|
|
58
|
-
expect.anything(),
|
|
59
|
-
expect.objectContaining({ method: 'DELETE', path: '/api/v1/skills/demo' }),
|
|
60
|
-
expect.anything(),
|
|
61
|
-
)
|
|
62
|
-
})
|
|
63
|
-
|
|
64
|
-
it('calls undelete endpoint with --yes', async () => {
|
|
65
|
-
mockApiRequest.mockResolvedValueOnce({ ok: true })
|
|
66
|
-
await cmdUndeleteSkill(makeOpts(), 'demo', { yes: true }, false)
|
|
67
|
-
expect(mockApiRequest).toHaveBeenCalledWith(
|
|
68
|
-
expect.anything(),
|
|
69
|
-
expect.objectContaining({ method: 'POST', path: '/api/v1/skills/demo/undelete' }),
|
|
70
|
-
expect.anything(),
|
|
71
|
-
)
|
|
72
|
-
})
|
|
73
|
-
})
|
|
@@ -1,83 +0,0 @@
|
|
|
1
|
-
import { readGlobalConfig } from '../../config.js'
|
|
2
|
-
import { apiRequest } from '../../http.js'
|
|
3
|
-
import { ApiRoutes, ApiV1DeleteResponseSchema, parseArk } from '../../schema/index.js'
|
|
4
|
-
import { getRegistry } from '../registry.js'
|
|
5
|
-
import type { GlobalOpts } from '../types.js'
|
|
6
|
-
import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js'
|
|
7
|
-
|
|
8
|
-
async function requireToken() {
|
|
9
|
-
const cfg = await readGlobalConfig()
|
|
10
|
-
const token = cfg?.token
|
|
11
|
-
if (!token) fail('Not logged in. Run: pilothub login')
|
|
12
|
-
return token
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export async function cmdDeleteSkill(
|
|
16
|
-
opts: GlobalOpts,
|
|
17
|
-
slugArg: string,
|
|
18
|
-
options: { yes?: boolean },
|
|
19
|
-
inputAllowed: boolean,
|
|
20
|
-
) {
|
|
21
|
-
const slug = slugArg.trim().toLowerCase()
|
|
22
|
-
if (!slug) fail('Slug required')
|
|
23
|
-
const allowPrompt = isInteractive() && inputAllowed !== false
|
|
24
|
-
|
|
25
|
-
if (!options.yes) {
|
|
26
|
-
if (!allowPrompt) fail('Pass --yes (no input)')
|
|
27
|
-
const ok = await promptConfirm(`Delete ${slug}? (soft delete)`)
|
|
28
|
-
if (!ok) return
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
const token = await requireToken()
|
|
32
|
-
const registry = await getRegistry(opts, { cache: true })
|
|
33
|
-
const spinner = createSpinner(`Deleting ${slug}`)
|
|
34
|
-
try {
|
|
35
|
-
const result = await apiRequest(
|
|
36
|
-
registry,
|
|
37
|
-
{ method: 'DELETE', path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}`, token },
|
|
38
|
-
ApiV1DeleteResponseSchema,
|
|
39
|
-
)
|
|
40
|
-
spinner.succeed(`OK. Deleted ${slug}`)
|
|
41
|
-
return parseArk(ApiV1DeleteResponseSchema, result, 'Delete response')
|
|
42
|
-
} catch (error) {
|
|
43
|
-
spinner.fail(formatError(error))
|
|
44
|
-
throw error
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
export async function cmdUndeleteSkill(
|
|
49
|
-
opts: GlobalOpts,
|
|
50
|
-
slugArg: string,
|
|
51
|
-
options: { yes?: boolean },
|
|
52
|
-
inputAllowed: boolean,
|
|
53
|
-
) {
|
|
54
|
-
const slug = slugArg.trim().toLowerCase()
|
|
55
|
-
if (!slug) fail('Slug required')
|
|
56
|
-
const allowPrompt = isInteractive() && inputAllowed !== false
|
|
57
|
-
|
|
58
|
-
if (!options.yes) {
|
|
59
|
-
if (!allowPrompt) fail('Pass --yes (no input)')
|
|
60
|
-
const ok = await promptConfirm(`Undelete ${slug}?`)
|
|
61
|
-
if (!ok) return
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
const token = await requireToken()
|
|
65
|
-
const registry = await getRegistry(opts, { cache: true })
|
|
66
|
-
const spinner = createSpinner(`Undeleting ${slug}`)
|
|
67
|
-
try {
|
|
68
|
-
const result = await apiRequest(
|
|
69
|
-
registry,
|
|
70
|
-
{
|
|
71
|
-
method: 'POST',
|
|
72
|
-
path: `${ApiRoutes.skills}/${encodeURIComponent(slug)}/undelete`,
|
|
73
|
-
token,
|
|
74
|
-
},
|
|
75
|
-
ApiV1DeleteResponseSchema,
|
|
76
|
-
)
|
|
77
|
-
spinner.succeed(`OK. Undeleted ${slug}`)
|
|
78
|
-
return parseArk(ApiV1DeleteResponseSchema, result, 'Undelete response')
|
|
79
|
-
} catch (error) {
|
|
80
|
-
spinner.fail(formatError(error))
|
|
81
|
-
throw error
|
|
82
|
-
}
|
|
83
|
-
}
|