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,380 @@
|
|
|
1
|
+
import { mkdir, rm, stat } from 'node:fs/promises'
|
|
2
|
+
import { join } from 'node:path'
|
|
3
|
+
import semver from 'semver'
|
|
4
|
+
import { apiRequest, downloadZip } from '../../http.js'
|
|
5
|
+
import {
|
|
6
|
+
ApiRoutes,
|
|
7
|
+
ApiV1SearchResponseSchema,
|
|
8
|
+
ApiV1SkillListResponseSchema,
|
|
9
|
+
ApiV1SkillResolveResponseSchema,
|
|
10
|
+
ApiV1SkillResponseSchema,
|
|
11
|
+
} from '../../schema/index.js'
|
|
12
|
+
import {
|
|
13
|
+
extractZipToDir,
|
|
14
|
+
hashSkillFiles,
|
|
15
|
+
listTextFiles,
|
|
16
|
+
readLockfile,
|
|
17
|
+
readSkillOrigin,
|
|
18
|
+
writeLockfile,
|
|
19
|
+
writeSkillOrigin,
|
|
20
|
+
} from '../../skills.js'
|
|
21
|
+
import { getRegistry } from '../registry.js'
|
|
22
|
+
import type { GlobalOpts, ResolveResult } from '../types.js'
|
|
23
|
+
import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js'
|
|
24
|
+
|
|
25
|
+
export async function cmdSearch(opts: GlobalOpts, query: string, limit?: number) {
|
|
26
|
+
if (!query) fail('Query required')
|
|
27
|
+
|
|
28
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
29
|
+
const spinner = createSpinner('Searching')
|
|
30
|
+
try {
|
|
31
|
+
const url = new URL(ApiRoutes.search, registry)
|
|
32
|
+
url.searchParams.set('q', query)
|
|
33
|
+
if (typeof limit === 'number' && Number.isFinite(limit)) {
|
|
34
|
+
url.searchParams.set('limit', String(limit))
|
|
35
|
+
}
|
|
36
|
+
const result = await apiRequest(
|
|
37
|
+
registry,
|
|
38
|
+
{ method: 'GET', url: url.toString() },
|
|
39
|
+
ApiV1SearchResponseSchema,
|
|
40
|
+
)
|
|
41
|
+
|
|
42
|
+
spinner.stop()
|
|
43
|
+
for (const entry of result.results) {
|
|
44
|
+
const slug = entry.slug ?? 'unknown'
|
|
45
|
+
const name = entry.displayName ?? slug
|
|
46
|
+
const version = entry.version ? ` v${entry.version}` : ''
|
|
47
|
+
console.log(`${slug}${version} ${name} (${entry.score.toFixed(3)})`)
|
|
48
|
+
}
|
|
49
|
+
} catch (error) {
|
|
50
|
+
spinner.fail(formatError(error))
|
|
51
|
+
throw error
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export async function cmdInstall(
|
|
56
|
+
opts: GlobalOpts,
|
|
57
|
+
slug: string,
|
|
58
|
+
versionFlag?: string,
|
|
59
|
+
force = false,
|
|
60
|
+
) {
|
|
61
|
+
const trimmed = slug.trim()
|
|
62
|
+
if (!trimmed) fail('Slug required')
|
|
63
|
+
|
|
64
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
65
|
+
await mkdir(opts.dir, { recursive: true })
|
|
66
|
+
const target = join(opts.dir, trimmed)
|
|
67
|
+
if (!force) {
|
|
68
|
+
const exists = await fileExists(target)
|
|
69
|
+
if (exists) fail(`Already installed: ${target} (use --force)`)
|
|
70
|
+
} else {
|
|
71
|
+
await rm(target, { recursive: true, force: true })
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const spinner = createSpinner(`Resolving ${trimmed}`)
|
|
75
|
+
try {
|
|
76
|
+
const resolvedVersion =
|
|
77
|
+
versionFlag ??
|
|
78
|
+
(
|
|
79
|
+
await apiRequest(
|
|
80
|
+
registry,
|
|
81
|
+
{ method: 'GET', path: `${ApiRoutes.skills}/${encodeURIComponent(trimmed)}` },
|
|
82
|
+
ApiV1SkillResponseSchema,
|
|
83
|
+
)
|
|
84
|
+
).latestVersion?.version ??
|
|
85
|
+
null
|
|
86
|
+
if (!resolvedVersion) fail('Could not resolve latest version')
|
|
87
|
+
|
|
88
|
+
spinner.text = `Downloading ${trimmed}@${resolvedVersion}`
|
|
89
|
+
const zip = await downloadZip(registry, { slug: trimmed, version: resolvedVersion })
|
|
90
|
+
await extractZipToDir(zip, target)
|
|
91
|
+
|
|
92
|
+
await writeSkillOrigin(target, {
|
|
93
|
+
version: 1,
|
|
94
|
+
registry,
|
|
95
|
+
slug: trimmed,
|
|
96
|
+
installedVersion: resolvedVersion,
|
|
97
|
+
installedAt: Date.now(),
|
|
98
|
+
})
|
|
99
|
+
|
|
100
|
+
const lock = await readLockfile(opts.workdir)
|
|
101
|
+
lock.skills[trimmed] = {
|
|
102
|
+
version: resolvedVersion,
|
|
103
|
+
installedAt: Date.now(),
|
|
104
|
+
}
|
|
105
|
+
await writeLockfile(opts.workdir, lock)
|
|
106
|
+
spinner.succeed(`OK. Installed ${trimmed} -> ${target}`)
|
|
107
|
+
} catch (error) {
|
|
108
|
+
spinner.fail(formatError(error))
|
|
109
|
+
throw error
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
export async function cmdUpdate(
|
|
114
|
+
opts: GlobalOpts,
|
|
115
|
+
slugArg: string | undefined,
|
|
116
|
+
options: { all?: boolean; version?: string; force?: boolean },
|
|
117
|
+
inputAllowed: boolean,
|
|
118
|
+
) {
|
|
119
|
+
const slug = slugArg?.trim()
|
|
120
|
+
const all = Boolean(options.all)
|
|
121
|
+
if (!slug && !all) fail('Provide <slug> or --all')
|
|
122
|
+
if (slug && all) fail('Use either <slug> or --all')
|
|
123
|
+
if (options.version && !slug) fail('--version requires a single <slug>')
|
|
124
|
+
if (options.version && !semver.valid(options.version)) fail('--version must be valid semver')
|
|
125
|
+
const allowPrompt = isInteractive() && inputAllowed !== false
|
|
126
|
+
|
|
127
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
128
|
+
const lock = await readLockfile(opts.workdir)
|
|
129
|
+
const slugs = slug ? [slug] : Object.keys(lock.skills)
|
|
130
|
+
if (slugs.length === 0) {
|
|
131
|
+
console.log('No installed skills.')
|
|
132
|
+
return
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
for (const entry of slugs) {
|
|
136
|
+
const spinner = createSpinner(`Checking ${entry}`)
|
|
137
|
+
try {
|
|
138
|
+
const target = join(opts.dir, entry)
|
|
139
|
+
const exists = await fileExists(target)
|
|
140
|
+
|
|
141
|
+
let localFingerprint: string | null = null
|
|
142
|
+
if (exists) {
|
|
143
|
+
const filesOnDisk = await listTextFiles(target)
|
|
144
|
+
if (filesOnDisk.length > 0) {
|
|
145
|
+
const hashed = hashSkillFiles(filesOnDisk)
|
|
146
|
+
localFingerprint = hashed.fingerprint
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
let resolveResult: ResolveResult
|
|
151
|
+
if (localFingerprint) {
|
|
152
|
+
resolveResult = await resolveSkillVersion(registry, entry, localFingerprint)
|
|
153
|
+
} else {
|
|
154
|
+
const meta = await apiRequest(
|
|
155
|
+
registry,
|
|
156
|
+
{ method: 'GET', path: `${ApiRoutes.skills}/${encodeURIComponent(entry)}` },
|
|
157
|
+
ApiV1SkillResponseSchema,
|
|
158
|
+
)
|
|
159
|
+
resolveResult = { match: null, latestVersion: meta.latestVersion ?? null }
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
const latest = resolveResult.latestVersion?.version ?? null
|
|
163
|
+
const matched = resolveResult.match?.version ?? null
|
|
164
|
+
|
|
165
|
+
if (matched && lock.skills[entry]?.version !== matched) {
|
|
166
|
+
lock.skills[entry] = {
|
|
167
|
+
version: matched,
|
|
168
|
+
installedAt: lock.skills[entry]?.installedAt ?? Date.now(),
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
if (!latest) {
|
|
173
|
+
spinner.fail(`${entry}: not found`)
|
|
174
|
+
continue
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
if (!matched && localFingerprint && !options.force) {
|
|
178
|
+
spinner.stop()
|
|
179
|
+
if (!allowPrompt) {
|
|
180
|
+
console.log(`${entry}: local changes (no match). Use --force to overwrite.`)
|
|
181
|
+
continue
|
|
182
|
+
}
|
|
183
|
+
const confirm = await promptConfirm(
|
|
184
|
+
`${entry}: local changes (no match). Overwrite with ${options.version ?? latest}?`,
|
|
185
|
+
)
|
|
186
|
+
if (!confirm) {
|
|
187
|
+
console.log(`${entry}: skipped`)
|
|
188
|
+
continue
|
|
189
|
+
}
|
|
190
|
+
spinner.start(`Updating ${entry} -> ${options.version ?? latest}`)
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
const targetVersion = options.version ?? latest
|
|
194
|
+
if (options.version) {
|
|
195
|
+
if (matched && matched === targetVersion) {
|
|
196
|
+
spinner.succeed(`${entry}: already at ${matched}`)
|
|
197
|
+
continue
|
|
198
|
+
}
|
|
199
|
+
} else if (matched && semver.valid(matched) && semver.gte(matched, targetVersion)) {
|
|
200
|
+
spinner.succeed(`${entry}: up to date (${matched})`)
|
|
201
|
+
continue
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
if (spinner.isSpinning) {
|
|
205
|
+
spinner.text = `Updating ${entry} -> ${targetVersion}`
|
|
206
|
+
} else {
|
|
207
|
+
spinner.start(`Updating ${entry} -> ${targetVersion}`)
|
|
208
|
+
}
|
|
209
|
+
await rm(target, { recursive: true, force: true })
|
|
210
|
+
const zip = await downloadZip(registry, { slug: entry, version: targetVersion })
|
|
211
|
+
await extractZipToDir(zip, target)
|
|
212
|
+
|
|
213
|
+
const existingOrigin = await readSkillOrigin(target)
|
|
214
|
+
await writeSkillOrigin(target, {
|
|
215
|
+
version: 1,
|
|
216
|
+
registry: existingOrigin?.registry ?? registry,
|
|
217
|
+
slug: existingOrigin?.slug ?? entry,
|
|
218
|
+
installedVersion: targetVersion,
|
|
219
|
+
installedAt: existingOrigin?.installedAt ?? Date.now(),
|
|
220
|
+
})
|
|
221
|
+
|
|
222
|
+
lock.skills[entry] = { version: targetVersion, installedAt: Date.now() }
|
|
223
|
+
spinner.succeed(`${entry}: updated -> ${targetVersion}`)
|
|
224
|
+
} catch (error) {
|
|
225
|
+
spinner.fail(formatError(error))
|
|
226
|
+
throw error
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
await writeLockfile(opts.workdir, lock)
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
export async function cmdList(opts: GlobalOpts) {
|
|
234
|
+
const lock = await readLockfile(opts.workdir)
|
|
235
|
+
const entries = Object.entries(lock.skills)
|
|
236
|
+
if (entries.length === 0) {
|
|
237
|
+
console.log('No installed skills.')
|
|
238
|
+
return
|
|
239
|
+
}
|
|
240
|
+
for (const [slug, entry] of entries) {
|
|
241
|
+
console.log(`${slug} ${entry.version ?? 'latest'}`)
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
type ExploreSort = 'newest' | 'downloads' | 'rating' | 'installs' | 'installsAllTime' | 'trending'
|
|
246
|
+
type ApiExploreSort =
|
|
247
|
+
| 'updated'
|
|
248
|
+
| 'downloads'
|
|
249
|
+
| 'stars'
|
|
250
|
+
| 'installsCurrent'
|
|
251
|
+
| 'installsAllTime'
|
|
252
|
+
| 'trending'
|
|
253
|
+
|
|
254
|
+
export async function cmdExplore(
|
|
255
|
+
opts: GlobalOpts,
|
|
256
|
+
options: { limit?: number; sort?: string; json?: boolean } = {},
|
|
257
|
+
) {
|
|
258
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
259
|
+
const spinner = createSpinner('Fetching latest skills')
|
|
260
|
+
try {
|
|
261
|
+
const url = new URL(ApiRoutes.skills, registry)
|
|
262
|
+
const boundedLimit = clampLimit(options.limit ?? 25)
|
|
263
|
+
const { apiSort } = resolveExploreSort(options.sort)
|
|
264
|
+
url.searchParams.set('limit', String(boundedLimit))
|
|
265
|
+
if (apiSort !== 'updated') url.searchParams.set('sort', apiSort)
|
|
266
|
+
const result = await apiRequest(
|
|
267
|
+
registry,
|
|
268
|
+
{ method: 'GET', url: url.toString() },
|
|
269
|
+
ApiV1SkillListResponseSchema,
|
|
270
|
+
)
|
|
271
|
+
|
|
272
|
+
spinner.stop()
|
|
273
|
+
if (options.json) {
|
|
274
|
+
console.log(JSON.stringify(result, null, 2))
|
|
275
|
+
return
|
|
276
|
+
}
|
|
277
|
+
if (result.items.length === 0) {
|
|
278
|
+
console.log('No skills found.')
|
|
279
|
+
return
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
for (const item of result.items) {
|
|
283
|
+
console.log(formatExploreLine(item))
|
|
284
|
+
}
|
|
285
|
+
} catch (error) {
|
|
286
|
+
spinner.fail(formatError(error))
|
|
287
|
+
throw error
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
export function formatExploreLine(item: {
|
|
292
|
+
slug: string
|
|
293
|
+
summary?: string | null
|
|
294
|
+
updatedAt: number
|
|
295
|
+
latestVersion?: { version: string } | null
|
|
296
|
+
}) {
|
|
297
|
+
const version = item.latestVersion?.version ?? '?'
|
|
298
|
+
const age = formatRelativeTime(item.updatedAt)
|
|
299
|
+
const summary = item.summary ? ` ${truncate(item.summary, 50)}` : ''
|
|
300
|
+
return `${item.slug} v${version} ${age}${summary}`
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
export function clampLimit(limit: number, fallback = 25) {
|
|
304
|
+
if (!Number.isFinite(limit)) return fallback
|
|
305
|
+
return Math.min(Math.max(1, limit), 200)
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
function formatRelativeTime(timestamp: number): string {
|
|
309
|
+
const now = Date.now()
|
|
310
|
+
const diff = now - timestamp
|
|
311
|
+
const seconds = Math.floor(diff / 1000)
|
|
312
|
+
const minutes = Math.floor(seconds / 60)
|
|
313
|
+
const hours = Math.floor(minutes / 60)
|
|
314
|
+
const days = Math.floor(hours / 24)
|
|
315
|
+
|
|
316
|
+
if (days > 30) {
|
|
317
|
+
const months = Math.floor(days / 30)
|
|
318
|
+
return `${months}mo ago`
|
|
319
|
+
}
|
|
320
|
+
if (days > 0) return `${days}d ago`
|
|
321
|
+
if (hours > 0) return `${hours}h ago`
|
|
322
|
+
if (minutes > 0) return `${minutes}m ago`
|
|
323
|
+
return 'just now'
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
function truncate(str: string, maxLen: number): string {
|
|
327
|
+
if (str.length <= maxLen) return str
|
|
328
|
+
return `${str.slice(0, maxLen - 1)}…`
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
function resolveExploreSort(raw?: string): { sort: ExploreSort; apiSort: ApiExploreSort } {
|
|
332
|
+
const normalized = raw?.trim().toLowerCase()
|
|
333
|
+
if (!normalized || normalized === 'newest' || normalized === 'updated') {
|
|
334
|
+
return { sort: 'newest', apiSort: 'updated' }
|
|
335
|
+
}
|
|
336
|
+
if (normalized === 'downloads' || normalized === 'download') {
|
|
337
|
+
return { sort: 'downloads', apiSort: 'downloads' }
|
|
338
|
+
}
|
|
339
|
+
if (normalized === 'rating' || normalized === 'stars' || normalized === 'star') {
|
|
340
|
+
return { sort: 'rating', apiSort: 'stars' }
|
|
341
|
+
}
|
|
342
|
+
if (
|
|
343
|
+
normalized === 'installs' ||
|
|
344
|
+
normalized === 'install' ||
|
|
345
|
+
normalized === 'installscurrent' ||
|
|
346
|
+
normalized === 'installs-current' ||
|
|
347
|
+
normalized === 'current'
|
|
348
|
+
) {
|
|
349
|
+
return { sort: 'installs', apiSort: 'installsCurrent' }
|
|
350
|
+
}
|
|
351
|
+
if (normalized === 'installsalltime' || normalized === 'installs-all-time') {
|
|
352
|
+
return { sort: 'installsAllTime', apiSort: 'installsAllTime' }
|
|
353
|
+
}
|
|
354
|
+
if (normalized === 'trending') {
|
|
355
|
+
return { sort: 'trending', apiSort: 'trending' }
|
|
356
|
+
}
|
|
357
|
+
fail(
|
|
358
|
+
`Invalid sort "${raw}". Use newest, downloads, rating, installs, installsAllTime, or trending.`,
|
|
359
|
+
)
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
async function resolveSkillVersion(registry: string, slug: string, hash: string) {
|
|
363
|
+
const url = new URL(ApiRoutes.resolve, registry)
|
|
364
|
+
url.searchParams.set('slug', slug)
|
|
365
|
+
url.searchParams.set('hash', hash)
|
|
366
|
+
return apiRequest(
|
|
367
|
+
registry,
|
|
368
|
+
{ method: 'GET', url: url.toString() },
|
|
369
|
+
ApiV1SkillResolveResponseSchema,
|
|
370
|
+
)
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
async function fileExists(path: string) {
|
|
374
|
+
try {
|
|
375
|
+
await stat(path)
|
|
376
|
+
return true
|
|
377
|
+
} catch {
|
|
378
|
+
return false
|
|
379
|
+
}
|
|
380
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { readGlobalConfig } from '../../config.js'
|
|
2
|
+
import { apiRequest } from '../../http.js'
|
|
3
|
+
import { ApiRoutes, ApiV1StarResponseSchema } 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 cmdStarSkill(
|
|
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(`Star ${slug}?`)
|
|
28
|
+
if (!ok) return
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const token = await requireToken()
|
|
32
|
+
const registry = await getRegistry(opts, { cache: true })
|
|
33
|
+
const spinner = createSpinner(`Starring ${slug}`)
|
|
34
|
+
try {
|
|
35
|
+
const result = await apiRequest(
|
|
36
|
+
registry,
|
|
37
|
+
{ method: 'POST', path: `${ApiRoutes.stars}/${encodeURIComponent(slug)}`, token },
|
|
38
|
+
ApiV1StarResponseSchema,
|
|
39
|
+
)
|
|
40
|
+
spinner.succeed(result.alreadyStarred ? `OK. ${slug} already starred.` : `OK. Starred ${slug}`)
|
|
41
|
+
return result
|
|
42
|
+
} catch (error) {
|
|
43
|
+
spinner.fail(formatError(error))
|
|
44
|
+
throw error
|
|
45
|
+
}
|
|
46
|
+
}
|