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
@@ -0,0 +1,120 @@
1
+ /* @vitest-environment node */
2
+ import { mkdir, mkdtemp, rm, writeFile } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { afterEach, describe, expect, it, vi } from 'vitest';
6
+ import { sha256Hex } from '../../skills';
7
+ vi.mock('../../config.js', () => ({
8
+ readGlobalConfig: vi.fn(async () => ({ registry: 'https://clawdhub.com', token: 'tkn' })),
9
+ }));
10
+ const mockGetRegistry = vi.fn(async (_opts, _params) => 'https://clawdhub.com');
11
+ vi.mock('../registry.js', () => ({
12
+ getRegistry: (opts, params) => mockGetRegistry(opts, params),
13
+ }));
14
+ const mockApiRequest = vi.fn();
15
+ vi.mock('../../http.js', () => ({
16
+ apiRequest: (registry, args, schema) => mockApiRequest(registry, args, schema),
17
+ }));
18
+ const mockFail = vi.fn((message) => {
19
+ throw new Error(message);
20
+ });
21
+ const mockSpinner = { text: '', succeed: vi.fn(), fail: vi.fn() };
22
+ vi.mock('../ui.js', () => ({
23
+ createSpinner: vi.fn(() => mockSpinner),
24
+ fail: (message) => mockFail(message),
25
+ formatError: (error) => (error instanceof Error ? error.message : String(error)),
26
+ }));
27
+ const { cmdPublish } = await import('./publish');
28
+ async function makeTmpWorkdir() {
29
+ const root = await mkdtemp(join(tmpdir(), 'clawdhub-publish-'));
30
+ return root;
31
+ }
32
+ function makeOpts(workdir) {
33
+ return {
34
+ workdir,
35
+ dir: join(workdir, 'skills'),
36
+ site: 'https://clawdhub.com',
37
+ registry: 'https://clawdhub.com',
38
+ };
39
+ }
40
+ afterEach(() => {
41
+ vi.unstubAllGlobals();
42
+ vi.clearAllMocks();
43
+ });
44
+ describe('cmdPublish', () => {
45
+ it('publishes SKILL.md from disk (mocked HTTP)', async () => {
46
+ const workdir = await makeTmpWorkdir();
47
+ try {
48
+ const folder = join(workdir, 'my-skill');
49
+ await mkdir(folder, { recursive: true });
50
+ const skillContent = '# Skill\n\nHello\n';
51
+ const notesContent = 'notes\n';
52
+ await writeFile(join(folder, 'SKILL.md'), skillContent, 'utf8');
53
+ await writeFile(join(folder, 'notes.md'), notesContent, 'utf8');
54
+ let uploadIndex = 0;
55
+ mockApiRequest.mockImplementation(async (_registry, args) => {
56
+ if (args.method === 'GET' && args.path.startsWith('/api/skill?slug=')) {
57
+ return { skill: null, latestVersion: { version: '9.9.9' } };
58
+ }
59
+ if (args.method === 'POST' && args.path === '/api/cli/upload-url') {
60
+ uploadIndex += 1;
61
+ return { uploadUrl: `https://upload.example/${uploadIndex}` };
62
+ }
63
+ if (args.method === 'POST' && args.path === '/api/cli/publish') {
64
+ return { ok: true, skillId: 'skill_1', versionId: 'ver_1' };
65
+ }
66
+ throw new Error(`Unexpected apiRequest: ${args.method} ${args.path}`);
67
+ });
68
+ vi.stubGlobal('fetch', vi.fn(async (url, init) => {
69
+ expect(url).toMatch(/^https:\/\/upload\.example\/\d+$/);
70
+ expect(init?.method).toBe('POST');
71
+ expect(init?.headers?.['Content-Type']).toMatch(/text\/(markdown|plain)/);
72
+ return new Response(JSON.stringify({ storageId: `st_${String(url).split('/').pop()}` }), {
73
+ status: 200,
74
+ headers: { 'Content-Type': 'application/json' },
75
+ });
76
+ }));
77
+ await cmdPublish(makeOpts(workdir), 'my-skill', {
78
+ slug: 'my-skill',
79
+ name: 'My Skill',
80
+ version: '1.0.0',
81
+ changelog: '',
82
+ tags: 'latest',
83
+ });
84
+ const publishCall = mockApiRequest.mock.calls.find((call) => {
85
+ const req = call[1];
86
+ return req?.path === '/api/cli/publish';
87
+ });
88
+ if (!publishCall)
89
+ throw new Error('Missing publish call');
90
+ const publishBody = publishCall[1].body;
91
+ expect(publishBody.slug).toBe('my-skill');
92
+ expect(publishBody.displayName).toBe('My Skill');
93
+ expect(publishBody.version).toBe('1.0.0');
94
+ expect(publishBody.changelog).toBe('');
95
+ expect(publishBody.tags).toEqual(['latest']);
96
+ const byPath = Object.fromEntries(publishBody.files.map((f) => [f.path, f]));
97
+ expect(Object.keys(byPath).sort()).toEqual(['SKILL.md', 'notes.md']);
98
+ expect(byPath['SKILL.md']?.sha256).toBe(sha256Hex(new TextEncoder().encode(skillContent)));
99
+ expect(byPath['notes.md']?.sha256).toBe(sha256Hex(new TextEncoder().encode(notesContent)));
100
+ }
101
+ finally {
102
+ await rm(workdir, { recursive: true, force: true });
103
+ }
104
+ });
105
+ it('requires --changelog when updating an existing skill', async () => {
106
+ const workdir = await makeTmpWorkdir();
107
+ try {
108
+ const folder = join(workdir, 'existing-skill');
109
+ await mkdir(folder, { recursive: true });
110
+ await writeFile(join(folder, 'SKILL.md'), '# Skill\n', 'utf8');
111
+ mockApiRequest.mockImplementation(async () => ({ skill: { slug: 'existing-skill' } }));
112
+ vi.stubGlobal('fetch', vi.fn(async () => new Response('{}', { status: 200 })));
113
+ await expect(cmdPublish(makeOpts(workdir), 'existing-skill', { version: '1.0.0', changelog: '' })).rejects.toThrow(/changelog/i);
114
+ }
115
+ finally {
116
+ await rm(workdir, { recursive: true, force: true });
117
+ }
118
+ });
119
+ });
120
+ //# sourceMappingURL=publish.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"publish.test.js","sourceRoot":"","sources":["../../../src/cli/commands/publish.test.ts"],"names":[],"mappings":"AAAA,8BAA8B;AAE9B,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAA;AAChE,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAA;AAChC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAA;AAChC,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAA;AAGxC,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,GAAG,EAAE,CAAC,CAAC;IAChC,gBAAgB,EAAE,EAAE,CAAC,EAAE,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,EAAE,QAAQ,EAAE,sBAAsB,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;CAC1F,CAAC,CAAC,CAAA;AAEH,MAAM,eAAe,GAAG,EAAE,CAAC,EAAE,CAAC,KAAK,EAAE,KAAc,EAAE,OAAiB,EAAE,EAAE,CAAC,sBAAsB,CAAC,CAAA;AAClG,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,GAAG,EAAE,CAAC,CAAC;IAC/B,WAAW,EAAE,CAAC,IAAa,EAAE,MAAgB,EAAE,EAAE,CAAC,eAAe,CAAC,IAAI,EAAE,MAAM,CAAC;CAChF,CAAC,CAAC,CAAA;AAEH,MAAM,cAAc,GAAG,EAAE,CAAC,EAAE,EAAE,CAAA;AAC9B,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE,GAAG,EAAE,CAAC,CAAC;IAC9B,UAAU,EAAE,CAAC,QAAiB,EAAE,IAAa,EAAE,MAAgB,EAAE,EAAE,CACjE,cAAc,CAAC,QAAQ,EAAE,IAAI,EAAE,MAAM,CAAC;CACzC,CAAC,CAAC,CAAA;AAEH,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC,OAAe,EAAE,EAAE;IACzC,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,CAAA;AAC1B,CAAC,CAAC,CAAA;AACF,MAAM,WAAW,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,EAAE,EAAE,EAAE,CAAA;AACjE,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE,CAAC,CAAC;IACzB,aAAa,EAAE,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC;IACvC,IAAI,EAAE,CAAC,OAAe,EAAE,EAAE,CAAC,QAAQ,CAAC,OAAO,CAAC;IAC5C,WAAW,EAAE,CAAC,KAAc,EAAE,EAAE,CAAC,CAAC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;CAC1F,CAAC,CAAC,CAAA;AAEH,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,MAAM,CAAC,WAAW,CAAC,CAAA;AAEhD,KAAK,UAAU,cAAc;IAC3B,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,mBAAmB,CAAC,CAAC,CAAA;IAC/D,OAAO,IAAI,CAAA;AACb,CAAC;AAED,SAAS,QAAQ,CAAC,OAAe;IAC/B,OAAO;QACL,OAAO;QACP,GAAG,EAAE,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC;QAC5B,IAAI,EAAE,sBAAsB;QAC5B,QAAQ,EAAE,sBAAsB;KACjC,CAAA;AACH,CAAC;AAED,SAAS,CAAC,GAAG,EAAE;IACb,EAAE,CAAC,gBAAgB,EAAE,CAAA;IACrB,EAAE,CAAC,aAAa,EAAE,CAAA;AACpB,CAAC,CAAC,CAAA;AAEF,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;IAC1B,EAAE,CAAC,4CAA4C,EAAE,KAAK,IAAI,EAAE;QAC1D,MAAM,OAAO,GAAG,MAAM,cAAc,EAAE,CAAA;QACtC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAA;YACxC,MAAM,KAAK,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;YACxC,MAAM,YAAY,GAAG,oBAAoB,CAAA;YACzC,MAAM,YAAY,GAAG,SAAS,CAAA;YAC9B,MAAM,SAAS,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,EAAE,YAAY,EAAE,MAAM,CAAC,CAAA;YAC/D,MAAM,SAAS,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,EAAE,YAAY,EAAE,MAAM,CAAC,CAAA;YAE/D,IAAI,WAAW,GAAG,CAAC,CAAA;YACnB,cAAc,CAAC,kBAAkB,CAC/B,KAAK,EAAE,SAAiB,EAAE,IAAsC,EAAE,EAAE;gBAClE,IAAI,IAAI,CAAC,MAAM,KAAK,KAAK,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,EAAE,CAAC;oBACtE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,aAAa,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,EAAE,CAAA;gBAC7D,CAAC;gBACD,IAAI,IAAI,CAAC,MAAM,KAAK,MAAM,IAAI,IAAI,CAAC,IAAI,KAAK,qBAAqB,EAAE,CAAC;oBAClE,WAAW,IAAI,CAAC,CAAA;oBAChB,OAAO,EAAE,SAAS,EAAE,0BAA0B,WAAW,EAAE,EAAE,CAAA;gBAC/D,CAAC;gBACD,IAAI,IAAI,CAAC,MAAM,KAAK,MAAM,IAAI,IAAI,CAAC,IAAI,KAAK,kBAAkB,EAAE,CAAC;oBAC/D,OAAO,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,EAAE,CAAA;gBAC7D,CAAC;gBACD,MAAM,IAAI,KAAK,CAAC,0BAA0B,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,CAAA;YACvE,CAAC,CACF,CAAA;YAED,EAAE,CAAC,UAAU,CACX,OAAO,EACP,EAAE,CAAC,EAAE,CAAC,KAAK,EAAE,GAAW,EAAE,IAAkB,EAAE,EAAE;gBAC9C,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,kCAAkC,CAAC,CAAA;gBACvD,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;gBACjC,MAAM,CAAE,IAAI,EAAE,OAAkC,EAAE,CAAC,cAAc,CAAC,CAAC,CAAC,OAAO,CACzE,wBAAwB,CACzB,CAAA;gBACD,OAAO,IAAI,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,EAAE,CAAC,EAAE;oBACvF,MAAM,EAAE,GAAG;oBACX,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;iBAChD,CAAC,CAAA;YACJ,CAAC,CAA4B,CAC9B,CAAA;YAED,MAAM,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,UAAU,EAAE;gBAC9C,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,UAAU;gBAChB,OAAO,EAAE,OAAO;gBAChB,SAAS,EAAE,EAAE;gBACb,IAAI,EAAE,QAAQ;aACf,CAAC,CAAA;YAEF,MAAM,WAAW,GAAG,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE;gBAC1D,MAAM,GAAG,GAAG,IAAI,CAAC,CAAC,CAAkC,CAAA;gBACpD,OAAO,GAAG,EAAE,IAAI,KAAK,kBAAkB,CAAA;YACzC,CAAC,CAAC,CAAA;YACF,IAAI,CAAC,WAAW;gBAAE,MAAM,IAAI,KAAK,CAAC,sBAAsB,CAAC,CAAA;YACzD,MAAM,WAAW,GAAI,WAAW,CAAC,CAAC,CAAwB,CAAC,IAO1D,CAAA;YAED,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YACzC,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;YAChD,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;YACzC,MAAM,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACtC,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAA;YAE5C,MAAM,MAAM,GAAG,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAA;YAC5E,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAA;YACpE,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,MAAM,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAA;YAC1F,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,MAAM,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAA;QAC5F,CAAC;gBAAS,CAAC;YACT,MAAM,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAA;QACrD,CAAC;IACH,CAAC,CAAC,CAAA;IAEF,EAAE,CAAC,sDAAsD,EAAE,KAAK,IAAI,EAAE;QACpE,MAAM,OAAO,GAAG,MAAM,cAAc,EAAE,CAAA;QACtC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,EAAE,gBAAgB,CAAC,CAAA;YAC9C,MAAM,KAAK,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;YACxC,MAAM,SAAS,CAAC,IAAI,CAAC,MAAM,EAAE,UAAU,CAAC,EAAE,WAAW,EAAE,MAAM,CAAC,CAAA;YAE9D,cAAc,CAAC,kBAAkB,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,gBAAgB,EAAE,EAAE,CAAC,CAAC,CAAA;YACtF,EAAE,CAAC,UAAU,CACX,OAAO,EACP,EAAE,CAAC,EAAE,CAAC,KAAK,IAAI,EAAE,CAAC,IAAI,QAAQ,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,CAA4B,CAClF,CAAA;YAED,MAAM,MAAM,CACV,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,gBAAgB,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC,CACrF,CAAC,OAAO,CAAC,OAAO,CAAC,YAAY,CAAC,CAAA;QACjC,CAAC;gBAAS,CAAC;YACT,MAAM,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAA;QACrD,CAAC;IACH,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
@@ -0,0 +1,9 @@
1
+ import type { GlobalOpts } from '../types.js';
2
+ export declare function cmdSearch(opts: GlobalOpts, query: string, limit?: number): Promise<void>;
3
+ export declare function cmdInstall(opts: GlobalOpts, slug: string, versionFlag?: string, force?: boolean): Promise<void>;
4
+ export declare function cmdUpdate(opts: GlobalOpts, slugArg: string | undefined, options: {
5
+ all?: boolean;
6
+ version?: string;
7
+ force?: boolean;
8
+ }, inputAllowed: boolean): Promise<void>;
9
+ export declare function cmdList(opts: GlobalOpts): Promise<void>;
@@ -0,0 +1,195 @@
1
+ import { mkdir, rm, stat } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { ApiRoutes, ApiSearchResponseSchema, ApiSkillMetaResponseSchema, ApiSkillResolveResponseSchema, } from 'clawdhub-schema';
4
+ import semver from 'semver';
5
+ import { apiRequest, downloadZip } from '../../http.js';
6
+ import { extractZipToDir, hashSkillFiles, listTextFiles, readLockfile, writeLockfile, } from '../../skills.js';
7
+ import { getRegistry } from '../registry.js';
8
+ import { createSpinner, fail, formatError, isInteractive, promptConfirm } from '../ui.js';
9
+ export async function cmdSearch(opts, query, limit) {
10
+ if (!query)
11
+ fail('Query required');
12
+ const registry = await getRegistry(opts, { cache: true });
13
+ const spinner = createSpinner('Searching');
14
+ try {
15
+ const url = new URL(ApiRoutes.search, registry);
16
+ url.searchParams.set('q', query);
17
+ if (typeof limit === 'number' && Number.isFinite(limit)) {
18
+ url.searchParams.set('limit', String(limit));
19
+ }
20
+ const result = await apiRequest(registry, { method: 'GET', url: url.toString() }, ApiSearchResponseSchema);
21
+ spinner.stop();
22
+ for (const entry of result.results) {
23
+ const slug = entry.slug ?? 'unknown';
24
+ const name = entry.displayName ?? slug;
25
+ const version = entry.version ? ` v${entry.version}` : '';
26
+ console.log(`${slug}${version} ${name} (${entry.score.toFixed(3)})`);
27
+ }
28
+ }
29
+ catch (error) {
30
+ spinner.fail(formatError(error));
31
+ throw error;
32
+ }
33
+ }
34
+ export async function cmdInstall(opts, slug, versionFlag, force = false) {
35
+ const trimmed = slug.trim();
36
+ if (!trimmed)
37
+ fail('Slug required');
38
+ const registry = await getRegistry(opts, { cache: true });
39
+ await mkdir(opts.dir, { recursive: true });
40
+ const target = join(opts.dir, trimmed);
41
+ if (!force) {
42
+ const exists = await fileExists(target);
43
+ if (exists)
44
+ fail(`Already installed: ${target} (use --force)`);
45
+ }
46
+ else {
47
+ await rm(target, { recursive: true, force: true });
48
+ }
49
+ const spinner = createSpinner(`Resolving ${trimmed}`);
50
+ try {
51
+ const resolvedVersion = versionFlag ??
52
+ (await apiRequest(registry, { method: 'GET', path: `/api/skill?slug=${encodeURIComponent(trimmed)}` }, ApiSkillMetaResponseSchema)).latestVersion?.version ??
53
+ null;
54
+ if (!resolvedVersion)
55
+ fail('Could not resolve latest version');
56
+ spinner.text = `Downloading ${trimmed}@${resolvedVersion}`;
57
+ const zip = await downloadZip(registry, { slug: trimmed, version: resolvedVersion });
58
+ await extractZipToDir(zip, target);
59
+ const lock = await readLockfile(opts.workdir);
60
+ lock.skills[trimmed] = {
61
+ version: resolvedVersion,
62
+ installedAt: Date.now(),
63
+ };
64
+ await writeLockfile(opts.workdir, lock);
65
+ spinner.succeed(`OK. Installed ${trimmed} -> ${target}`);
66
+ }
67
+ catch (error) {
68
+ spinner.fail(formatError(error));
69
+ throw error;
70
+ }
71
+ }
72
+ export async function cmdUpdate(opts, slugArg, options, inputAllowed) {
73
+ const slug = slugArg?.trim();
74
+ const all = Boolean(options.all);
75
+ if (!slug && !all)
76
+ fail('Provide <slug> or --all');
77
+ if (slug && all)
78
+ fail('Use either <slug> or --all');
79
+ if (options.version && !slug)
80
+ fail('--version requires a single <slug>');
81
+ if (options.version && !semver.valid(options.version))
82
+ fail('--version must be valid semver');
83
+ const allowPrompt = isInteractive() && inputAllowed !== false;
84
+ const registry = await getRegistry(opts, { cache: true });
85
+ const lock = await readLockfile(opts.workdir);
86
+ const slugs = slug ? [slug] : Object.keys(lock.skills);
87
+ if (slugs.length === 0) {
88
+ console.log('No installed skills.');
89
+ return;
90
+ }
91
+ for (const entry of slugs) {
92
+ const spinner = createSpinner(`Checking ${entry}`);
93
+ try {
94
+ const target = join(opts.dir, entry);
95
+ const exists = await fileExists(target);
96
+ let localFingerprint = null;
97
+ if (exists) {
98
+ const filesOnDisk = await listTextFiles(target);
99
+ if (filesOnDisk.length > 0) {
100
+ const hashed = hashSkillFiles(filesOnDisk);
101
+ localFingerprint = hashed.fingerprint;
102
+ }
103
+ }
104
+ let resolveResult;
105
+ if (localFingerprint) {
106
+ resolveResult = await resolveSkillVersion(registry, entry, localFingerprint);
107
+ }
108
+ else {
109
+ const url = new URL(ApiRoutes.skill, registry);
110
+ url.searchParams.set('slug', entry);
111
+ const meta = await apiRequest(registry, { method: 'GET', url: url.toString() }, ApiSkillMetaResponseSchema);
112
+ resolveResult = { match: null, latestVersion: meta.latestVersion ?? null };
113
+ }
114
+ const latest = resolveResult.latestVersion?.version ?? null;
115
+ const matched = resolveResult.match?.version ?? null;
116
+ if (matched && lock.skills[entry]?.version !== matched) {
117
+ lock.skills[entry] = {
118
+ version: matched,
119
+ installedAt: lock.skills[entry]?.installedAt ?? Date.now(),
120
+ };
121
+ }
122
+ if (!latest) {
123
+ spinner.fail(`${entry}: not found`);
124
+ continue;
125
+ }
126
+ if (!matched && localFingerprint && !options.force) {
127
+ spinner.stop();
128
+ if (!allowPrompt) {
129
+ console.log(`${entry}: local changes (no match). Use --force to overwrite.`);
130
+ continue;
131
+ }
132
+ const confirm = await promptConfirm(`${entry}: local changes (no match). Overwrite with ${options.version ?? latest}?`);
133
+ if (!confirm) {
134
+ console.log(`${entry}: skipped`);
135
+ continue;
136
+ }
137
+ spinner.start(`Updating ${entry} -> ${options.version ?? latest}`);
138
+ }
139
+ const targetVersion = options.version ?? latest;
140
+ if (options.version) {
141
+ if (matched && matched === targetVersion) {
142
+ spinner.succeed(`${entry}: already at ${matched}`);
143
+ continue;
144
+ }
145
+ }
146
+ else if (matched && semver.valid(matched) && semver.gte(matched, targetVersion)) {
147
+ spinner.succeed(`${entry}: up to date (${matched})`);
148
+ continue;
149
+ }
150
+ if (spinner.isSpinning) {
151
+ spinner.text = `Updating ${entry} -> ${targetVersion}`;
152
+ }
153
+ else {
154
+ spinner.start(`Updating ${entry} -> ${targetVersion}`);
155
+ }
156
+ await rm(target, { recursive: true, force: true });
157
+ const zip = await downloadZip(registry, { slug: entry, version: targetVersion });
158
+ await extractZipToDir(zip, target);
159
+ lock.skills[entry] = { version: targetVersion, installedAt: Date.now() };
160
+ spinner.succeed(`${entry}: updated -> ${targetVersion}`);
161
+ }
162
+ catch (error) {
163
+ spinner.fail(formatError(error));
164
+ throw error;
165
+ }
166
+ }
167
+ await writeLockfile(opts.workdir, lock);
168
+ }
169
+ export async function cmdList(opts) {
170
+ const lock = await readLockfile(opts.workdir);
171
+ const entries = Object.entries(lock.skills);
172
+ if (entries.length === 0) {
173
+ console.log('No installed skills.');
174
+ return;
175
+ }
176
+ for (const [slug, entry] of entries) {
177
+ console.log(`${slug} ${entry.version ?? 'latest'}`);
178
+ }
179
+ }
180
+ async function resolveSkillVersion(registry, slug, hash) {
181
+ const url = new URL(ApiRoutes.skillResolve, registry);
182
+ url.searchParams.set('slug', slug);
183
+ url.searchParams.set('hash', hash);
184
+ return apiRequest(registry, { method: 'GET', url: url.toString() }, ApiSkillResolveResponseSchema);
185
+ }
186
+ async function fileExists(path) {
187
+ try {
188
+ await stat(path);
189
+ return true;
190
+ }
191
+ catch {
192
+ return false;
193
+ }
194
+ }
195
+ //# sourceMappingURL=skills.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"skills.js","sourceRoot":"","sources":["../../../src/cli/commands/skills.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { GlobalOpts } from '../types.js';
2
+ type SyncOptions = {
3
+ root?: string[];
4
+ all?: boolean;
5
+ dryRun?: boolean;
6
+ bump?: 'patch' | 'minor' | 'major';
7
+ changelog?: string;
8
+ tags?: string;
9
+ };
10
+ export declare function cmdSync(opts: GlobalOpts, options: SyncOptions, inputAllowed: boolean): Promise<void>;
11
+ export {};
@@ -0,0 +1,273 @@
1
+ import { homedir } from 'node:os';
2
+ import { resolve } from 'node:path';
3
+ import { intro, isCancel, multiselect, note, outro, text } from '@clack/prompts';
4
+ import { ApiCliWhoamiResponseSchema, ApiRoutes, ApiSkillMetaResponseSchema, ApiSkillResolveResponseSchema, } from 'clawdhub-schema';
5
+ import semver from 'semver';
6
+ import { readGlobalConfig } from '../../config.js';
7
+ import { apiRequest, downloadZip } from '../../http.js';
8
+ import { hashSkillFiles, hashSkillZip, listTextFiles } from '../../skills.js';
9
+ import { getRegistry } from '../registry.js';
10
+ import { findSkillFolders, getFallbackSkillRoots } from '../scanSkills.js';
11
+ import { createSpinner, fail, formatError, isInteractive } from '../ui.js';
12
+ import { cmdPublish } from './publish.js';
13
+ export async function cmdSync(opts, options, inputAllowed) {
14
+ const allowPrompt = isInteractive() && inputAllowed !== false;
15
+ intro('ClawdHub sync');
16
+ const cfg = await readGlobalConfig();
17
+ const token = cfg?.token;
18
+ if (!token)
19
+ fail('Not logged in. Run: clawdhub login');
20
+ const registry = await getRegistryWithAuth(opts, token);
21
+ const selectedRoots = buildScanRoots(opts, options.root);
22
+ const spinner = createSpinner('Scanning for local skills');
23
+ let scan = await scanRoots(selectedRoots);
24
+ if (scan.skills.length === 0) {
25
+ const fallback = getFallbackSkillRoots(opts.workdir);
26
+ scan = await scanRoots(fallback);
27
+ spinner.stop();
28
+ if (scan.skills.length === 0)
29
+ fail('No skills found (checked workdir and known Clawdis/Clawd locations)');
30
+ note(`No skills in workdir. Found ${scan.skills.length} in legacy locations.`, formatList(scan.rootsWithSkills, 10));
31
+ }
32
+ else {
33
+ spinner.stop();
34
+ }
35
+ let skills = scan.skills;
36
+ skills = await maybeSelectLocalSkills(skills, {
37
+ allowPrompt,
38
+ all: Boolean(options.all),
39
+ });
40
+ if (skills.length === 0) {
41
+ outro('Nothing selected.');
42
+ return;
43
+ }
44
+ const candidatesSpinner = createSpinner('Checking registry sync state');
45
+ const candidates = [];
46
+ let supportsResolve = null;
47
+ try {
48
+ for (const skill of skills) {
49
+ const filesOnDisk = await listTextFiles(skill.folder);
50
+ const hashed = hashSkillFiles(filesOnDisk);
51
+ const fingerprint = hashed.fingerprint;
52
+ const meta = await apiRequest(registry, { method: 'GET', path: `${ApiRoutes.skill}?slug=${encodeURIComponent(skill.slug)}` }, ApiSkillMetaResponseSchema).catch(() => null);
53
+ const latestVersion = meta?.latestVersion?.version ?? null;
54
+ if (!latestVersion) {
55
+ candidates.push({
56
+ ...skill,
57
+ fingerprint,
58
+ fileCount: filesOnDisk.length,
59
+ status: 'new',
60
+ matchVersion: null,
61
+ latestVersion: null,
62
+ });
63
+ continue;
64
+ }
65
+ let matchVersion = null;
66
+ if (supportsResolve !== false) {
67
+ try {
68
+ const resolved = await apiRequest(registry, {
69
+ method: 'GET',
70
+ path: `${ApiRoutes.skillResolve}?slug=${encodeURIComponent(skill.slug)}&hash=${encodeURIComponent(fingerprint)}`,
71
+ }, ApiSkillResolveResponseSchema);
72
+ supportsResolve = true;
73
+ matchVersion = resolved.match?.version ?? null;
74
+ }
75
+ catch (error) {
76
+ const message = formatError(error);
77
+ if (/skill not found/i.test(message)) {
78
+ matchVersion = null;
79
+ }
80
+ else if (/no matching routes found/i.test(message) || /not found/i.test(message)) {
81
+ supportsResolve = false;
82
+ }
83
+ else {
84
+ throw error;
85
+ }
86
+ }
87
+ }
88
+ if (supportsResolve === false) {
89
+ const zip = await downloadZip(registry, { slug: skill.slug, version: latestVersion });
90
+ const remote = hashSkillZip(zip).fingerprint;
91
+ matchVersion = remote === fingerprint ? latestVersion : null;
92
+ }
93
+ candidates.push({
94
+ ...skill,
95
+ fingerprint,
96
+ fileCount: filesOnDisk.length,
97
+ status: matchVersion ? 'synced' : 'update',
98
+ matchVersion,
99
+ latestVersion,
100
+ });
101
+ }
102
+ }
103
+ catch (error) {
104
+ candidatesSpinner.fail(formatError(error));
105
+ throw error;
106
+ }
107
+ finally {
108
+ candidatesSpinner.stop();
109
+ }
110
+ const synced = candidates.filter((candidate) => candidate.status === 'synced');
111
+ if (synced.length > 0) {
112
+ const lines = synced
113
+ .map((candidate) => `${candidate.slug} synced (${candidate.matchVersion ?? 'unknown'})`)
114
+ .join('\n');
115
+ note('Already synced', lines);
116
+ }
117
+ const actionable = candidates.filter((candidate) => candidate.status !== 'synced');
118
+ if (actionable.length === 0) {
119
+ outro('Everything is already synced.');
120
+ return;
121
+ }
122
+ const selected = await selectToUpload(actionable, {
123
+ allowPrompt,
124
+ all: Boolean(options.all),
125
+ bump: options.bump ?? 'patch',
126
+ });
127
+ if (selected.length === 0) {
128
+ outro('Nothing selected.');
129
+ return;
130
+ }
131
+ if (options.dryRun) {
132
+ outro(`Dry run: would upload ${selected.length} skill(s).`);
133
+ return;
134
+ }
135
+ const bump = options.bump ?? 'patch';
136
+ const tags = options.tags ?? 'latest';
137
+ for (const skill of selected) {
138
+ const { publishVersion, changelog } = await resolvePublishMeta(skill, {
139
+ bump,
140
+ allowPrompt,
141
+ changelogFlag: options.changelog,
142
+ });
143
+ await cmdPublish(opts, skill.folder, {
144
+ slug: skill.slug,
145
+ name: skill.displayName,
146
+ version: publishVersion,
147
+ changelog,
148
+ tags,
149
+ });
150
+ }
151
+ outro(`Uploaded ${selected.length} skill(s).`);
152
+ }
153
+ function buildScanRoots(opts, extraRoots) {
154
+ const roots = [opts.workdir, opts.dir, ...(extraRoots ?? [])];
155
+ return Array.from(new Set(roots.map((root) => resolve(root))));
156
+ }
157
+ async function scanRoots(roots) {
158
+ const all = [];
159
+ const rootsWithSkills = [];
160
+ for (const root of roots) {
161
+ const found = await findSkillFolders(root);
162
+ if (found.length > 0)
163
+ rootsWithSkills.push(root);
164
+ all.push(...found);
165
+ }
166
+ const byFolder = new Map();
167
+ for (const folder of all) {
168
+ byFolder.set(folder.folder, folder);
169
+ }
170
+ return { skills: Array.from(byFolder.values()), rootsWithSkills };
171
+ }
172
+ async function maybeSelectLocalSkills(skills, params) {
173
+ if (params.all || !params.allowPrompt)
174
+ return skills;
175
+ if (skills.length <= 30)
176
+ return skills;
177
+ const valueByKey = new Map();
178
+ const choices = skills.map((skill) => {
179
+ const key = skill.folder;
180
+ valueByKey.set(key, skill);
181
+ return {
182
+ value: key,
183
+ label: skill.slug,
184
+ hint: abbreviatePath(skill.folder),
185
+ };
186
+ });
187
+ const picked = await multiselect({
188
+ message: `Found ${skills.length} local skills — select what to sync`,
189
+ options: choices,
190
+ initialValues: [],
191
+ required: false,
192
+ });
193
+ if (isCancel(picked))
194
+ fail('Canceled');
195
+ return picked.map((key) => valueByKey.get(String(key))).filter(Boolean);
196
+ }
197
+ async function selectToUpload(candidates, params) {
198
+ if (params.all || !params.allowPrompt)
199
+ return candidates;
200
+ const valueByKey = new Map();
201
+ const choices = candidates.map((candidate) => {
202
+ const key = candidate.folder;
203
+ valueByKey.set(key, candidate);
204
+ const latest = candidate.latestVersion;
205
+ const next = latest ? semver.inc(latest, params.bump) : null;
206
+ const status = candidate.status === 'new' ? 'NEW' : latest && next ? `UPDATE ${latest} → ${next}` : 'UPDATE';
207
+ return {
208
+ value: key,
209
+ label: `${candidate.slug} ${status}`,
210
+ hint: candidate.folder,
211
+ };
212
+ });
213
+ const picked = await multiselect({
214
+ message: 'Select skills to upload',
215
+ options: choices,
216
+ initialValues: candidates.length <= 10 ? choices.map((choice) => choice.value) : [],
217
+ required: false,
218
+ });
219
+ if (isCancel(picked))
220
+ fail('Canceled');
221
+ const selected = picked.map((key) => valueByKey.get(String(key))).filter(Boolean);
222
+ return selected;
223
+ }
224
+ async function resolvePublishMeta(skill, params) {
225
+ if (skill.status === 'new') {
226
+ return { publishVersion: '1.0.0', changelog: '' };
227
+ }
228
+ const latest = skill.latestVersion;
229
+ if (!latest)
230
+ fail(`Could not resolve latest version for ${skill.slug}`);
231
+ const publishVersion = semver.inc(latest, params.bump);
232
+ if (!publishVersion)
233
+ fail(`Could not bump version for ${skill.slug}`);
234
+ const fromFlag = params.changelogFlag?.trim();
235
+ if (fromFlag)
236
+ return { publishVersion, changelog: fromFlag };
237
+ if (!params.allowPrompt) {
238
+ return { publishVersion, changelog: 'Sync update' };
239
+ }
240
+ const entered = await text({
241
+ message: `Changelog for ${skill.slug}@${publishVersion}`,
242
+ placeholder: 'What changed?',
243
+ defaultValue: 'Sync update',
244
+ });
245
+ if (isCancel(entered))
246
+ fail('Canceled');
247
+ const changelog = String(entered ?? '').trim();
248
+ if (!changelog)
249
+ fail('--changelog required for updates');
250
+ return { publishVersion, changelog };
251
+ }
252
+ async function getRegistryWithAuth(opts, token) {
253
+ const registry = await getRegistry(opts, { cache: true });
254
+ await apiRequest(registry, { method: 'GET', path: ApiRoutes.cliWhoami, token }, ApiCliWhoamiResponseSchema);
255
+ return registry;
256
+ }
257
+ function formatList(values, max) {
258
+ if (values.length === 0)
259
+ return '';
260
+ const shown = values.map(abbreviatePath);
261
+ if (shown.length <= max)
262
+ return shown.join('\n');
263
+ const head = shown.slice(0, Math.max(1, max - 1));
264
+ const rest = values.length - head.length;
265
+ return [...head, `… +${rest} more`].join('\n');
266
+ }
267
+ function abbreviatePath(value) {
268
+ const home = homedir();
269
+ if (value.startsWith(home))
270
+ return `~${value.slice(home.length)}`;
271
+ return value;
272
+ }
273
+ //# sourceMappingURL=sync.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync.js","sourceRoot":"","sources":["../../../src/cli/commands/sync.ts"],"names":[],"mappings":"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"}