pilothub 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.local.example +19 -0
- package/.github/workflows/ci.yml +40 -0
- package/.oxlintrc.json +3 -0
- package/AGENTS.md +45 -0
- package/CHANGELOG.md +138 -0
- package/DEPRECATIONS.md +7 -0
- package/LICENSE +21 -0
- package/README.md +150 -0
- package/biome.json +41 -0
- package/convex/_generated/api.d.ts +153 -0
- package/convex/_generated/api.js +23 -0
- package/convex/_generated/dataModel.d.ts +60 -0
- package/convex/_generated/server.d.ts +143 -0
- package/convex/_generated/server.js +93 -0
- package/convex/auth.config.ts +8 -0
- package/convex/auth.ts +19 -0
- package/convex/comments.ts +88 -0
- package/convex/crons.ts +34 -0
- package/convex/devSeed.ts +459 -0
- package/convex/devSeedExtra.ts +541 -0
- package/convex/downloads.ts +78 -0
- package/convex/githubBackups.ts +170 -0
- package/convex/githubBackupsNode.ts +183 -0
- package/convex/githubImport.ts +317 -0
- package/convex/githubSoulBackups.ts +170 -0
- package/convex/githubSoulBackupsNode.ts +186 -0
- package/convex/http.ts +194 -0
- package/convex/httpApi.handlers.test.ts +488 -0
- package/convex/httpApi.test.ts +70 -0
- package/convex/httpApi.ts +305 -0
- package/convex/httpApiV1.handlers.test.ts +584 -0
- package/convex/httpApiV1.ts +1172 -0
- package/convex/leaderboards.ts +39 -0
- package/convex/lib/access.ts +36 -0
- package/convex/lib/apiTokenAuth.ts +36 -0
- package/convex/lib/badges.ts +50 -0
- package/convex/lib/changelog.test.ts +34 -0
- package/convex/lib/changelog.ts +278 -0
- package/convex/lib/embeddings.ts +38 -0
- package/convex/lib/githubBackup.ts +443 -0
- package/convex/lib/githubImport.test.ts +247 -0
- package/convex/lib/githubImport.ts +425 -0
- package/convex/lib/githubSoulBackup.ts +443 -0
- package/convex/lib/leaderboards.ts +103 -0
- package/convex/lib/moderation.ts +42 -0
- package/convex/lib/public.ts +89 -0
- package/convex/lib/searchText.test.ts +46 -0
- package/convex/lib/searchText.ts +27 -0
- package/convex/lib/skillBackfill.test.ts +34 -0
- package/convex/lib/skillBackfill.ts +67 -0
- package/convex/lib/skillPublish.test.ts +28 -0
- package/convex/lib/skillPublish.ts +284 -0
- package/convex/lib/skillStats.ts +80 -0
- package/convex/lib/skills.test.ts +197 -0
- package/convex/lib/skills.ts +273 -0
- package/convex/lib/soulChangelog.ts +273 -0
- package/convex/lib/soulPublish.ts +236 -0
- package/convex/lib/tokens.test.ts +33 -0
- package/convex/lib/tokens.ts +51 -0
- package/convex/lib/webhooks.test.ts +91 -0
- package/convex/lib/webhooks.ts +112 -0
- package/convex/maintenance.test.ts +270 -0
- package/convex/maintenance.ts +840 -0
- package/convex/rateLimits.ts +50 -0
- package/convex/schema.ts +472 -0
- package/convex/search.test.ts +12 -0
- package/convex/search.ts +254 -0
- package/convex/seed.test.ts +37 -0
- package/convex/seed.ts +254 -0
- package/convex/seedSouls.ts +111 -0
- package/convex/skillStatEvents.ts +568 -0
- package/convex/skills.ts +1606 -0
- package/convex/soulComments.ts +88 -0
- package/convex/soulDownloads.ts +14 -0
- package/convex/soulStars.ts +71 -0
- package/convex/souls.ts +570 -0
- package/convex/stars.ts +108 -0
- package/convex/statsMaintenance.ts +205 -0
- package/convex/telemetry.ts +434 -0
- package/convex/tokens.ts +88 -0
- package/convex/tsconfig.json +7 -0
- package/convex/uploads.ts +20 -0
- package/convex/users.ts +122 -0
- package/convex/webhooks.ts +50 -0
- package/convex.json +3 -0
- package/docs/README.md +32 -0
- package/docs/api.md +51 -0
- package/docs/architecture.md +61 -0
- package/docs/auth.md +54 -0
- package/docs/cli.md +117 -0
- package/docs/deploy.md +78 -0
- package/docs/diffing.md +84 -0
- package/docs/github-import.md +171 -0
- package/docs/http-api.md +187 -0
- package/docs/manual-testing.md +64 -0
- package/docs/mintlify.md +43 -0
- package/docs/quickstart.md +120 -0
- package/docs/skill-format.md +58 -0
- package/docs/soul-format.md +37 -0
- package/docs/spec.md +177 -0
- package/docs/telemetry.md +91 -0
- package/docs/troubleshooting.md +49 -0
- package/docs/webhook.md +51 -0
- package/e2e/menu-smoke.pw.test.ts +49 -0
- package/e2e/pilothub.e2e.test.ts +494 -0
- package/e2e/search-exact.pw.test.ts +97 -0
- package/package.json +84 -0
- package/packages/pilothub/LICENSE +22 -0
- package/packages/pilothub/README.md +57 -0
- package/packages/pilothub/bin/pilothub.js +2 -0
- package/packages/pilothub/package.json +41 -0
- package/packages/pilothub/src/browserAuth.test.ts +96 -0
- package/packages/pilothub/src/browserAuth.ts +174 -0
- package/packages/pilothub/src/cli/buildInfo.ts +94 -0
- package/packages/pilothub/src/cli/commands/auth.ts +97 -0
- package/packages/pilothub/src/cli/commands/delete.test.ts +73 -0
- package/packages/pilothub/src/cli/commands/delete.ts +83 -0
- package/packages/pilothub/src/cli/commands/publish.test.ts +122 -0
- package/packages/pilothub/src/cli/commands/publish.ts +108 -0
- package/packages/pilothub/src/cli/commands/skills.test.ts +191 -0
- package/packages/pilothub/src/cli/commands/skills.ts +380 -0
- package/packages/pilothub/src/cli/commands/star.ts +46 -0
- package/packages/pilothub/src/cli/commands/sync.test.ts +310 -0
- package/packages/pilothub/src/cli/commands/sync.ts +200 -0
- package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +26 -0
- package/packages/pilothub/src/cli/commands/syncHelpers.ts +427 -0
- package/packages/pilothub/src/cli/commands/syncTypes.ts +27 -0
- package/packages/pilothub/src/cli/commands/unstar.ts +48 -0
- package/packages/pilothub/src/cli/helpStyle.ts +45 -0
- package/packages/pilothub/src/cli/pilotbotConfig.test.ts +159 -0
- package/packages/pilothub/src/cli/pilotbotConfig.ts +147 -0
- package/packages/pilothub/src/cli/registry.test.ts +63 -0
- package/packages/pilothub/src/cli/registry.ts +43 -0
- package/packages/pilothub/src/cli/scanSkills.test.ts +64 -0
- package/packages/pilothub/src/cli/scanSkills.ts +84 -0
- package/packages/pilothub/src/cli/slug.ts +16 -0
- package/packages/pilothub/src/cli/types.ts +12 -0
- package/packages/pilothub/src/cli/ui.ts +75 -0
- package/packages/pilothub/src/cli.ts +311 -0
- package/packages/pilothub/src/config.ts +36 -0
- package/packages/pilothub/src/discovery.test.ts +75 -0
- package/packages/pilothub/src/discovery.ts +19 -0
- package/packages/pilothub/src/http.test.ts +156 -0
- package/packages/pilothub/src/http.ts +301 -0
- package/packages/pilothub/src/schema/ark.ts +29 -0
- package/packages/pilothub/src/schema/index.ts +5 -0
- package/packages/pilothub/src/schema/routes.ts +22 -0
- package/packages/pilothub/src/schema/schemas.ts +260 -0
- package/packages/pilothub/src/schema/textFiles.test.ts +23 -0
- package/packages/pilothub/src/schema/textFiles.ts +66 -0
- package/packages/pilothub/src/skills.test.ts +191 -0
- package/packages/pilothub/src/skills.ts +172 -0
- package/packages/pilothub/src/types.ts +10 -0
- package/packages/pilothub/tsconfig.json +14 -0
- package/packages/schema/README.md +3 -0
- package/packages/schema/dist/ark.d.ts +4 -0
- package/packages/schema/dist/ark.js +26 -0
- package/packages/schema/dist/ark.js.map +1 -0
- package/packages/schema/dist/index.d.ts +5 -0
- package/packages/schema/dist/index.js +5 -0
- package/packages/schema/dist/index.js.map +1 -0
- package/packages/schema/dist/routes.d.ts +21 -0
- package/packages/schema/dist/routes.js +22 -0
- package/packages/schema/dist/routes.js.map +1 -0
- package/packages/schema/dist/schemas.d.ts +297 -0
- package/packages/schema/dist/schemas.js +243 -0
- package/packages/schema/dist/schemas.js.map +1 -0
- package/packages/schema/dist/textFiles.d.ts +5 -0
- package/packages/schema/dist/textFiles.js +66 -0
- package/packages/schema/dist/textFiles.js.map +1 -0
- package/packages/schema/package.json +26 -0
- package/packages/schema/src/ark.ts +29 -0
- package/packages/schema/src/index.ts +5 -0
- package/packages/schema/src/routes.ts +22 -0
- package/packages/schema/src/schemas.test.ts +123 -0
- package/packages/schema/src/schemas.ts +287 -0
- package/packages/schema/src/textFiles.test.ts +23 -0
- package/packages/schema/src/textFiles.ts +66 -0
- package/packages/schema/tsconfig.json +15 -0
- package/pilothub +46 -0
- package/playwright.config.ts +33 -0
- package/public/.well-known/pilothub.json +6 -0
- package/public/api/v1/openapi.json +379 -0
- package/public/favicon.ico +0 -0
- package/public/logo192.png +0 -0
- package/public/logo512.png +0 -0
- package/public/manifest.json +25 -0
- package/public/og.png +0 -0
- package/public/og.svg +98 -0
- package/public/pilot-logo.png +0 -0
- package/public/pilot-mark.png +0 -0
- package/public/robots.txt +3 -0
- package/public/tanstack-circle-logo.png +0 -0
- package/public/tanstack-word-logo-white.svg +1 -0
- package/scripts/check-peer-deps.ts +56 -0
- package/scripts/docs-list.ts +148 -0
- package/scripts/run-playwright-local.sh +14 -0
- package/server/og/fetchSkillOgMeta.ts +27 -0
- package/server/og/fetchSoulOgMeta.ts +27 -0
- package/server/og/ogAssets.ts +80 -0
- package/server/og/skillOgSvg.test.ts +59 -0
- package/server/og/skillOgSvg.ts +258 -0
- package/server/og/soulOgSvg.ts +209 -0
- package/server/routes/og/skill.png.ts +103 -0
- package/server/routes/og/soul.png.ts +111 -0
- package/src/__tests__/skill-detail-page.test.tsx +86 -0
- package/src/__tests__/skills-index.test.tsx +145 -0
- package/src/__tests__/upload.route.test.tsx +228 -0
- package/src/components/AppProviders.tsx +19 -0
- package/src/components/ClientOnly.tsx +18 -0
- package/src/components/Footer.tsx +29 -0
- package/src/components/Header.tsx +295 -0
- package/src/components/InstallSwitcher.tsx +53 -0
- package/src/components/SkillCard.tsx +36 -0
- package/src/components/SkillDetailPage.tsx +817 -0
- package/src/components/SkillDiffCard.tsx +485 -0
- package/src/components/SoulCard.tsx +19 -0
- package/src/components/SoulDetailPage.tsx +263 -0
- package/src/components/UserBootstrap.tsx +18 -0
- package/src/components/ui/dropdown-menu.tsx +67 -0
- package/src/components/ui/toggle-group.tsx +35 -0
- package/src/convex/client.ts +3 -0
- package/src/lib/badges.ts +29 -0
- package/src/lib/diffing.test.ts +163 -0
- package/src/lib/diffing.ts +106 -0
- package/src/lib/gravatar.test.ts +9 -0
- package/src/lib/gravatar.ts +158 -0
- package/src/lib/og.test.ts +142 -0
- package/src/lib/og.ts +156 -0
- package/src/lib/publicUser.ts +39 -0
- package/src/lib/roles.ts +19 -0
- package/src/lib/site.test.ts +130 -0
- package/src/lib/site.ts +84 -0
- package/src/lib/theme-transition.test.ts +134 -0
- package/src/lib/theme-transition.ts +134 -0
- package/src/lib/theme.test.tsx +88 -0
- package/src/lib/theme.ts +43 -0
- package/src/lib/uploadFiles.jsdom.test.ts +33 -0
- package/src/lib/uploadFiles.test.ts +123 -0
- package/src/lib/uploadFiles.ts +245 -0
- package/src/lib/uploadUtils.test.ts +78 -0
- package/src/lib/uploadUtils.ts +93 -0
- package/src/lib/useAuthStatus.ts +12 -0
- package/src/lib/utils.test.ts +9 -0
- package/src/lib/utils.ts +6 -0
- package/src/logo.svg +12 -0
- package/src/routeTree.gen.ts +345 -0
- package/src/router.tsx +17 -0
- package/src/routes/$owner/$slug.tsx +55 -0
- package/src/routes/__root.tsx +136 -0
- package/src/routes/admin.tsx +11 -0
- package/src/routes/cli/auth.tsx +168 -0
- package/src/routes/dashboard.tsx +97 -0
- package/src/routes/import.tsx +415 -0
- package/src/routes/index.tsx +252 -0
- package/src/routes/management.tsx +529 -0
- package/src/routes/settings.tsx +203 -0
- package/src/routes/skills/index.tsx +422 -0
- package/src/routes/souls/$slug.tsx +55 -0
- package/src/routes/souls/index.tsx +243 -0
- package/src/routes/stars.tsx +68 -0
- package/src/routes/u/$handle.tsx +307 -0
- package/src/routes/upload/utils.ts +81 -0
- package/src/routes/upload.tsx +499 -0
- package/src/styles.css +2718 -0
- package/tsconfig.json +24 -0
- package/tsconfig.oxlint.json +16 -0
- package/vercel.json +8 -0
- package/vite.config.ts +48 -0
- package/vitest.config.ts +47 -0
- package/vitest.e2e.config.ts +11 -0
- package/vitest.setup.ts +1 -0
|
@@ -0,0 +1,83 @@
|
|
|
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
|
+
}
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
/* @vitest-environment node */
|
|
2
|
+
|
|
3
|
+
import { mkdir, mkdtemp, rm, writeFile } from 'node:fs/promises'
|
|
4
|
+
import { tmpdir } from 'node:os'
|
|
5
|
+
import { join } from 'node:path'
|
|
6
|
+
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
7
|
+
import type { GlobalOpts } from '../types'
|
|
8
|
+
|
|
9
|
+
vi.mock('../../config.js', () => ({
|
|
10
|
+
readGlobalConfig: vi.fn(async () => ({ registry: 'https://pilothub.com', token: 'tkn' })),
|
|
11
|
+
}))
|
|
12
|
+
|
|
13
|
+
const mockGetRegistry = vi.fn(async (_opts: unknown, _params?: unknown) => 'https://pilothub.com')
|
|
14
|
+
vi.mock('../registry.js', () => ({
|
|
15
|
+
getRegistry: (opts: unknown, params?: unknown) => mockGetRegistry(opts, params),
|
|
16
|
+
}))
|
|
17
|
+
|
|
18
|
+
const mockApiRequestForm = vi.fn()
|
|
19
|
+
vi.mock('../../http.js', () => ({
|
|
20
|
+
apiRequestForm: (registry: unknown, args: unknown, schema?: unknown) =>
|
|
21
|
+
mockApiRequestForm(registry, args, schema),
|
|
22
|
+
}))
|
|
23
|
+
|
|
24
|
+
const mockFail = vi.fn((message: string) => {
|
|
25
|
+
throw new Error(message)
|
|
26
|
+
})
|
|
27
|
+
const mockSpinner = { text: '', succeed: vi.fn(), fail: vi.fn() }
|
|
28
|
+
vi.mock('../ui.js', () => ({
|
|
29
|
+
createSpinner: vi.fn(() => mockSpinner),
|
|
30
|
+
fail: (message: string) => mockFail(message),
|
|
31
|
+
formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
|
|
32
|
+
}))
|
|
33
|
+
|
|
34
|
+
const { cmdPublish } = await import('./publish')
|
|
35
|
+
|
|
36
|
+
async function makeTmpWorkdir() {
|
|
37
|
+
const root = await mkdtemp(join(tmpdir(), 'pilothub-publish-'))
|
|
38
|
+
return root
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function makeOpts(workdir: string): GlobalOpts {
|
|
42
|
+
return {
|
|
43
|
+
workdir,
|
|
44
|
+
dir: join(workdir, 'skills'),
|
|
45
|
+
site: 'https://pilothub.com',
|
|
46
|
+
registry: 'https://pilothub.com',
|
|
47
|
+
registrySource: 'default',
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
afterEach(() => {
|
|
52
|
+
vi.unstubAllGlobals()
|
|
53
|
+
vi.clearAllMocks()
|
|
54
|
+
})
|
|
55
|
+
|
|
56
|
+
describe('cmdPublish', () => {
|
|
57
|
+
it('publishes SKILL.md from disk (mocked HTTP)', async () => {
|
|
58
|
+
const workdir = await makeTmpWorkdir()
|
|
59
|
+
try {
|
|
60
|
+
const folder = join(workdir, 'my-skill')
|
|
61
|
+
await mkdir(folder, { recursive: true })
|
|
62
|
+
const skillContent = '# Skill\n\nHello\n'
|
|
63
|
+
const notesContent = 'notes\n'
|
|
64
|
+
await writeFile(join(folder, 'SKILL.md'), skillContent, 'utf8')
|
|
65
|
+
await writeFile(join(folder, 'notes.md'), notesContent, 'utf8')
|
|
66
|
+
|
|
67
|
+
mockApiRequestForm.mockResolvedValueOnce({ ok: true, skillId: 'skill_1', versionId: 'ver_1' })
|
|
68
|
+
|
|
69
|
+
await cmdPublish(makeOpts(workdir), 'my-skill', {
|
|
70
|
+
slug: 'my-skill',
|
|
71
|
+
name: 'My Skill',
|
|
72
|
+
version: '1.0.0',
|
|
73
|
+
changelog: '',
|
|
74
|
+
tags: 'latest',
|
|
75
|
+
})
|
|
76
|
+
|
|
77
|
+
const publishCall = mockApiRequestForm.mock.calls.find((call) => {
|
|
78
|
+
const req = call[1] as { path?: string } | undefined
|
|
79
|
+
return req?.path === '/api/v1/skills'
|
|
80
|
+
})
|
|
81
|
+
if (!publishCall) throw new Error('Missing publish call')
|
|
82
|
+
const publishForm = (publishCall[1] as { form?: FormData }).form as FormData
|
|
83
|
+
const payloadEntry = publishForm.get('payload')
|
|
84
|
+
if (typeof payloadEntry !== 'string') throw new Error('Missing publish payload')
|
|
85
|
+
const payload = JSON.parse(payloadEntry)
|
|
86
|
+
expect(payload.slug).toBe('my-skill')
|
|
87
|
+
expect(payload.displayName).toBe('My Skill')
|
|
88
|
+
expect(payload.version).toBe('1.0.0')
|
|
89
|
+
expect(payload.changelog).toBe('')
|
|
90
|
+
expect(payload.tags).toEqual(['latest'])
|
|
91
|
+
const files = publishForm.getAll('files') as Array<Blob & { name?: string }>
|
|
92
|
+
expect(files.map((file) => String(file.name ?? '')).sort()).toEqual(['SKILL.md', 'notes.md'])
|
|
93
|
+
} finally {
|
|
94
|
+
await rm(workdir, { recursive: true, force: true })
|
|
95
|
+
}
|
|
96
|
+
})
|
|
97
|
+
|
|
98
|
+
it('allows empty changelog when updating an existing skill', async () => {
|
|
99
|
+
const workdir = await makeTmpWorkdir()
|
|
100
|
+
try {
|
|
101
|
+
const folder = join(workdir, 'existing-skill')
|
|
102
|
+
await mkdir(folder, { recursive: true })
|
|
103
|
+
await writeFile(join(folder, 'SKILL.md'), '# Skill\n', 'utf8')
|
|
104
|
+
|
|
105
|
+
mockApiRequestForm.mockResolvedValueOnce({ ok: true, skillId: 'skill_1', versionId: 'ver_2' })
|
|
106
|
+
|
|
107
|
+
await cmdPublish(makeOpts(workdir), 'existing-skill', {
|
|
108
|
+
version: '1.0.1',
|
|
109
|
+
changelog: '',
|
|
110
|
+
tags: 'latest',
|
|
111
|
+
})
|
|
112
|
+
|
|
113
|
+
expect(mockApiRequestForm).toHaveBeenCalledWith(
|
|
114
|
+
expect.anything(),
|
|
115
|
+
expect.objectContaining({ path: '/api/v1/skills', method: 'POST' }),
|
|
116
|
+
expect.anything(),
|
|
117
|
+
)
|
|
118
|
+
} finally {
|
|
119
|
+
await rm(workdir, { recursive: true, force: true })
|
|
120
|
+
}
|
|
121
|
+
})
|
|
122
|
+
})
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { stat } from 'node:fs/promises'
|
|
2
|
+
import { basename, resolve } from 'node:path'
|
|
3
|
+
import semver from 'semver'
|
|
4
|
+
import { readGlobalConfig } from '../../config.js'
|
|
5
|
+
import { apiRequestForm } from '../../http.js'
|
|
6
|
+
import { ApiRoutes, ApiV1PublishResponseSchema } from '../../schema/index.js'
|
|
7
|
+
import { listTextFiles } from '../../skills.js'
|
|
8
|
+
import { getRegistry } from '../registry.js'
|
|
9
|
+
import { sanitizeSlug, titleCase } from '../slug.js'
|
|
10
|
+
import type { GlobalOpts } from '../types.js'
|
|
11
|
+
import { createSpinner, fail, formatError } from '../ui.js'
|
|
12
|
+
|
|
13
|
+
export async function cmdPublish(
|
|
14
|
+
opts: GlobalOpts,
|
|
15
|
+
folderArg: string,
|
|
16
|
+
options: {
|
|
17
|
+
slug?: string
|
|
18
|
+
name?: string
|
|
19
|
+
version?: string
|
|
20
|
+
changelog?: string
|
|
21
|
+
tags?: string
|
|
22
|
+
forkOf?: string
|
|
23
|
+
},
|
|
24
|
+
) {
|
|
25
|
+
const folder = folderArg ? resolve(opts.workdir, folderArg) : null
|
|
26
|
+
if (!folder) fail('Path required')
|
|
27
|
+
const folderStat = await stat(folder).catch(() => null)
|
|
28
|
+
if (!folderStat || !folderStat.isDirectory()) fail('Path must be a folder')
|
|
29
|
+
|
|
30
|
+
const cfg = await readGlobalConfig()
|
|
31
|
+
const token = cfg?.token
|
|
32
|
+
if (!token) fail('Not logged in. Run: pilothub login')
|
|
33
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
34
|
+
|
|
35
|
+
const slug = options.slug ?? sanitizeSlug(basename(folder))
|
|
36
|
+
const displayName = options.name ?? titleCase(basename(folder))
|
|
37
|
+
const version = options.version
|
|
38
|
+
const changelog = options.changelog ?? ''
|
|
39
|
+
const tagsValue = options.tags ?? 'latest'
|
|
40
|
+
const tags = tagsValue
|
|
41
|
+
.split(',')
|
|
42
|
+
.map((tag) => tag.trim())
|
|
43
|
+
.filter(Boolean)
|
|
44
|
+
|
|
45
|
+
const forkOfRaw = options.forkOf?.trim()
|
|
46
|
+
const forkOf = forkOfRaw ? parseForkOf(forkOfRaw) : undefined
|
|
47
|
+
|
|
48
|
+
if (!slug) fail('--slug required')
|
|
49
|
+
if (!displayName) fail('--name required')
|
|
50
|
+
if (!version || !semver.valid(version)) fail('--version must be valid semver')
|
|
51
|
+
|
|
52
|
+
const spinner = createSpinner(`Preparing ${slug}@${version}`)
|
|
53
|
+
try {
|
|
54
|
+
const filesOnDisk = await listTextFiles(folder)
|
|
55
|
+
if (filesOnDisk.length === 0) fail('No files found')
|
|
56
|
+
if (
|
|
57
|
+
!filesOnDisk.some((file) => {
|
|
58
|
+
const lower = file.relPath.toLowerCase()
|
|
59
|
+
return lower === 'skill.md' || lower === 'skills.md'
|
|
60
|
+
})
|
|
61
|
+
) {
|
|
62
|
+
fail('SKILL.md required')
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
const form = new FormData()
|
|
66
|
+
form.set(
|
|
67
|
+
'payload',
|
|
68
|
+
JSON.stringify({
|
|
69
|
+
slug,
|
|
70
|
+
displayName,
|
|
71
|
+
version,
|
|
72
|
+
changelog,
|
|
73
|
+
tags,
|
|
74
|
+
...(forkOf ? { forkOf } : {}),
|
|
75
|
+
}),
|
|
76
|
+
)
|
|
77
|
+
|
|
78
|
+
let index = 0
|
|
79
|
+
for (const file of filesOnDisk) {
|
|
80
|
+
index += 1
|
|
81
|
+
spinner.text = `Uploading ${file.relPath} (${index}/${filesOnDisk.length})`
|
|
82
|
+
const blob = new Blob([Buffer.from(file.bytes)], { type: file.contentType ?? 'text/plain' })
|
|
83
|
+
form.append('files', blob, file.relPath)
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
spinner.text = `Publishing ${slug}@${version}`
|
|
87
|
+
const result = await apiRequestForm(
|
|
88
|
+
registry,
|
|
89
|
+
{ method: 'POST', path: ApiRoutes.skills, token, form },
|
|
90
|
+
ApiV1PublishResponseSchema,
|
|
91
|
+
)
|
|
92
|
+
|
|
93
|
+
spinner.succeed(`OK. Published ${slug}@${version} (${result.versionId})`)
|
|
94
|
+
} catch (error) {
|
|
95
|
+
spinner.fail(formatError(error))
|
|
96
|
+
throw error
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
function parseForkOf(value: string) {
|
|
101
|
+
const trimmed = value.trim()
|
|
102
|
+
const [slugRaw, versionRaw] = trimmed.split('@')
|
|
103
|
+
const slug = (slugRaw ?? '').trim().toLowerCase()
|
|
104
|
+
if (!slug) fail('--fork-of must be <slug> or <slug@version>')
|
|
105
|
+
const version = (versionRaw ?? '').trim()
|
|
106
|
+
if (version && !semver.valid(version)) fail('--fork-of version must be valid semver')
|
|
107
|
+
return { slug, version: version || undefined }
|
|
108
|
+
}
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
/* @vitest-environment node */
|
|
2
|
+
|
|
3
|
+
import { afterEach, describe, expect, it, vi } from 'vitest'
|
|
4
|
+
import { ApiRoutes } from '../../schema/index.js'
|
|
5
|
+
import type { GlobalOpts } from '../types.js'
|
|
6
|
+
|
|
7
|
+
const mockApiRequest = vi.fn()
|
|
8
|
+
const mockDownloadZip = vi.fn()
|
|
9
|
+
vi.mock('../../http.js', () => ({
|
|
10
|
+
apiRequest: (...args: unknown[]) => mockApiRequest(...args),
|
|
11
|
+
downloadZip: (...args: unknown[]) => mockDownloadZip(...args),
|
|
12
|
+
}))
|
|
13
|
+
|
|
14
|
+
const mockGetRegistry = vi.fn(async () => 'https://pilothub.com')
|
|
15
|
+
vi.mock('../registry.js', () => ({
|
|
16
|
+
getRegistry: () => mockGetRegistry(),
|
|
17
|
+
}))
|
|
18
|
+
|
|
19
|
+
const mockSpinner = {
|
|
20
|
+
stop: vi.fn(),
|
|
21
|
+
fail: vi.fn(),
|
|
22
|
+
start: vi.fn(),
|
|
23
|
+
succeed: vi.fn(),
|
|
24
|
+
isSpinning: false,
|
|
25
|
+
text: '',
|
|
26
|
+
}
|
|
27
|
+
vi.mock('../ui.js', () => ({
|
|
28
|
+
createSpinner: vi.fn(() => mockSpinner),
|
|
29
|
+
fail: (message: string) => {
|
|
30
|
+
throw new Error(message)
|
|
31
|
+
},
|
|
32
|
+
formatError: (error: unknown) => (error instanceof Error ? error.message : String(error)),
|
|
33
|
+
isInteractive: () => false,
|
|
34
|
+
promptConfirm: vi.fn(async () => false),
|
|
35
|
+
}))
|
|
36
|
+
|
|
37
|
+
vi.mock('../../skills.js', () => ({
|
|
38
|
+
extractZipToDir: vi.fn(),
|
|
39
|
+
hashSkillFiles: vi.fn(),
|
|
40
|
+
listTextFiles: vi.fn(),
|
|
41
|
+
readLockfile: vi.fn(),
|
|
42
|
+
readSkillOrigin: vi.fn(),
|
|
43
|
+
writeLockfile: vi.fn(),
|
|
44
|
+
writeSkillOrigin: vi.fn(),
|
|
45
|
+
}))
|
|
46
|
+
|
|
47
|
+
vi.mock('node:fs/promises', () => ({
|
|
48
|
+
mkdir: vi.fn(),
|
|
49
|
+
rm: vi.fn(),
|
|
50
|
+
stat: vi.fn(),
|
|
51
|
+
}))
|
|
52
|
+
|
|
53
|
+
const { clampLimit, cmdExplore, cmdUpdate, formatExploreLine } = await import('./skills.js')
|
|
54
|
+
const {
|
|
55
|
+
extractZipToDir,
|
|
56
|
+
hashSkillFiles,
|
|
57
|
+
listTextFiles,
|
|
58
|
+
readLockfile,
|
|
59
|
+
readSkillOrigin,
|
|
60
|
+
writeLockfile,
|
|
61
|
+
writeSkillOrigin,
|
|
62
|
+
} = await import('../../skills.js')
|
|
63
|
+
const { rm, stat } = await import('node:fs/promises')
|
|
64
|
+
|
|
65
|
+
const mockLog = vi.spyOn(console, 'log').mockImplementation(() => {})
|
|
66
|
+
|
|
67
|
+
function makeOpts(): GlobalOpts {
|
|
68
|
+
return {
|
|
69
|
+
workdir: '/work',
|
|
70
|
+
dir: '/work/skills',
|
|
71
|
+
site: 'https://pilothub.com',
|
|
72
|
+
registry: 'https://pilothub.com',
|
|
73
|
+
registrySource: 'default',
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
afterEach(() => {
|
|
78
|
+
vi.clearAllMocks()
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
describe('explore helpers', () => {
|
|
82
|
+
it('clamps explore limits and handles non-finite values', () => {
|
|
83
|
+
expect(clampLimit(-5)).toBe(1)
|
|
84
|
+
expect(clampLimit(0)).toBe(1)
|
|
85
|
+
expect(clampLimit(1)).toBe(1)
|
|
86
|
+
expect(clampLimit(50)).toBe(50)
|
|
87
|
+
expect(clampLimit(99)).toBe(99)
|
|
88
|
+
expect(clampLimit(200)).toBe(200)
|
|
89
|
+
expect(clampLimit(250)).toBe(200)
|
|
90
|
+
expect(clampLimit(Number.NaN)).toBe(25)
|
|
91
|
+
expect(clampLimit(Number.POSITIVE_INFINITY)).toBe(25)
|
|
92
|
+
expect(clampLimit(Number.NaN, 10)).toBe(10)
|
|
93
|
+
})
|
|
94
|
+
|
|
95
|
+
it('formats explore lines with relative time and truncation', () => {
|
|
96
|
+
const now = 4 * 60 * 60 * 1000
|
|
97
|
+
const nowSpy = vi.spyOn(Date, 'now').mockReturnValue(now)
|
|
98
|
+
const summary = 'a'.repeat(60)
|
|
99
|
+
const line = formatExploreLine({
|
|
100
|
+
slug: 'weather',
|
|
101
|
+
summary,
|
|
102
|
+
updatedAt: now - 2 * 60 * 60 * 1000,
|
|
103
|
+
latestVersion: null,
|
|
104
|
+
})
|
|
105
|
+
expect(line).toBe(`weather v? 2h ago ${'a'.repeat(49)}…`)
|
|
106
|
+
nowSpy.mockRestore()
|
|
107
|
+
})
|
|
108
|
+
})
|
|
109
|
+
|
|
110
|
+
describe('cmdExplore', () => {
|
|
111
|
+
it('clamps limit and handles empty results', async () => {
|
|
112
|
+
mockApiRequest.mockResolvedValue({ items: [] })
|
|
113
|
+
|
|
114
|
+
await cmdExplore(makeOpts(), { limit: 0 })
|
|
115
|
+
|
|
116
|
+
const [, args] = mockApiRequest.mock.calls[0] ?? []
|
|
117
|
+
const url = new URL(String(args?.url))
|
|
118
|
+
expect(url.searchParams.get('limit')).toBe('1')
|
|
119
|
+
expect(mockLog).toHaveBeenCalledWith('No skills found.')
|
|
120
|
+
})
|
|
121
|
+
|
|
122
|
+
it('prints formatted results', async () => {
|
|
123
|
+
const now = 10 * 60 * 1000
|
|
124
|
+
const nowSpy = vi.spyOn(Date, 'now').mockReturnValue(now)
|
|
125
|
+
const item = {
|
|
126
|
+
slug: 'gog',
|
|
127
|
+
summary: 'Google Workspace CLI for Gmail, Calendar, Drive and more.',
|
|
128
|
+
updatedAt: now - 90 * 1000,
|
|
129
|
+
latestVersion: { version: '1.2.3' },
|
|
130
|
+
}
|
|
131
|
+
mockApiRequest.mockResolvedValue({ items: [item] })
|
|
132
|
+
|
|
133
|
+
await cmdExplore(makeOpts(), { limit: 250 })
|
|
134
|
+
|
|
135
|
+
const [, args] = mockApiRequest.mock.calls[0] ?? []
|
|
136
|
+
const url = new URL(String(args?.url))
|
|
137
|
+
expect(url.searchParams.get('limit')).toBe('200')
|
|
138
|
+
expect(mockLog).toHaveBeenCalledWith(formatExploreLine(item))
|
|
139
|
+
nowSpy.mockRestore()
|
|
140
|
+
})
|
|
141
|
+
|
|
142
|
+
it('supports sort and json output', async () => {
|
|
143
|
+
const payload = { items: [], nextCursor: null }
|
|
144
|
+
mockApiRequest.mockResolvedValue(payload)
|
|
145
|
+
|
|
146
|
+
await cmdExplore(makeOpts(), { limit: 10, sort: 'installs', json: true })
|
|
147
|
+
|
|
148
|
+
const [, args] = mockApiRequest.mock.calls[0] ?? []
|
|
149
|
+
const url = new URL(String(args?.url))
|
|
150
|
+
expect(url.searchParams.get('limit')).toBe('10')
|
|
151
|
+
expect(url.searchParams.get('sort')).toBe('installsCurrent')
|
|
152
|
+
expect(mockLog).toHaveBeenCalledWith(JSON.stringify(payload, null, 2))
|
|
153
|
+
})
|
|
154
|
+
|
|
155
|
+
it('supports all-time installs and trending sorts', async () => {
|
|
156
|
+
mockApiRequest.mockResolvedValue({ items: [], nextCursor: null })
|
|
157
|
+
|
|
158
|
+
await cmdExplore(makeOpts(), { limit: 5, sort: 'installsAllTime' })
|
|
159
|
+
await cmdExplore(makeOpts(), { limit: 5, sort: 'trending' })
|
|
160
|
+
|
|
161
|
+
const first = new URL(String(mockApiRequest.mock.calls[0]?.[1]?.url))
|
|
162
|
+
const second = new URL(String(mockApiRequest.mock.calls[1]?.[1]?.url))
|
|
163
|
+
expect(first.searchParams.get('sort')).toBe('installsAllTime')
|
|
164
|
+
expect(second.searchParams.get('sort')).toBe('trending')
|
|
165
|
+
})
|
|
166
|
+
})
|
|
167
|
+
|
|
168
|
+
describe('cmdUpdate', () => {
|
|
169
|
+
it('uses path-based skill lookup when no local fingerprint is available', async () => {
|
|
170
|
+
mockApiRequest.mockResolvedValue({ latestVersion: { version: '1.0.0' } })
|
|
171
|
+
mockDownloadZip.mockResolvedValue(new Uint8Array([1, 2, 3]))
|
|
172
|
+
vi.mocked(readLockfile).mockResolvedValue({
|
|
173
|
+
version: 1,
|
|
174
|
+
skills: { demo: { version: '0.1.0', installedAt: 123 } },
|
|
175
|
+
})
|
|
176
|
+
vi.mocked(writeLockfile).mockResolvedValue()
|
|
177
|
+
vi.mocked(readSkillOrigin).mockResolvedValue(null)
|
|
178
|
+
vi.mocked(writeSkillOrigin).mockResolvedValue()
|
|
179
|
+
vi.mocked(extractZipToDir).mockResolvedValue()
|
|
180
|
+
vi.mocked(listTextFiles).mockResolvedValue([])
|
|
181
|
+
vi.mocked(hashSkillFiles).mockReturnValue({ fingerprint: 'hash', files: [] })
|
|
182
|
+
vi.mocked(stat).mockRejectedValue(new Error('missing'))
|
|
183
|
+
vi.mocked(rm).mockResolvedValue()
|
|
184
|
+
|
|
185
|
+
await cmdUpdate(makeOpts(), 'demo', {}, false)
|
|
186
|
+
|
|
187
|
+
const [, args] = mockApiRequest.mock.calls[0] ?? []
|
|
188
|
+
expect(args?.path).toBe(`${ApiRoutes.skills}/${encodeURIComponent('demo')}`)
|
|
189
|
+
expect(args?.url).toBeUndefined()
|
|
190
|
+
})
|
|
191
|
+
})
|