clawdhub 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.
Files changed (79) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +57 -0
  3. package/bin/clawdhub.js +2 -0
  4. package/dist/browserAuth.d.ts +20 -0
  5. package/dist/browserAuth.js +156 -0
  6. package/dist/browserAuth.js.map +1 -0
  7. package/dist/browserAuth.test.d.ts +1 -0
  8. package/dist/browserAuth.test.js +39 -0
  9. package/dist/browserAuth.test.js.map +1 -0
  10. package/dist/cli/buildInfo.d.ts +3 -0
  11. package/dist/cli/buildInfo.js +103 -0
  12. package/dist/cli/buildInfo.js.map +1 -0
  13. package/dist/cli/commands/auth.d.ts +9 -0
  14. package/dist/cli/commands/auth.js +75 -0
  15. package/dist/cli/commands/auth.js.map +1 -0
  16. package/dist/cli/commands/publish.d.ts +8 -0
  17. package/dist/cli/commands/publish.js +91 -0
  18. package/dist/cli/commands/publish.js.map +1 -0
  19. package/dist/cli/commands/publish.test.d.ts +1 -0
  20. package/dist/cli/commands/publish.test.js +120 -0
  21. package/dist/cli/commands/publish.test.js.map +1 -0
  22. package/dist/cli/commands/skills.d.ts +9 -0
  23. package/dist/cli/commands/skills.js +195 -0
  24. package/dist/cli/commands/skills.js.map +1 -0
  25. package/dist/cli/commands/sync.d.ts +11 -0
  26. package/dist/cli/commands/sync.js +273 -0
  27. package/dist/cli/commands/sync.js.map +1 -0
  28. package/dist/cli/commands/sync.test.d.ts +1 -0
  29. package/dist/cli/commands/sync.test.js +106 -0
  30. package/dist/cli/commands/sync.test.js.map +1 -0
  31. package/dist/cli/helpStyle.d.ts +13 -0
  32. package/dist/cli/helpStyle.js +38 -0
  33. package/dist/cli/helpStyle.js.map +1 -0
  34. package/dist/cli/registry.d.ts +7 -0
  35. package/dist/cli/registry.js +27 -0
  36. package/dist/cli/registry.js.map +1 -0
  37. package/dist/cli/scanSkills.d.ts +7 -0
  38. package/dist/cli/scanSkills.js +75 -0
  39. package/dist/cli/scanSkills.js.map +1 -0
  40. package/dist/cli/scanSkills.test.d.ts +1 -0
  41. package/dist/cli/scanSkills.test.js +47 -0
  42. package/dist/cli/scanSkills.test.js.map +1 -0
  43. package/dist/cli/slug.d.ts +2 -0
  44. package/dist/cli/slug.js +16 -0
  45. package/dist/cli/slug.js.map +1 -0
  46. package/dist/cli/types.d.ts +14 -0
  47. package/dist/cli/types.js +2 -0
  48. package/dist/cli/types.js.map +1 -0
  49. package/dist/cli/ui.d.ts +7 -0
  50. package/dist/cli/ui.js +72 -0
  51. package/dist/cli/ui.js.map +1 -0
  52. package/dist/cli.d.ts +2 -0
  53. package/dist/cli.js +179 -0
  54. package/dist/cli.js.map +1 -0
  55. package/dist/config.d.ts +4 -0
  56. package/dist/config.js +38 -0
  57. package/dist/config.js.map +1 -0
  58. package/dist/discovery.d.ts +5 -0
  59. package/dist/discovery.js +21 -0
  60. package/dist/discovery.js.map +1 -0
  61. package/dist/discovery.test.d.ts +1 -0
  62. package/dist/discovery.test.js +24 -0
  63. package/dist/discovery.test.js.map +1 -0
  64. package/dist/http.d.ts +19 -0
  65. package/dist/http.js +46 -0
  66. package/dist/http.js.map +1 -0
  67. package/dist/http.test.d.ts +1 -0
  68. package/dist/http.test.js +70 -0
  69. package/dist/http.test.js.map +1 -0
  70. package/dist/skills.d.ts +34 -0
  71. package/dist/skills.js +135 -0
  72. package/dist/skills.js.map +1 -0
  73. package/dist/skills.test.d.ts +1 -0
  74. package/dist/skills.test.js +83 -0
  75. package/dist/skills.test.js.map +1 -0
  76. package/dist/types.d.ts +7 -0
  77. package/dist/types.js +2 -0
  78. package/dist/types.js.map +1 -0
  79. package/package.json +39 -0
package/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 Peter Steinberger
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
22
+
package/README.md ADDED
@@ -0,0 +1,57 @@
1
+ # `clawdhub`
2
+
3
+ ClawdHub CLI — install, update, search, and publish agent skills as folders.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ # From this repo (shortcut script at repo root)
9
+ bun clawdhub --help
10
+
11
+ # Once published to npm
12
+ # npm i -g clawdhub
13
+ ```
14
+
15
+ ## Auth (publish)
16
+
17
+ ```bash
18
+ clawdhub login
19
+ # or
20
+ clawdhub auth login
21
+
22
+ # Headless / token paste
23
+ # or (token paste / headless)
24
+ clawdhub login --token clh_...
25
+ ```
26
+
27
+ Notes:
28
+
29
+ - Browser login opens `https://clawdhub.com/cli/auth` and completes via a loopback callback.
30
+ - Token stored in `~/Library/Application Support/clawdhub/config.json` on macOS (override via `CLAWDHUB_CONFIG_PATH`).
31
+
32
+ ## Examples
33
+
34
+ ```bash
35
+ clawdhub search "postgres backups"
36
+ clawdhub install my-skill-pack
37
+ clawdhub update --all
38
+ clawdhub update --all --no-input --force
39
+ clawdhub publish ./my-skill-pack --slug my-skill-pack --name "My Skill Pack" --version 1.2.0 --changelog "Fixes + docs"
40
+ ```
41
+
42
+ ## Sync (upload local skills)
43
+
44
+ ```bash
45
+ # Start anywhere; scans workdir first, then legacy Clawdis/Clawd locations.
46
+ clawdhub sync
47
+
48
+ # Explicit roots + non-interactive dry-run
49
+ clawdhub sync --root ../clawdis/skills --all --dry-run
50
+ ```
51
+
52
+ ## Defaults
53
+
54
+ - Site: `https://clawdhub.com` (override via `--site` or `CLAWDHUB_SITE`)
55
+ - Registry: discovered from `/.well-known/clawdhub.json` on the site (override via `--registry` or `CLAWDHUB_REGISTRY`)
56
+ - Workdir: current directory (override via `--workdir`)
57
+ - Install dir: `./skills` under workdir (override via `--dir`)
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ import('../dist/cli.js')
@@ -0,0 +1,20 @@
1
+ export type LoopbackAuthResult = {
2
+ token: string;
3
+ registry?: string;
4
+ state?: string;
5
+ };
6
+ export declare function buildCliAuthUrl(params: {
7
+ siteUrl: string;
8
+ redirectUri: string;
9
+ label?: string;
10
+ state: string;
11
+ }): string;
12
+ export declare function isAllowedLoopbackRedirectUri(value: string): boolean;
13
+ export declare function startLoopbackAuthServer(params?: {
14
+ timeoutMs?: number;
15
+ }): Promise<{
16
+ redirectUri: string;
17
+ state: string;
18
+ waitForResult: () => Promise<LoopbackAuthResult>;
19
+ close: () => import("node:http").Server<typeof import("node:http").IncomingMessage, typeof import("node:http").ServerResponse>;
20
+ }>;
@@ -0,0 +1,156 @@
1
+ import { createServer } from 'node:http';
2
+ export function buildCliAuthUrl(params) {
3
+ const url = new URL('/cli/auth', params.siteUrl);
4
+ url.searchParams.set('redirect_uri', params.redirectUri);
5
+ if (params.label)
6
+ url.searchParams.set('label_b64', encodeBase64Url(params.label));
7
+ url.searchParams.set('state', params.state);
8
+ return url.toString();
9
+ }
10
+ export function isAllowedLoopbackRedirectUri(value) {
11
+ let url;
12
+ try {
13
+ url = new URL(value);
14
+ }
15
+ catch {
16
+ return false;
17
+ }
18
+ if (url.protocol !== 'http:')
19
+ return false;
20
+ const host = url.hostname.toLowerCase();
21
+ if (host !== '127.0.0.1' && host !== 'localhost' && host !== '::1' && host !== '[::1]') {
22
+ return false;
23
+ }
24
+ return true;
25
+ }
26
+ export async function startLoopbackAuthServer(params) {
27
+ const timeoutMs = params?.timeoutMs ?? 5 * 60_000;
28
+ const expectedState = generateState();
29
+ let resolveToken = null;
30
+ let rejectToken = null;
31
+ const tokenPromise = new Promise((resolve, reject) => {
32
+ resolveToken = resolve;
33
+ rejectToken = reject;
34
+ });
35
+ const server = createServer((req, res) => {
36
+ const method = req.method ?? 'GET';
37
+ const url = req.url ?? '/';
38
+ if (method === 'GET' && (url === '/' || url.startsWith('/callback'))) {
39
+ res.statusCode = 200;
40
+ res.setHeader('Content-Type', 'text/html; charset=utf-8');
41
+ res.end(CALLBACK_HTML);
42
+ return;
43
+ }
44
+ if (method === 'POST' && url === '/token') {
45
+ const chunks = [];
46
+ req.on('data', (chunk) => chunks.push(chunk));
47
+ req.on('end', () => {
48
+ try {
49
+ const raw = Buffer.concat(chunks).toString('utf8');
50
+ const parsed = JSON.parse(raw);
51
+ if (!parsed || typeof parsed !== 'object')
52
+ throw new Error('invalid payload');
53
+ const token = parsed.token;
54
+ const registry = parsed.registry;
55
+ const state = parsed.state;
56
+ if (typeof token !== 'string' || !token.trim())
57
+ throw new Error('token required');
58
+ if (typeof state !== 'string' || state !== expectedState) {
59
+ throw new Error('state mismatch');
60
+ }
61
+ res.statusCode = 200;
62
+ res.setHeader('Content-Type', 'application/json');
63
+ res.end(JSON.stringify({ ok: true }));
64
+ resolveToken?.({
65
+ token: token.trim(),
66
+ registry: typeof registry === 'string' ? registry : undefined,
67
+ state,
68
+ });
69
+ }
70
+ catch (error) {
71
+ res.statusCode = 400;
72
+ res.setHeader('Content-Type', 'application/json');
73
+ res.end(JSON.stringify({ ok: false }));
74
+ const message = error instanceof Error ? error.message : 'invalid payload';
75
+ rejectToken?.(new Error(message));
76
+ }
77
+ finally {
78
+ server.close();
79
+ }
80
+ });
81
+ return;
82
+ }
83
+ res.statusCode = 404;
84
+ res.setHeader('Content-Type', 'text/plain; charset=utf-8');
85
+ res.end('Not found');
86
+ });
87
+ await new Promise((resolve, reject) => {
88
+ server.once('error', reject);
89
+ server.listen(0, '127.0.0.1', () => resolve());
90
+ });
91
+ const address = server.address();
92
+ if (!address) {
93
+ server.close();
94
+ throw new Error('Failed to bind loopback server');
95
+ }
96
+ const redirectUri = `http://127.0.0.1:${address.port}/callback`;
97
+ const timeout = setTimeout(() => {
98
+ server.close();
99
+ rejectToken?.(new Error('Timed out waiting for browser login'));
100
+ }, timeoutMs);
101
+ tokenPromise.finally(() => clearTimeout(timeout)).catch(() => { });
102
+ return {
103
+ redirectUri,
104
+ state: expectedState,
105
+ waitForResult: () => tokenPromise,
106
+ close: () => server.close(),
107
+ };
108
+ }
109
+ const CALLBACK_HTML = `<!doctype html>
110
+ <html lang="en">
111
+ <meta charset="utf-8" />
112
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
113
+ <title>ClawdHub CLI Login</title>
114
+ <style>
115
+ :root { color-scheme: light dark; }
116
+ body { font-family: ui-sans-serif, system-ui, -apple-system, Segoe UI, sans-serif; padding: 24px; }
117
+ .card { max-width: 560px; margin: 40px auto; padding: 18px 16px; border: 1px solid rgba(127,127,127,.35); border-radius: 12px; }
118
+ code { font-family: ui-monospace, SFMono-Regular, Menlo, monospace; }
119
+ </style>
120
+ <body>
121
+ <div class="card">
122
+ <h1 style="margin: 0 0 10px; font-size: 18px;">Completing login…</h1>
123
+ <p id="status" style="margin: 0; opacity: .8;">Waiting for token.</p>
124
+ </div>
125
+ <script>
126
+ const statusEl = document.getElementById('status')
127
+ const params = new URLSearchParams(location.hash.replace(/^#/, ''))
128
+ const token = params.get('token')
129
+ const registry = params.get('registry')
130
+ const state = params.get('state')
131
+ if (!token) {
132
+ statusEl.textContent = 'Missing token in URL. You can close this tab and try again.'
133
+ } else if (!state) {
134
+ statusEl.textContent = 'Missing state in URL. You can close this tab and try again.'
135
+ } else {
136
+ fetch('/token', {
137
+ method: 'POST',
138
+ headers: { 'Content-Type': 'application/json' },
139
+ body: JSON.stringify({ token, registry, state }),
140
+ }).then(() => {
141
+ statusEl.textContent = 'Logged in. You can close this tab.'
142
+ setTimeout(() => window.close(), 250)
143
+ }).catch(() => {
144
+ statusEl.textContent = 'Failed to send token to CLI. You can close this tab and try again.'
145
+ })
146
+ }
147
+ </script>
148
+ </body>
149
+ </html>`;
150
+ function encodeBase64Url(value) {
151
+ return Buffer.from(value, 'utf8').toString('base64url');
152
+ }
153
+ function generateState() {
154
+ return Buffer.from(crypto.getRandomValues(new Uint8Array(16))).toString('hex');
155
+ }
156
+ //# sourceMappingURL=browserAuth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"browserAuth.js","sourceRoot":"","sources":["../src/browserAuth.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAA;AASxC,MAAM,UAAU,eAAe,CAAC,MAK/B;IACC,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,WAAW,EAAE,MAAM,CAAC,OAAO,CAAC,CAAA;IAChD,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,cAAc,EAAE,MAAM,CAAC,WAAW,CAAC,CAAA;IACxD,IAAI,MAAM,CAAC,KAAK;QAAE,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,WAAW,EAAE,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAA;IAClF,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,KAAK,CAAC,CAAA;IAC3C,OAAO,GAAG,CAAC,QAAQ,EAAE,CAAA;AACvB,CAAC;AAED,MAAM,UAAU,4BAA4B,CAAC,KAAa;IACxD,IAAI,GAAQ,CAAA;IACZ,IAAI,CAAC;QACH,GAAG,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,CAAA;IACtB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAA;IACd,CAAC;IACD,IAAI,GAAG,CAAC,QAAQ,KAAK,OAAO;QAAE,OAAO,KAAK,CAAA;IAC1C,MAAM,IAAI,GAAG,GAAG,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAA;IACvC,IAAI,IAAI,KAAK,WAAW,IAAI,IAAI,KAAK,WAAW,IAAI,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;QACvF,OAAO,KAAK,CAAA;IACd,CAAC;IACD,OAAO,IAAI,CAAA;AACb,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,uBAAuB,CAAC,MAA+B;IAC3E,MAAM,SAAS,GAAG,MAAM,EAAE,SAAS,IAAI,CAAC,GAAG,MAAM,CAAA;IACjD,MAAM,aAAa,GAAG,aAAa,EAAE,CAAA;IAErC,IAAI,YAAY,GAAiD,IAAI,CAAA;IACrE,IAAI,WAAW,GAAoC,IAAI,CAAA;IACvD,MAAM,YAAY,GAAG,IAAI,OAAO,CAAqB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACvE,YAAY,GAAG,OAAO,CAAA;QACtB,WAAW,GAAG,MAAM,CAAA;IACtB,CAAC,CAAC,CAAA;IAEF,MAAM,MAAM,GAAG,YAAY,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACvC,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,IAAI,KAAK,CAAA;QAClC,MAAM,GAAG,GAAG,GAAG,CAAC,GAAG,IAAI,GAAG,CAAA;QAE1B,IAAI,MAAM,KAAK,KAAK,IAAI,CAAC,GAAG,KAAK,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC,EAAE,CAAC;YACrE,GAAG,CAAC,UAAU,GAAG,GAAG,CAAA;YACpB,GAAG,CAAC,SAAS,CAAC,cAAc,EAAE,0BAA0B,CAAC,CAAA;YACzD,GAAG,CAAC,GAAG,CAAC,aAAa,CAAC,CAAA;YACtB,OAAM;QACR,CAAC;QAED,IAAI,MAAM,KAAK,MAAM,IAAI,GAAG,KAAK,QAAQ,EAAE,CAAC;YAC1C,MAAM,MAAM,GAAiB,EAAE,CAAA;YAC/B,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,KAAmB,CAAC,CAAC,CAAA;YAC3D,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;gBACjB,IAAI,CAAC;oBACH,MAAM,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;oBAClD,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAY,CAAA;oBACzC,IAAI,CAAC,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ;wBAAE,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAA;oBAC7E,MAAM,KAAK,GAAI,MAA8B,CAAC,KAAK,CAAA;oBACnD,MAAM,QAAQ,GAAI,MAAiC,CAAC,QAAQ,CAAA;oBAC5D,MAAM,KAAK,GAAI,MAA8B,CAAC,KAAK,CAAA;oBACnD,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;wBAAE,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAA;oBACjF,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,aAAa,EAAE,CAAC;wBACzD,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAA;oBACnC,CAAC;oBACD,GAAG,CAAC,UAAU,GAAG,GAAG,CAAA;oBACpB,GAAG,CAAC,SAAS,CAAC,cAAc,EAAE,kBAAkB,CAAC,CAAA;oBACjD,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC,CAAA;oBACrC,YAAY,EAAE,CAAC;wBACb,KAAK,EAAE,KAAK,CAAC,IAAI,EAAE;wBACnB,QAAQ,EAAE,OAAO,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS;wBAC7D,KAAK;qBACN,CAAC,CAAA;gBACJ,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,GAAG,CAAC,UAAU,GAAG,GAAG,CAAA;oBACpB,GAAG,CAAC,SAAS,CAAC,cAAc,EAAE,kBAAkB,CAAC,CAAA;oBACjD,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC,CAAA;oBACtC,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,iBAAiB,CAAA;oBAC1E,WAAW,EAAE,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAA;gBACnC,CAAC;wBAAS,CAAC;oBACT,MAAM,CAAC,KAAK,EAAE,CAAA;gBAChB,CAAC;YACH,CAAC,CAAC,CAAA;YACF,OAAM;QACR,CAAC;QAED,GAAG,CAAC,UAAU,GAAG,GAAG,CAAA;QACpB,GAAG,CAAC,SAAS,CAAC,cAAc,EAAE,2BAA2B,CAAC,CAAA;QAC1D,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,CAAA;IACtB,CAAC,CAAC,CAAA;IAEF,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QAC1C,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;QAC5B,MAAM,CAAC,MAAM,CAAC,CAAC,EAAE,WAAW,EAAE,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAA;IAChD,CAAC,CAAC,CAAA;IACF,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,EAAwB,CAAA;IACtD,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,CAAC,KAAK,EAAE,CAAA;QACd,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAA;IACnD,CAAC;IACD,MAAM,WAAW,GAAG,oBAAoB,OAAO,CAAC,IAAI,WAAW,CAAA;IAE/D,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;QAC9B,MAAM,CAAC,KAAK,EAAE,CAAA;QACd,WAAW,EAAE,CAAC,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAC,CAAA;IACjE,CAAC,EAAE,SAAS,CAAC,CAAA;IACb,YAAY,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAA;IAEjE,OAAO;QACL,WAAW;QACX,KAAK,EAAE,aAAa;QACpB,aAAa,EAAE,GAAG,EAAE,CAAC,YAAY;QACjC,KAAK,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE;KAC5B,CAAA;AACH,CAAC;AAED,MAAM,aAAa,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;QAwCd,CAAA;AAER,SAAS,eAAe,CAAC,KAAa;IACpC,OAAO,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAA;AACzD,CAAC;AAED,SAAS,aAAa;IACpB,OAAO,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,IAAI,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAA;AAChF,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,39 @@
1
+ /* @vitest-environment node */
2
+ import { describe, expect, it } from 'vitest';
3
+ import { buildCliAuthUrl, isAllowedLoopbackRedirectUri, startLoopbackAuthServer, } from './browserAuth';
4
+ describe('browserAuth', () => {
5
+ it('builds auth url', () => {
6
+ const url = buildCliAuthUrl({
7
+ siteUrl: 'https://example.com',
8
+ redirectUri: 'http://127.0.0.1:1234/callback',
9
+ label: 'CLI token',
10
+ state: 'state123',
11
+ });
12
+ expect(url).toContain('https://example.com/cli/auth?');
13
+ expect(url).toContain('redirect_uri=');
14
+ expect(url).toContain('label_b64=');
15
+ expect(url).toContain('state=');
16
+ });
17
+ it('accepts only loopback http redirect uris', () => {
18
+ expect(isAllowedLoopbackRedirectUri('http://127.0.0.1:1234/callback')).toBe(true);
19
+ expect(isAllowedLoopbackRedirectUri('http://localhost:1234/callback')).toBe(true);
20
+ expect(isAllowedLoopbackRedirectUri('http://[::1]:1234/callback')).toBe(true);
21
+ expect(isAllowedLoopbackRedirectUri('https://127.0.0.1:1234/callback')).toBe(false);
22
+ expect(isAllowedLoopbackRedirectUri('http://evil.com/callback')).toBe(false);
23
+ });
24
+ it('receives token via loopback server', async () => {
25
+ const server = await startLoopbackAuthServer({ timeoutMs: 2000 });
26
+ const payload = {
27
+ token: 'clh_test',
28
+ registry: 'https://example.convex.site',
29
+ state: server.state,
30
+ };
31
+ await fetch(server.redirectUri.replace('/callback', '/token'), {
32
+ method: 'POST',
33
+ headers: { 'Content-Type': 'application/json' },
34
+ body: JSON.stringify(payload),
35
+ });
36
+ await expect(server.waitForResult()).resolves.toEqual(payload);
37
+ });
38
+ });
39
+ //# sourceMappingURL=browserAuth.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"browserAuth.test.js","sourceRoot":"","sources":["../src/browserAuth.test.ts"],"names":[],"mappings":"AAAA,8BAA8B;AAE9B,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAC7C,OAAO,EACL,eAAe,EACf,4BAA4B,EAC5B,uBAAuB,GACxB,MAAM,eAAe,CAAA;AAEtB,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;IAC3B,EAAE,CAAC,iBAAiB,EAAE,GAAG,EAAE;QACzB,MAAM,GAAG,GAAG,eAAe,CAAC;YAC1B,OAAO,EAAE,qBAAqB;YAC9B,WAAW,EAAE,gCAAgC;YAC7C,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE,UAAU;SAClB,CAAC,CAAA;QACF,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,+BAA+B,CAAC,CAAA;QACtD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAA;QACtC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAA;QACnC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAA;IACjC,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,CAAC,4BAA4B,CAAC,gCAAgC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACjF,MAAM,CAAC,4BAA4B,CAAC,gCAAgC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QACjF,MAAM,CAAC,4BAA4B,CAAC,4BAA4B,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;QAC7E,MAAM,CAAC,4BAA4B,CAAC,iCAAiC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;QACnF,MAAM,CAAC,4BAA4B,CAAC,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAC9E,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,MAAM,GAAG,MAAM,uBAAuB,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;QACjE,MAAM,OAAO,GAAG;YACd,KAAK,EAAE,UAAU;YACjB,QAAQ,EAAE,6BAA6B;YACvC,KAAK,EAAE,MAAM,CAAC,KAAK;SACpB,CAAA;QACD,MAAM,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,WAAW,EAAE,QAAQ,CAAC,EAAE;YAC7D,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;YAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;SAC9B,CAAC,CAAA;QACF,MAAM,MAAM,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,CAAA;IAChE,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
@@ -0,0 +1,3 @@
1
+ export declare function getCliCommit(): string | null;
2
+ export declare function getCliVersion(): string;
3
+ export declare function getCliBuildLabel(): string;
@@ -0,0 +1,103 @@
1
+ import { existsSync, readFileSync, statSync } from 'node:fs';
2
+ import { dirname, join, resolve } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ function readPackageVersion() {
5
+ try {
6
+ const path = join(dirname(fileURLToPath(import.meta.url)), '../../package.json');
7
+ const raw = readFileSync(path, 'utf8');
8
+ const pkg = JSON.parse(raw);
9
+ return typeof pkg.version === 'string' ? pkg.version : '0.0.0';
10
+ }
11
+ catch {
12
+ return '0.0.0';
13
+ }
14
+ }
15
+ function shortCommit(value) {
16
+ const trimmed = value.trim();
17
+ if (!trimmed)
18
+ return null;
19
+ if (trimmed.length <= 8)
20
+ return trimmed;
21
+ return trimmed.slice(0, 8);
22
+ }
23
+ export function getCliCommit() {
24
+ const candidates = [
25
+ process.env.CLAWDHUB_COMMIT,
26
+ process.env.VERCEL_GIT_COMMIT_SHA,
27
+ process.env.GITHUB_SHA,
28
+ process.env.COMMIT_SHA,
29
+ ];
30
+ for (const candidate of candidates) {
31
+ if (!candidate)
32
+ continue;
33
+ const short = shortCommit(candidate);
34
+ if (short)
35
+ return short;
36
+ }
37
+ return readGitCommitFromCwd();
38
+ }
39
+ export function getCliVersion() {
40
+ return readPackageVersion();
41
+ }
42
+ export function getCliBuildLabel() {
43
+ const version = getCliVersion();
44
+ const commit = getCliCommit();
45
+ return commit ? `v${version} (${commit})` : `v${version}`;
46
+ }
47
+ function readGitCommitFromCwd() {
48
+ try {
49
+ const gitDir = findGitDir(process.cwd());
50
+ if (!gitDir)
51
+ return null;
52
+ const headPath = join(gitDir, 'HEAD');
53
+ if (!existsSync(headPath))
54
+ return null;
55
+ const head = readFileSync(headPath, 'utf8').trim();
56
+ if (!head)
57
+ return null;
58
+ if (!head.startsWith('ref:'))
59
+ return shortCommit(head);
60
+ const ref = head.replace(/^ref:\s*/, '').trim();
61
+ if (!ref)
62
+ return null;
63
+ const refPath = join(gitDir, ref);
64
+ if (!existsSync(refPath))
65
+ return null;
66
+ const sha = readFileSync(refPath, 'utf8').trim();
67
+ return shortCommit(sha);
68
+ }
69
+ catch {
70
+ return null;
71
+ }
72
+ }
73
+ function findGitDir(start) {
74
+ let current = resolve(start);
75
+ for (;;) {
76
+ const dotGit = join(current, '.git');
77
+ if (existsSync(dotGit)) {
78
+ try {
79
+ const stat = statSync(dotGit);
80
+ if (stat.isDirectory())
81
+ return dotGit;
82
+ }
83
+ catch {
84
+ // ignore
85
+ }
86
+ try {
87
+ const content = readFileSync(dotGit, 'utf8').trim();
88
+ const match = content.match(/^gitdir:\s*(.+)$/);
89
+ if (match?.[1])
90
+ return resolve(current, match[1]);
91
+ }
92
+ catch {
93
+ return dotGit;
94
+ }
95
+ return dotGit;
96
+ }
97
+ const parent = resolve(current, '..');
98
+ if (parent === current)
99
+ return null;
100
+ current = parent;
101
+ }
102
+ }
103
+ //# sourceMappingURL=buildInfo.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"buildInfo.js","sourceRoot":"","sources":["../../src/cli/buildInfo.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { GlobalOpts } from '../types.js';
2
+ export declare function cmdLoginFlow(opts: GlobalOpts, options: {
3
+ token?: string;
4
+ label?: string;
5
+ browser?: boolean;
6
+ }, inputAllowed: boolean): Promise<void>;
7
+ export declare function cmdLogin(opts: GlobalOpts, tokenFlag: string | undefined, inputAllowed: boolean): Promise<void>;
8
+ export declare function cmdLogout(opts: GlobalOpts): Promise<void>;
9
+ export declare function cmdWhoami(opts: GlobalOpts): Promise<void>;
@@ -0,0 +1,75 @@
1
+ import { ApiCliWhoamiResponseSchema, ApiRoutes } from 'clawdhub-schema';
2
+ import { buildCliAuthUrl, startLoopbackAuthServer } from '../../browserAuth.js';
3
+ import { readGlobalConfig, writeGlobalConfig } from '../../config.js';
4
+ import { discoverRegistryFromSite } from '../../discovery.js';
5
+ import { apiRequest } from '../../http.js';
6
+ import { getRegistry } from '../registry.js';
7
+ import { createSpinner, fail, formatError, openInBrowser, promptHidden } from '../ui.js';
8
+ export async function cmdLoginFlow(opts, options, inputAllowed) {
9
+ if (options.token) {
10
+ await cmdLogin(opts, options.token, inputAllowed);
11
+ return;
12
+ }
13
+ if (options.browser === false) {
14
+ fail('Token required (use --token or remove --no-browser)');
15
+ }
16
+ const label = String(options.label ?? 'CLI token').trim() || 'CLI token';
17
+ const receiver = await startLoopbackAuthServer();
18
+ const discovery = await discoverRegistryFromSite(opts.site).catch(() => null);
19
+ const authBase = discovery?.authBase?.trim() || opts.site;
20
+ const authUrl = buildCliAuthUrl({
21
+ siteUrl: authBase,
22
+ redirectUri: receiver.redirectUri,
23
+ label,
24
+ state: receiver.state,
25
+ });
26
+ console.log(`Opening browser: ${authUrl}`);
27
+ openInBrowser(authUrl);
28
+ const result = await receiver.waitForResult();
29
+ const registry = result.registry?.trim() || opts.registry;
30
+ await cmdLogin({ ...opts, registry }, result.token, inputAllowed);
31
+ }
32
+ export async function cmdLogin(opts, tokenFlag, inputAllowed) {
33
+ if (!tokenFlag && !inputAllowed)
34
+ fail('Token required (use --token or remove --no-input)');
35
+ const token = tokenFlag || (await promptHidden('ClawdHub token: '));
36
+ if (!token)
37
+ fail('Token required');
38
+ const registry = await getRegistry(opts, { cache: true });
39
+ const spinner = createSpinner('Verifying token');
40
+ try {
41
+ const whoami = await apiRequest(registry, { method: 'GET', path: ApiRoutes.cliWhoami, token }, ApiCliWhoamiResponseSchema);
42
+ if (!whoami.user)
43
+ fail('Login failed');
44
+ await writeGlobalConfig({ registry, token });
45
+ const handle = whoami.user.handle ? `@${whoami.user.handle}` : 'unknown user';
46
+ spinner.succeed(`OK. Logged in as ${handle}.`);
47
+ }
48
+ catch (error) {
49
+ spinner.fail(formatError(error));
50
+ throw error;
51
+ }
52
+ }
53
+ export async function cmdLogout(opts) {
54
+ const cfg = await readGlobalConfig();
55
+ const registry = cfg?.registry || (await getRegistry(opts, { cache: true }));
56
+ await writeGlobalConfig({ registry, token: undefined });
57
+ console.log('OK. Logged out.');
58
+ }
59
+ export async function cmdWhoami(opts) {
60
+ const cfg = await readGlobalConfig();
61
+ const token = cfg?.token;
62
+ if (!token)
63
+ fail('Not logged in. Run: clawdhub login');
64
+ const registry = await getRegistry(opts, { cache: true });
65
+ const spinner = createSpinner('Checking token');
66
+ try {
67
+ const whoami = await apiRequest(registry, { method: 'GET', path: ApiRoutes.cliWhoami, token }, ApiCliWhoamiResponseSchema);
68
+ spinner.succeed(whoami.user.handle ?? 'unknown');
69
+ }
70
+ catch (error) {
71
+ spinner.fail(formatError(error));
72
+ throw error;
73
+ }
74
+ }
75
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/cli/commands/auth.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import type { GlobalOpts } from '../types.js';
2
+ export declare function cmdPublish(opts: GlobalOpts, folderArg: string, options: {
3
+ slug?: string;
4
+ name?: string;
5
+ version?: string;
6
+ changelog?: string;
7
+ tags?: string;
8
+ }): Promise<void>;
@@ -0,0 +1,91 @@
1
+ import { stat } from 'node:fs/promises';
2
+ import { basename, resolve } from 'node:path';
3
+ import { ApiCliPublishResponseSchema, ApiCliUploadUrlResponseSchema, ApiRoutes, ApiSkillMetaResponseSchema, ApiUploadFileResponseSchema, CliPublishRequestSchema, parseArk, } from 'clawdhub-schema';
4
+ import semver from 'semver';
5
+ import { readGlobalConfig } from '../../config.js';
6
+ import { apiRequest } from '../../http.js';
7
+ import { listTextFiles, sha256Hex } from '../../skills.js';
8
+ import { getRegistry } from '../registry.js';
9
+ import { sanitizeSlug, titleCase } from '../slug.js';
10
+ import { createSpinner, fail, formatError } from '../ui.js';
11
+ export async function cmdPublish(opts, folderArg, options) {
12
+ const folder = folderArg ? resolve(opts.workdir, folderArg) : null;
13
+ if (!folder)
14
+ fail('Path required');
15
+ const folderStat = await stat(folder).catch(() => null);
16
+ if (!folderStat || !folderStat.isDirectory())
17
+ fail('Path must be a folder');
18
+ const cfg = await readGlobalConfig();
19
+ const token = cfg?.token;
20
+ if (!token)
21
+ fail('Not logged in. Run: clawdhub login');
22
+ const registry = await getRegistry(opts, { cache: true });
23
+ const slug = options.slug ?? sanitizeSlug(basename(folder));
24
+ const displayName = options.name ?? titleCase(basename(folder));
25
+ const version = options.version;
26
+ const changelog = options.changelog ?? '';
27
+ const tagsValue = options.tags ?? 'latest';
28
+ const tags = tagsValue
29
+ .split(',')
30
+ .map((tag) => tag.trim())
31
+ .filter(Boolean);
32
+ if (!slug)
33
+ fail('--slug required');
34
+ if (!displayName)
35
+ fail('--name required');
36
+ if (!version || !semver.valid(version))
37
+ fail('--version must be valid semver');
38
+ const spinner = createSpinner(`Preparing ${slug}@${version}`);
39
+ try {
40
+ const meta = await apiRequest(registry, { method: 'GET', path: `/api/skill?slug=${encodeURIComponent(slug)}` }, ApiSkillMetaResponseSchema).catch(() => null);
41
+ const exists = Boolean(meta?.skill);
42
+ if (exists && !changelog.trim())
43
+ fail('--changelog required for updates');
44
+ const filesOnDisk = await listTextFiles(folder);
45
+ if (filesOnDisk.length === 0)
46
+ fail('No files found');
47
+ if (!filesOnDisk.some((file) => {
48
+ const lower = file.relPath.toLowerCase();
49
+ return lower === 'skill.md' || lower === 'skills.md';
50
+ })) {
51
+ fail('SKILL.md required');
52
+ }
53
+ const uploaded = [];
54
+ let index = 0;
55
+ for (const file of filesOnDisk) {
56
+ index += 1;
57
+ spinner.text = `Uploading ${file.relPath} (${index}/${filesOnDisk.length})`;
58
+ const { uploadUrl } = await apiRequest(registry, { method: 'POST', path: ApiRoutes.cliUploadUrl, token }, ApiCliUploadUrlResponseSchema);
59
+ const storageId = await uploadFile(uploadUrl, file.bytes, file.contentType ?? 'text/plain');
60
+ const sha256 = sha256Hex(file.bytes);
61
+ uploaded.push({
62
+ path: file.relPath,
63
+ size: file.bytes.byteLength,
64
+ storageId,
65
+ sha256,
66
+ contentType: file.contentType ?? undefined,
67
+ });
68
+ }
69
+ spinner.text = `Publishing ${slug}@${version}`;
70
+ const body = parseArk(CliPublishRequestSchema, { slug, displayName, version, changelog, tags, files: uploaded }, 'Publish payload');
71
+ const result = await apiRequest(registry, { method: 'POST', path: ApiRoutes.cliPublish, token, body }, ApiCliPublishResponseSchema);
72
+ spinner.succeed(`OK. Published ${slug}@${version} (${result.versionId})`);
73
+ }
74
+ catch (error) {
75
+ spinner.fail(formatError(error));
76
+ throw error;
77
+ }
78
+ }
79
+ async function uploadFile(uploadUrl, bytes, contentType) {
80
+ const response = await fetch(uploadUrl, {
81
+ method: 'POST',
82
+ headers: { 'Content-Type': contentType || 'application/octet-stream' },
83
+ body: Buffer.from(bytes),
84
+ });
85
+ if (!response.ok) {
86
+ throw new Error(`Upload failed: ${await response.text()}`);
87
+ }
88
+ const payload = parseArk(ApiUploadFileResponseSchema, (await response.json()), 'Upload response');
89
+ return payload.storageId;
90
+ }
91
+ //# sourceMappingURL=publish.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"publish.js","sourceRoot":"","sources":["../../../src/cli/commands/publish.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};