@proxygate/cli 0.1.0

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 (249) hide show
  1. package/LICENSE +17 -0
  2. package/README.md +311 -0
  3. package/dist/commands/apis.d.ts +3 -0
  4. package/dist/commands/apis.d.ts.map +1 -0
  5. package/dist/commands/apis.js +68 -0
  6. package/dist/commands/apis.js.map +1 -0
  7. package/dist/commands/apis.test.d.ts +2 -0
  8. package/dist/commands/apis.test.d.ts.map +1 -0
  9. package/dist/commands/apis.test.js +103 -0
  10. package/dist/commands/apis.test.js.map +1 -0
  11. package/dist/commands/balance.d.ts +9 -0
  12. package/dist/commands/balance.d.ts.map +1 -0
  13. package/dist/commands/balance.js +50 -0
  14. package/dist/commands/balance.js.map +1 -0
  15. package/dist/commands/balance.test.d.ts +2 -0
  16. package/dist/commands/balance.test.d.ts.map +1 -0
  17. package/dist/commands/balance.test.js +106 -0
  18. package/dist/commands/balance.test.js.map +1 -0
  19. package/dist/commands/categories.d.ts +3 -0
  20. package/dist/commands/categories.d.ts.map +1 -0
  21. package/dist/commands/categories.js +45 -0
  22. package/dist/commands/categories.js.map +1 -0
  23. package/dist/commands/categories.test.d.ts +2 -0
  24. package/dist/commands/categories.test.d.ts.map +1 -0
  25. package/dist/commands/categories.test.js +84 -0
  26. package/dist/commands/categories.test.js.map +1 -0
  27. package/dist/commands/create.d.ts +3 -0
  28. package/dist/commands/create.d.ts.map +1 -0
  29. package/dist/commands/create.js +136 -0
  30. package/dist/commands/create.js.map +1 -0
  31. package/dist/commands/create.test.d.ts +2 -0
  32. package/dist/commands/create.test.d.ts.map +1 -0
  33. package/dist/commands/create.test.js +95 -0
  34. package/dist/commands/create.test.js.map +1 -0
  35. package/dist/commands/deposit.d.ts +9 -0
  36. package/dist/commands/deposit.d.ts.map +1 -0
  37. package/dist/commands/deposit.js +78 -0
  38. package/dist/commands/deposit.js.map +1 -0
  39. package/dist/commands/deposit.test.d.ts +2 -0
  40. package/dist/commands/deposit.test.d.ts.map +1 -0
  41. package/dist/commands/deposit.test.js +93 -0
  42. package/dist/commands/deposit.test.js.map +1 -0
  43. package/dist/commands/dev.d.ts +3 -0
  44. package/dist/commands/dev.d.ts.map +1 -0
  45. package/dist/commands/dev.js +188 -0
  46. package/dist/commands/dev.js.map +1 -0
  47. package/dist/commands/dev.test.d.ts +2 -0
  48. package/dist/commands/dev.test.d.ts.map +1 -0
  49. package/dist/commands/dev.test.js +89 -0
  50. package/dist/commands/dev.test.js.map +1 -0
  51. package/dist/commands/getting-started-steps.d.ts +12 -0
  52. package/dist/commands/getting-started-steps.d.ts.map +1 -0
  53. package/dist/commands/getting-started-steps.js +163 -0
  54. package/dist/commands/getting-started-steps.js.map +1 -0
  55. package/dist/commands/getting-started.d.ts +9 -0
  56. package/dist/commands/getting-started.d.ts.map +1 -0
  57. package/dist/commands/getting-started.js +44 -0
  58. package/dist/commands/getting-started.js.map +1 -0
  59. package/dist/commands/init.d.ts +9 -0
  60. package/dist/commands/init.d.ts.map +1 -0
  61. package/dist/commands/init.js +74 -0
  62. package/dist/commands/init.js.map +1 -0
  63. package/dist/commands/init.test.d.ts +2 -0
  64. package/dist/commands/init.test.d.ts.map +1 -0
  65. package/dist/commands/init.test.js +115 -0
  66. package/dist/commands/init.test.js.map +1 -0
  67. package/dist/commands/jobs.d.ts +11 -0
  68. package/dist/commands/jobs.d.ts.map +1 -0
  69. package/dist/commands/jobs.js +370 -0
  70. package/dist/commands/jobs.js.map +1 -0
  71. package/dist/commands/listings/create.d.ts +4 -0
  72. package/dist/commands/listings/create.d.ts.map +1 -0
  73. package/dist/commands/listings/create.js +133 -0
  74. package/dist/commands/listings/create.js.map +1 -0
  75. package/dist/commands/listings/delete.d.ts +4 -0
  76. package/dist/commands/listings/delete.d.ts.map +1 -0
  77. package/dist/commands/listings/delete.js +32 -0
  78. package/dist/commands/listings/delete.js.map +1 -0
  79. package/dist/commands/listings/docs.d.ts +4 -0
  80. package/dist/commands/listings/docs.d.ts.map +1 -0
  81. package/dist/commands/listings/docs.js +48 -0
  82. package/dist/commands/listings/docs.js.map +1 -0
  83. package/dist/commands/listings/headers.d.ts +4 -0
  84. package/dist/commands/listings/headers.d.ts.map +1 -0
  85. package/dist/commands/listings/headers.js +78 -0
  86. package/dist/commands/listings/headers.js.map +1 -0
  87. package/dist/commands/listings/helpers.d.ts +10 -0
  88. package/dist/commands/listings/helpers.d.ts.map +1 -0
  89. package/dist/commands/listings/helpers.js +64 -0
  90. package/dist/commands/listings/helpers.js.map +1 -0
  91. package/dist/commands/listings/index.d.ts +11 -0
  92. package/dist/commands/listings/index.d.ts.map +1 -0
  93. package/dist/commands/listings/index.js +46 -0
  94. package/dist/commands/listings/index.js.map +1 -0
  95. package/dist/commands/listings/list.d.ts +4 -0
  96. package/dist/commands/listings/list.d.ts.map +1 -0
  97. package/dist/commands/listings/list.js +42 -0
  98. package/dist/commands/listings/list.js.map +1 -0
  99. package/dist/commands/listings/pause-unpause.d.ts +6 -0
  100. package/dist/commands/listings/pause-unpause.d.ts.map +1 -0
  101. package/dist/commands/listings/pause-unpause.js +37 -0
  102. package/dist/commands/listings/pause-unpause.js.map +1 -0
  103. package/dist/commands/listings/rotate-key.d.ts +4 -0
  104. package/dist/commands/listings/rotate-key.d.ts.map +1 -0
  105. package/dist/commands/listings/rotate-key.js +62 -0
  106. package/dist/commands/listings/rotate-key.js.map +1 -0
  107. package/dist/commands/listings/update.d.ts +4 -0
  108. package/dist/commands/listings/update.d.ts.map +1 -0
  109. package/dist/commands/listings/update.js +57 -0
  110. package/dist/commands/listings/update.js.map +1 -0
  111. package/dist/commands/listings.d.ts +2 -0
  112. package/dist/commands/listings.d.ts.map +1 -0
  113. package/dist/commands/listings.js +3 -0
  114. package/dist/commands/listings.js.map +1 -0
  115. package/dist/commands/listings.test.d.ts +2 -0
  116. package/dist/commands/listings.test.d.ts.map +1 -0
  117. package/dist/commands/listings.test.js +166 -0
  118. package/dist/commands/listings.test.js.map +1 -0
  119. package/dist/commands/pricing.d.ts +9 -0
  120. package/dist/commands/pricing.d.ts.map +1 -0
  121. package/dist/commands/pricing.js +67 -0
  122. package/dist/commands/pricing.js.map +1 -0
  123. package/dist/commands/pricing.test.d.ts +2 -0
  124. package/dist/commands/pricing.test.d.ts.map +1 -0
  125. package/dist/commands/pricing.test.js +87 -0
  126. package/dist/commands/pricing.test.js.map +1 -0
  127. package/dist/commands/proxy.d.ts +14 -0
  128. package/dist/commands/proxy.d.ts.map +1 -0
  129. package/dist/commands/proxy.js +138 -0
  130. package/dist/commands/proxy.js.map +1 -0
  131. package/dist/commands/proxy.test.d.ts +2 -0
  132. package/dist/commands/proxy.test.d.ts.map +1 -0
  133. package/dist/commands/proxy.test.js +235 -0
  134. package/dist/commands/proxy.test.js.map +1 -0
  135. package/dist/commands/rate.d.ts +3 -0
  136. package/dist/commands/rate.d.ts.map +1 -0
  137. package/dist/commands/rate.js +46 -0
  138. package/dist/commands/rate.js.map +1 -0
  139. package/dist/commands/rate.test.d.ts +2 -0
  140. package/dist/commands/rate.test.d.ts.map +1 -0
  141. package/dist/commands/rate.test.js +74 -0
  142. package/dist/commands/rate.test.js.map +1 -0
  143. package/dist/commands/services.d.ts +3 -0
  144. package/dist/commands/services.d.ts.map +1 -0
  145. package/dist/commands/services.js +48 -0
  146. package/dist/commands/services.js.map +1 -0
  147. package/dist/commands/services.test.d.ts +2 -0
  148. package/dist/commands/services.test.d.ts.map +1 -0
  149. package/dist/commands/services.test.js +87 -0
  150. package/dist/commands/services.test.js.map +1 -0
  151. package/dist/commands/settlements.d.ts +3 -0
  152. package/dist/commands/settlements.d.ts.map +1 -0
  153. package/dist/commands/settlements.js +79 -0
  154. package/dist/commands/settlements.js.map +1 -0
  155. package/dist/commands/settlements.test.d.ts +2 -0
  156. package/dist/commands/settlements.test.d.ts.map +1 -0
  157. package/dist/commands/settlements.test.js +98 -0
  158. package/dist/commands/settlements.test.js.map +1 -0
  159. package/dist/commands/skills.d.ts +3 -0
  160. package/dist/commands/skills.d.ts.map +1 -0
  161. package/dist/commands/skills.js +96 -0
  162. package/dist/commands/skills.js.map +1 -0
  163. package/dist/commands/skills.test.d.ts +2 -0
  164. package/dist/commands/skills.test.d.ts.map +1 -0
  165. package/dist/commands/skills.test.js +89 -0
  166. package/dist/commands/skills.test.js.map +1 -0
  167. package/dist/commands/test-service.d.ts +3 -0
  168. package/dist/commands/test-service.d.ts.map +1 -0
  169. package/dist/commands/test-service.js +200 -0
  170. package/dist/commands/test-service.js.map +1 -0
  171. package/dist/commands/test-service.test.d.ts +2 -0
  172. package/dist/commands/test-service.test.d.ts.map +1 -0
  173. package/dist/commands/test-service.test.js +119 -0
  174. package/dist/commands/test-service.test.js.map +1 -0
  175. package/dist/commands/tunnel-handlers.d.ts +19 -0
  176. package/dist/commands/tunnel-handlers.d.ts.map +1 -0
  177. package/dist/commands/tunnel-handlers.js +103 -0
  178. package/dist/commands/tunnel-handlers.js.map +1 -0
  179. package/dist/commands/tunnel.d.ts +7 -0
  180. package/dist/commands/tunnel.d.ts.map +1 -0
  181. package/dist/commands/tunnel.js +184 -0
  182. package/dist/commands/tunnel.js.map +1 -0
  183. package/dist/commands/usage.d.ts +9 -0
  184. package/dist/commands/usage.d.ts.map +1 -0
  185. package/dist/commands/usage.js +78 -0
  186. package/dist/commands/usage.js.map +1 -0
  187. package/dist/commands/usage.test.d.ts +2 -0
  188. package/dist/commands/usage.test.d.ts.map +1 -0
  189. package/dist/commands/usage.test.js +102 -0
  190. package/dist/commands/usage.test.js.map +1 -0
  191. package/dist/commands/withdraw-confirm.d.ts +10 -0
  192. package/dist/commands/withdraw-confirm.d.ts.map +1 -0
  193. package/dist/commands/withdraw-confirm.js +51 -0
  194. package/dist/commands/withdraw-confirm.js.map +1 -0
  195. package/dist/commands/withdraw-confirm.test.d.ts +2 -0
  196. package/dist/commands/withdraw-confirm.test.d.ts.map +1 -0
  197. package/dist/commands/withdraw-confirm.test.js +75 -0
  198. package/dist/commands/withdraw-confirm.test.js.map +1 -0
  199. package/dist/commands/withdraw.d.ts +9 -0
  200. package/dist/commands/withdraw.d.ts.map +1 -0
  201. package/dist/commands/withdraw.js +78 -0
  202. package/dist/commands/withdraw.js.map +1 -0
  203. package/dist/commands/withdraw.test.d.ts +2 -0
  204. package/dist/commands/withdraw.test.d.ts.map +1 -0
  205. package/dist/commands/withdraw.test.js +96 -0
  206. package/dist/commands/withdraw.test.js.map +1 -0
  207. package/dist/config.d.ts +20 -0
  208. package/dist/config.d.ts.map +1 -0
  209. package/dist/config.js +38 -0
  210. package/dist/config.js.map +1 -0
  211. package/dist/config.test.d.ts +2 -0
  212. package/dist/config.test.d.ts.map +1 -0
  213. package/dist/config.test.js +54 -0
  214. package/dist/config.test.js.map +1 -0
  215. package/dist/format.d.ts +46 -0
  216. package/dist/format.d.ts.map +1 -0
  217. package/dist/format.js +101 -0
  218. package/dist/format.js.map +1 -0
  219. package/dist/format.test.d.ts +2 -0
  220. package/dist/format.test.d.ts.map +1 -0
  221. package/dist/format.test.js +85 -0
  222. package/dist/format.test.js.map +1 -0
  223. package/dist/generated/skills.d.ts +2 -0
  224. package/dist/generated/skills.d.ts.map +1 -0
  225. package/dist/generated/skills.js +23 -0
  226. package/dist/generated/skills.js.map +1 -0
  227. package/dist/helpers.d.ts +12 -0
  228. package/dist/helpers.d.ts.map +1 -0
  229. package/dist/helpers.js +21 -0
  230. package/dist/helpers.js.map +1 -0
  231. package/dist/helpers.test.d.ts +2 -0
  232. package/dist/helpers.test.d.ts.map +1 -0
  233. package/dist/helpers.test.js +59 -0
  234. package/dist/helpers.test.js.map +1 -0
  235. package/dist/index.d.ts +3 -0
  236. package/dist/index.d.ts.map +1 -0
  237. package/dist/index.js +70 -0
  238. package/dist/index.js.map +1 -0
  239. package/package.json +61 -0
  240. package/templates/http-api/README.md +20 -0
  241. package/templates/http-api/package.json +21 -0
  242. package/templates/http-api/proxygate.tunnel.yaml +9 -0
  243. package/templates/http-api/src/index.ts +77 -0
  244. package/templates/http-api/tsconfig.json +14 -0
  245. package/templates/llm-agent/README.md +23 -0
  246. package/templates/llm-agent/package.json +22 -0
  247. package/templates/llm-agent/proxygate.tunnel.yaml +13 -0
  248. package/templates/llm-agent/src/index.ts +142 -0
  249. package/templates/llm-agent/tsconfig.json +14 -0
@@ -0,0 +1,106 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerBalanceCommand } from './balance.js';
4
+ // Mock the SDK with vault.balance() pattern
5
+ const mockBalance = vi.fn();
6
+ vi.mock('@proxygate/sdk', () => ({
7
+ ProxyGateClient: {
8
+ create: vi.fn().mockResolvedValue({
9
+ vault: {
10
+ balance: (...args) => mockBalance(...args),
11
+ },
12
+ }),
13
+ },
14
+ ProxyGateError: class ProxyGateError extends Error {
15
+ code;
16
+ action;
17
+ constructor(msg, code, action) {
18
+ super(msg);
19
+ this.code = code;
20
+ this.action = action;
21
+ }
22
+ },
23
+ }));
24
+ vi.mock('../config.js', () => ({
25
+ loadConfig: vi.fn().mockResolvedValue({
26
+ gatewayUrl: 'http://localhost:3001',
27
+ keypairPath: '/tmp/test-key.json',
28
+ }),
29
+ }));
30
+ describe('balance command', () => {
31
+ let logSpy;
32
+ beforeEach(() => {
33
+ vi.clearAllMocks();
34
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
35
+ vi.spyOn(console, 'error').mockImplementation(() => { });
36
+ });
37
+ afterEach(() => {
38
+ vi.restoreAllMocks();
39
+ });
40
+ const runBalance = async (...args) => {
41
+ const program = new Command('proxygate');
42
+ program
43
+ .option('--gateway <url>', 'Override gateway URL')
44
+ .option('--keypair <path>', 'Override keypair path')
45
+ .option('--json', 'Output raw JSON');
46
+ registerBalanceCommand(program);
47
+ await program.parseAsync(['node', 'proxygate', 'balance', ...args]);
48
+ };
49
+ it('outputs formatted vault balance with breakdown', async () => {
50
+ mockBalance.mockResolvedValue({
51
+ balance: 3_500_000,
52
+ pending_settlement: 250_000,
53
+ available: 3_250_000,
54
+ in_cooldown: false,
55
+ currency: 'lamports',
56
+ });
57
+ await runBalance();
58
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
59
+ expect(output).toContain('Vault Balance');
60
+ expect(output).toContain('3.500000 USDC');
61
+ expect(output).toContain('0.250000 USDC');
62
+ expect(output).toContain('3.250000 USDC');
63
+ expect(output).toContain('Cooldown:');
64
+ expect(output).toContain('No');
65
+ });
66
+ it('shows cooldown status when in cooldown', async () => {
67
+ mockBalance.mockResolvedValue({
68
+ balance: 3_500_000,
69
+ pending_settlement: 250_000,
70
+ available: 3_250_000,
71
+ in_cooldown: true,
72
+ currency: 'lamports',
73
+ });
74
+ await runBalance();
75
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
76
+ expect(output).toContain('Cooldown:');
77
+ expect(output).toContain('Yes');
78
+ });
79
+ it('outputs raw JSON with --json flag', async () => {
80
+ const result = {
81
+ balance: 3_500_000,
82
+ pending_settlement: 250_000,
83
+ available: 3_250_000,
84
+ in_cooldown: false,
85
+ currency: 'lamports',
86
+ };
87
+ mockBalance.mockResolvedValue(result);
88
+ await runBalance('--json');
89
+ expect(logSpy).toHaveBeenCalledTimes(1);
90
+ const parsed = JSON.parse(logSpy.mock.calls[0][0]);
91
+ expect(parsed).toEqual(result);
92
+ });
93
+ it('--json output does not contain ANSI codes', async () => {
94
+ mockBalance.mockResolvedValue({
95
+ balance: 100,
96
+ pending_settlement: 0,
97
+ available: 100,
98
+ in_cooldown: false,
99
+ currency: 'lamports',
100
+ });
101
+ await runBalance('--json');
102
+ const output = logSpy.mock.calls[0][0];
103
+ expect(output).not.toContain('\x1b[');
104
+ });
105
+ });
106
+ //# sourceMappingURL=balance.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"balance.test.js","sourceRoot":"","sources":["../../src/commands/balance.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { Command } from 'commander';
2
+ export declare function registerCategoriesCommand(program: Command): void;
3
+ //# sourceMappingURL=categories.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"categories.d.ts","sourceRoot":"","sources":["../../src/commands/categories.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAKzC,wBAAgB,yBAAyB,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA+ChE"}
@@ -0,0 +1,45 @@
1
+ import { ProxyGateError } from '@proxygate/sdk';
2
+ import { getClient } from '../helpers.js';
3
+ import { bold, red, dim, cyan, formatTable } from '../format.js';
4
+ export function registerCategoriesCommand(program) {
5
+ program
6
+ .command('categories')
7
+ .description('List API categories (no auth required)')
8
+ .addHelpText('after', '\nExamples:\n' +
9
+ ' $ proxygate categories\n' +
10
+ ' $ proxygate categories --json')
11
+ .action(async () => {
12
+ const parentOpts = program.opts();
13
+ try {
14
+ const client = await getClient(parentOpts);
15
+ const result = await client.categories();
16
+ if (parentOpts.json) {
17
+ console.log(JSON.stringify(result, null, 2));
18
+ return;
19
+ }
20
+ if (result.categories.length === 0) {
21
+ console.log(dim('No categories available.'));
22
+ return;
23
+ }
24
+ console.log(bold('Categories'));
25
+ console.log();
26
+ const headers = ['Category', 'Listings', 'Subcategories'];
27
+ const rows = result.categories.map((c) => [
28
+ `${c.icon} ${bold(cyan(c.name))} ${dim(`(${c.slug})`)}`,
29
+ String(c.listing_count),
30
+ c.subcategories.map((s) => s.name).join(', ') || dim('none'),
31
+ ]);
32
+ console.log(formatTable(headers, rows));
33
+ }
34
+ catch (err) {
35
+ if (err instanceof ProxyGateError) {
36
+ console.error(red(`Error [${err.code}]: ${err.message}`));
37
+ if (err.action)
38
+ console.error(dim(`Suggestion: ${err.action}`));
39
+ process.exit(1);
40
+ }
41
+ throw err;
42
+ }
43
+ });
44
+ }
45
+ //# sourceMappingURL=categories.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"categories.js","sourceRoot":"","sources":["../../src/commands/categories.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1C,OAAO,EAAE,IAAI,EAAE,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,cAAc,CAAC;AAEjE,MAAM,UAAU,yBAAyB,CAAC,OAAgB;IACxD,OAAO;SACJ,OAAO,CAAC,YAAY,CAAC;SACrB,WAAW,CAAC,wCAAwC,CAAC;SACrD,WAAW,CACV,OAAO,EACP,eAAe;QACb,4BAA4B;QAC5B,iCAAiC,CACpC;SACA,MAAM,CAAC,KAAK,IAAI,EAAE;QACjB,MAAM,UAAU,GAAG,OAAO,CAAC,IAAI,EAA0D,CAAC;QAE1F,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,UAAU,CAAC,CAAC;YAC3C,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,UAAU,EAAE,CAAC;YAEzC,IAAI,UAAU,CAAC,IAAI,EAAE,CAAC;gBACpB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;gBAC7C,OAAO;YACT,CAAC;YAED,IAAI,MAAM,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACnC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC,CAAC;gBAC7C,OAAO;YACT,CAAC;YAED,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;YAChC,OAAO,CAAC,GAAG,EAAE,CAAC;YAEd,MAAM,OAAO,GAAG,CAAC,UAAU,EAAE,UAAU,EAAE,eAAe,CAAC,CAAC;YAC1D,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;gBACxC,GAAG,CAAC,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,EAAE;gBACvD,MAAM,CAAC,CAAC,CAAC,aAAa,CAAC;gBACvB,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC;aAC7D,CAAC,CAAC;YAEH,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC;QAC1C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,cAAc,EAAE,CAAC;gBAClC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,GAAG,CAAC,IAAI,MAAM,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBAC1D,IAAI,GAAG,CAAC,MAAM;oBAAE,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,eAAe,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;gBAChE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC,CAAC,CAAC;AACP,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=categories.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"categories.test.d.ts","sourceRoot":"","sources":["../../src/commands/categories.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,84 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerCategoriesCommand } from './categories.js';
4
+ const mockCategories = vi.fn();
5
+ vi.mock('@proxygate/sdk', () => ({
6
+ ProxyGateClient: {
7
+ create: vi.fn().mockResolvedValue({
8
+ categories: (...args) => mockCategories(...args),
9
+ }),
10
+ },
11
+ ProxyGateError: class extends Error {
12
+ code;
13
+ constructor(msg, code) {
14
+ super(msg);
15
+ this.code = code;
16
+ }
17
+ },
18
+ }));
19
+ vi.mock('../config.js', () => ({
20
+ loadConfig: vi.fn().mockResolvedValue({
21
+ gatewayUrl: 'http://localhost:3001',
22
+ keypairPath: '/tmp/key.json',
23
+ }),
24
+ }));
25
+ const CATEGORIES_RESULT = {
26
+ categories: [
27
+ {
28
+ slug: 'llm',
29
+ name: 'Language Models',
30
+ icon: '🤖',
31
+ listing_count: 12,
32
+ subcategories: [
33
+ { slug: 'chat', name: 'Chat', icon: '💬', listing_count: 8 },
34
+ { slug: 'completion', name: 'Completion', icon: '📝', listing_count: 4 },
35
+ ],
36
+ },
37
+ ],
38
+ has_more: false,
39
+ cursor: null,
40
+ };
41
+ describe('categories command', () => {
42
+ let logSpy;
43
+ beforeEach(() => {
44
+ vi.clearAllMocks();
45
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
46
+ vi.spyOn(console, 'error').mockImplementation(() => { });
47
+ });
48
+ afterEach(() => {
49
+ vi.restoreAllMocks();
50
+ });
51
+ const run = async (...args) => {
52
+ const program = new Command('proxygate');
53
+ program
54
+ .option('--gateway <url>')
55
+ .option('--keypair <path>')
56
+ .option('--json', 'Output raw JSON');
57
+ registerCategoriesCommand(program);
58
+ await program.parseAsync(['node', 'proxygate', 'categories', ...args]);
59
+ };
60
+ it('outputs formatted table by default', async () => {
61
+ mockCategories.mockResolvedValue(CATEGORIES_RESULT);
62
+ await run();
63
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
64
+ expect(output).toContain('Categories');
65
+ expect(output).toContain('Language Models');
66
+ expect(output).toContain('12');
67
+ expect(output).toContain('Chat');
68
+ expect(output).toContain('Completion');
69
+ });
70
+ it('outputs raw JSON with --json flag', async () => {
71
+ mockCategories.mockResolvedValue(CATEGORIES_RESULT);
72
+ await run('--json');
73
+ expect(logSpy).toHaveBeenCalledTimes(1);
74
+ const parsed = JSON.parse(logSpy.mock.calls[0][0]);
75
+ expect(parsed).toEqual(CATEGORIES_RESULT);
76
+ });
77
+ it('shows empty message when no categories', async () => {
78
+ mockCategories.mockResolvedValue({ categories: [], has_more: false, cursor: null });
79
+ await run();
80
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
81
+ expect(output).toContain('No categories available');
82
+ });
83
+ });
84
+ //# sourceMappingURL=categories.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"categories.test.js","sourceRoot":"","sources":["../../src/commands/categories.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { Command } from 'commander';
2
+ export declare function registerCreateCommand(program: Command): void;
3
+ //# sourceMappingURL=create.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create.d.ts","sourceRoot":"","sources":["../../src/commands/create.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAmDzC,wBAAgB,qBAAqB,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAiH5D"}
@@ -0,0 +1,136 @@
1
+ import { readdir, readFile, writeFile, mkdir, access } from 'node:fs/promises';
2
+ import { join, resolve, dirname } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ import { bold, green, red, dim, cyan } from '../format.js';
5
+ const TEMPLATES = ['http-api', 'llm-agent'];
6
+ const TEMPLATE_DESCRIPTIONS = {
7
+ 'http-api': 'Simple REST API (Hono)',
8
+ 'llm-agent': 'LLM-powered agent with streaming (Hono + OpenAI)',
9
+ };
10
+ /** Resolve the templates directory relative to this file's location. */
11
+ function templatesDir() {
12
+ const thisFile = fileURLToPath(import.meta.url);
13
+ // In dist/commands/create.js → go up to dist/, then to templates/
14
+ // In src/commands/create.ts → go up to src/, then to ../templates/
15
+ const cliRoot = resolve(dirname(thisFile), '..', '..');
16
+ return join(cliRoot, 'templates');
17
+ }
18
+ /** Recursively copy a directory, replacing placeholders in file contents. */
19
+ async function copyTemplate(srcDir, destDir, replacements) {
20
+ const created = [];
21
+ const entries = await readdir(srcDir, { withFileTypes: true });
22
+ await mkdir(destDir, { recursive: true });
23
+ for (const entry of entries) {
24
+ const srcPath = join(srcDir, entry.name);
25
+ const destPath = join(destDir, entry.name);
26
+ if (entry.isDirectory()) {
27
+ const nested = await copyTemplate(srcPath, destPath, replacements);
28
+ created.push(...nested);
29
+ }
30
+ else {
31
+ let content = await readFile(srcPath, 'utf-8');
32
+ for (const [placeholder, value] of Object.entries(replacements)) {
33
+ content = content.replaceAll(placeholder, value);
34
+ }
35
+ await writeFile(destPath, content, 'utf-8');
36
+ created.push(destPath);
37
+ }
38
+ }
39
+ return created;
40
+ }
41
+ export function registerCreateCommand(program) {
42
+ program
43
+ .command('create')
44
+ .argument('[name]', 'Agent project name')
45
+ .option('-t, --template <template>', 'Template to use (http-api, llm-agent)')
46
+ .option('-p, --port <port>', 'Local port', '3000')
47
+ .option('--price <price>', 'Price per request in micro-cents', '5000')
48
+ .description('Scaffold a new ProxyGate agent project')
49
+ .addHelpText('after', '\nExamples:\n' +
50
+ ' $ proxygate create\n' +
51
+ ' $ proxygate create my-agent --template http-api\n' +
52
+ ' $ proxygate create my-agent -t llm-agent -p 8080 --price 10000\n\n' +
53
+ 'Templates:\n' +
54
+ TEMPLATES.map((t) => ` ${t.padEnd(12)} ${TEMPLATE_DESCRIPTIONS[t]}`).join('\n'))
55
+ .action(async (nameArg, opts) => {
56
+ try {
57
+ // -----------------------------------------------------------------
58
+ // 1. Gather inputs (interactive if flags missing)
59
+ // -----------------------------------------------------------------
60
+ let name = nameArg;
61
+ let template = opts.template;
62
+ if (!name || !template) {
63
+ const prompts = await import('@inquirer/prompts');
64
+ if (!name) {
65
+ name = await prompts.input({
66
+ message: 'Agent name:',
67
+ default: 'my-agent',
68
+ validate: (v) => /^[a-z0-9][a-z0-9-]*$/.test(v) ||
69
+ 'Use lowercase letters, numbers, and hyphens',
70
+ });
71
+ }
72
+ if (!template) {
73
+ template = (await prompts.select({
74
+ message: 'Template:',
75
+ choices: TEMPLATES.map((t) => ({
76
+ name: `${t.padEnd(12)} — ${TEMPLATE_DESCRIPTIONS[t]}`,
77
+ value: t,
78
+ })),
79
+ }));
80
+ }
81
+ }
82
+ // Validate template
83
+ if (!TEMPLATES.includes(template)) {
84
+ console.error(red(`Unknown template: "${template}". Choose: ${TEMPLATES.join(', ')}`));
85
+ process.exit(1);
86
+ }
87
+ // -----------------------------------------------------------------
88
+ // 2. Check target directory doesn't exist
89
+ // -----------------------------------------------------------------
90
+ const targetDir = resolve(name);
91
+ try {
92
+ await access(targetDir);
93
+ console.error(red(`Directory "${name}" already exists.`));
94
+ process.exit(1);
95
+ }
96
+ catch {
97
+ // Good — directory doesn't exist
98
+ }
99
+ // -----------------------------------------------------------------
100
+ // 3. Copy template with placeholder replacement
101
+ // -----------------------------------------------------------------
102
+ const srcDir = join(templatesDir(), template);
103
+ const replacements = {
104
+ '{{name}}': name,
105
+ '{{port}}': opts.port,
106
+ '{{price}}': opts.price,
107
+ };
108
+ const created = await copyTemplate(srcDir, targetDir, replacements);
109
+ // -----------------------------------------------------------------
110
+ // 4. Print result
111
+ // -----------------------------------------------------------------
112
+ console.log();
113
+ console.log(` ${green('Created')} ${bold(name)}/`);
114
+ console.log();
115
+ // Show file tree (relative to target)
116
+ for (const file of created.sort()) {
117
+ const relative = file.replace(targetDir + '/', '');
118
+ console.log(` ${dim(relative)}`);
119
+ }
120
+ console.log();
121
+ console.log(` ${bold('Next steps:')}`);
122
+ console.log(` ${cyan(`cd ${name}`)}`);
123
+ console.log(` ${cyan('npm install')}`);
124
+ console.log(` ${cyan('npm run dev')} ${dim('# start your server')}`);
125
+ console.log(` ${cyan('proxygate test')} ${dim('# validate endpoints')}`);
126
+ console.log(` ${cyan('proxygate tunnel')} ${dim('# go live on ProxyGate')}`);
127
+ console.log();
128
+ }
129
+ catch (err) {
130
+ const message = err instanceof Error ? err.message : String(err);
131
+ console.error(red(`Error: ${message}`));
132
+ process.exit(1);
133
+ }
134
+ });
135
+ }
136
+ //# sourceMappingURL=create.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create.js","sourceRoot":"","sources":["../../src/commands/create.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=create.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create.test.d.ts","sourceRoot":"","sources":["../../src/commands/create.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,95 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerCreateCommand } from './create.js';
4
+ // Mock fs operations
5
+ const mockReaddir = vi.fn();
6
+ const mockReadFile = vi.fn();
7
+ const mockWriteFile = vi.fn();
8
+ const mockMkdir = vi.fn();
9
+ const mockAccess = vi.fn();
10
+ vi.mock('node:fs/promises', () => ({
11
+ readdir: (...args) => mockReaddir(...args),
12
+ readFile: (...args) => mockReadFile(...args),
13
+ writeFile: (...args) => mockWriteFile(...args),
14
+ mkdir: (...args) => mockMkdir(...args),
15
+ access: (...args) => mockAccess(...args),
16
+ }));
17
+ describe('create command', () => {
18
+ let logSpy;
19
+ let errorSpy;
20
+ beforeEach(() => {
21
+ vi.clearAllMocks();
22
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
23
+ errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
24
+ mockMkdir.mockResolvedValue(undefined);
25
+ mockWriteFile.mockResolvedValue(undefined);
26
+ // Target dir does not exist (ENOENT)
27
+ mockAccess.mockRejectedValue(new Error('ENOENT'));
28
+ // Template files — top-level returns dirs + files, sub-dir returns files only
29
+ mockReaddir
30
+ .mockResolvedValueOnce([
31
+ { name: 'src', isDirectory: () => true, isFile: () => false },
32
+ { name: 'package.json', isDirectory: () => false, isFile: () => true },
33
+ { name: 'proxygate.tunnel.yaml', isDirectory: () => false, isFile: () => true },
34
+ { name: 'README.md', isDirectory: () => false, isFile: () => true },
35
+ { name: 'tsconfig.json', isDirectory: () => false, isFile: () => true },
36
+ ])
37
+ .mockResolvedValue([
38
+ { name: 'index.ts', isDirectory: () => false, isFile: () => true },
39
+ ]);
40
+ mockReadFile.mockResolvedValue('{{name}} at port {{port}} costs {{price}}');
41
+ });
42
+ afterEach(() => {
43
+ vi.restoreAllMocks();
44
+ });
45
+ const runCreate = async (...args) => {
46
+ const program = new Command('proxygate');
47
+ registerCreateCommand(program);
48
+ await program.parseAsync(['node', 'proxygate', 'create', ...args]);
49
+ };
50
+ it('non-interactive: creates project with all flags', async () => {
51
+ await runCreate('my-agent', '--template', 'http-api', '--port', '4000', '--price', '2000');
52
+ // Should create project directory
53
+ expect(mockMkdir).toHaveBeenCalledWith(expect.stringContaining('my-agent'), expect.objectContaining({ recursive: true }));
54
+ // Should write files with placeholders replaced
55
+ const writeCall = mockWriteFile.mock.calls.find((c) => typeof c[1] === 'string' && c[1].includes('my-agent'));
56
+ expect(writeCall).toBeDefined();
57
+ expect(writeCall[1]).toContain('my-agent');
58
+ expect(writeCall[1]).toContain('4000');
59
+ expect(writeCall[1]).toContain('2000');
60
+ expect(writeCall[1]).not.toContain('{{name}}');
61
+ });
62
+ it('exits with error if target directory already exists', async () => {
63
+ mockAccess.mockResolvedValue(undefined); // dir exists
64
+ const exitCalls = [];
65
+ const mockExit = vi.spyOn(process, 'exit').mockImplementation((code) => {
66
+ exitCalls.push(code);
67
+ return undefined;
68
+ });
69
+ await runCreate('existing-dir', '--template', 'http-api');
70
+ expect(exitCalls).toContain(1);
71
+ const errOutput = errorSpy.mock.calls.map((c) => c[0]).join('\n');
72
+ expect(errOutput).toContain('already exists');
73
+ mockExit.mockRestore();
74
+ });
75
+ it('exits with error for unknown template', async () => {
76
+ const exitCalls = [];
77
+ const mockExit = vi.spyOn(process, 'exit').mockImplementation((code) => {
78
+ exitCalls.push(code);
79
+ return undefined;
80
+ });
81
+ await runCreate('my-agent', '--template', 'nonexistent');
82
+ expect(exitCalls).toContain(1);
83
+ const errOutput = errorSpy.mock.calls.map((c) => c[0]).join('\n');
84
+ expect(errOutput).toContain('Unknown template');
85
+ mockExit.mockRestore();
86
+ });
87
+ it('shows next steps after creation', async () => {
88
+ await runCreate('my-agent', '--template', 'http-api');
89
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
90
+ expect(output).toContain('cd my-agent');
91
+ expect(output).toContain('npm install');
92
+ expect(output).toContain('proxygate tunnel');
93
+ });
94
+ });
95
+ //# sourceMappingURL=create.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create.test.js","sourceRoot":"","sources":["../../src/commands/create.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { Command } from 'commander';
2
+ /**
3
+ * Register the `proxygate deposit` command.
4
+ *
5
+ * Initiates a vault deposit by sending USDC on-chain and confirming
6
+ * with the gateway. Uses client.vault.deposit() under the hood.
7
+ */
8
+ export declare function registerDepositCommand(program: Command): void;
9
+ //# sourceMappingURL=deposit.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deposit.d.ts","sourceRoot":"","sources":["../../src/commands/deposit.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAKzC;;;;;GAKG;AACH,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA4E7D"}
@@ -0,0 +1,78 @@
1
+ import { ProxyGateError } from '@proxygate/sdk';
2
+ import { getClient } from '../helpers.js';
3
+ import { bold, green, red, dim, formatUsdc } from '../format.js';
4
+ /**
5
+ * Register the `proxygate deposit` command.
6
+ *
7
+ * Initiates a vault deposit by sending USDC on-chain and confirming
8
+ * with the gateway. Uses client.vault.deposit() under the hood.
9
+ */
10
+ export function registerDepositCommand(program) {
11
+ program
12
+ .command('deposit')
13
+ .description('Deposit USDC from your Solana wallet into your ProxyGate vault')
14
+ .requiredOption('-a, --amount <lamports>', 'Amount in USDC base units (1 USDC = 1,000,000 lamports)')
15
+ .option('--rpc <url>', 'Solana RPC URL (default: devnet)')
16
+ .addHelpText('after', '\nExamples:\n' +
17
+ ' $ proxygate deposit -a 5000000 # Deposit 5 USDC\n' +
18
+ ' $ proxygate deposit -a 1000000 # Deposit 1 USDC\n' +
19
+ ' $ proxygate deposit -a 10000000 --rpc https://api.mainnet-beta.solana.com\n\n' +
20
+ 'USDC amounts:\n' +
21
+ ' 1 USDC = 1,000,000\n' +
22
+ ' 5 USDC = 5,000,000\n' +
23
+ ' 10 USDC = 10,000,000\n\n' +
24
+ 'The vault auto-initializes on your first deposit.')
25
+ .action(async (opts) => {
26
+ const parentOpts = program.opts();
27
+ try {
28
+ const client = await getClient(parentOpts);
29
+ const amount = parseInt(opts.amount, 10);
30
+ if (isNaN(amount) || amount <= 0) {
31
+ console.error(red('Error: --amount must be a positive integer (USDC base units)'));
32
+ process.exit(1);
33
+ }
34
+ const result = await client.vault.deposit({
35
+ amount,
36
+ ...(opts.rpc ? { rpcUrl: opts.rpc } : {}),
37
+ });
38
+ if (parentOpts.json) {
39
+ console.log(JSON.stringify(result, null, 2));
40
+ return;
41
+ }
42
+ console.log(bold('Vault Deposit'));
43
+ console.log();
44
+ console.log(` ${green('TX Signature:')} ${result.tx_signature}`);
45
+ console.log(` ${green('Deposited:')} ${formatUsdc(result.deposited)}`);
46
+ console.log(` ${green('New Balance:')} ${formatUsdc(result.balance)} (vault)`);
47
+ }
48
+ catch (err) {
49
+ if (err instanceof ProxyGateError) {
50
+ console.error(red(`Error [${err.code}]: ${err.message}`));
51
+ if (err.action)
52
+ console.error(dim(`Suggestion: ${err.action}`));
53
+ if (err.code === 'vault_not_found') {
54
+ console.error();
55
+ console.error(dim('Troubleshooting:'));
56
+ console.error(dim(' 1. The on-chain transaction may not be confirmed yet — wait a few seconds and retry'));
57
+ console.error(dim(' 2. Ensure you have sufficient USDC in your wallet'));
58
+ console.error(dim(' 3. The vault auto-initializes on first deposit — no separate setup needed'));
59
+ }
60
+ if (err.code === 'deposit_not_found') {
61
+ console.error();
62
+ console.error(dim('Troubleshooting:'));
63
+ console.error(dim(' 1. The transaction may still be confirming — wait and retry'));
64
+ console.error(dim(' 2. This deposit may have already been confirmed (duplicate)'));
65
+ console.error(dim(' 3. Check the transaction on Solana Explorer'));
66
+ }
67
+ process.exit(1);
68
+ }
69
+ if (err instanceof Error) {
70
+ console.error(red(`Error: ${err.message}`));
71
+ console.error(dim('Ensure you have sufficient USDC in your wallet for the deposit.'));
72
+ process.exit(1);
73
+ }
74
+ throw err;
75
+ }
76
+ });
77
+ }
78
+ //# sourceMappingURL=deposit.js.map