@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,102 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerUsageCommand } from './usage.js';
4
+ const mockUsage = vi.fn();
5
+ vi.mock('@proxygate/sdk', () => ({
6
+ ProxyGateClient: {
7
+ create: vi.fn().mockResolvedValue({
8
+ usage: (...args) => mockUsage(...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 USAGE_RESULT = {
26
+ usage: [
27
+ {
28
+ id: 'req-1',
29
+ timestamp: '2026-02-26T12:00:00Z',
30
+ service: 'openai',
31
+ model: 'gpt-4',
32
+ status_code: 200,
33
+ latency_ms: 150,
34
+ tokens_used: 500,
35
+ cost_micro_cents: 3000,
36
+ seller_id: 'seller-abc',
37
+ },
38
+ ],
39
+ summary: [
40
+ {
41
+ service: 'openai',
42
+ total_requests: 10,
43
+ total_cost: 30000,
44
+ avg_latency: 145,
45
+ },
46
+ ],
47
+ limit: 20,
48
+ offset: 0,
49
+ };
50
+ describe('usage command', () => {
51
+ let logSpy;
52
+ beforeEach(() => {
53
+ vi.clearAllMocks();
54
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
55
+ vi.spyOn(console, 'error').mockImplementation(() => { });
56
+ });
57
+ afterEach(() => {
58
+ vi.restoreAllMocks();
59
+ });
60
+ const runUsage = async (...args) => {
61
+ const program = new Command('proxygate');
62
+ program
63
+ .option('--gateway <url>')
64
+ .option('--keypair <path>')
65
+ .option('--json', 'Output raw JSON');
66
+ registerUsageCommand(program);
67
+ await program.parseAsync(['node', 'proxygate', 'usage', ...args]);
68
+ };
69
+ it('outputs formatted summary and table by default', async () => {
70
+ mockUsage.mockResolvedValue(USAGE_RESULT);
71
+ await runUsage();
72
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
73
+ expect(output).toContain('Usage Summary');
74
+ expect(output).toContain('Recent Requests');
75
+ expect(output).toContain('openai');
76
+ expect(output).toContain('200');
77
+ });
78
+ it('outputs raw JSON with --json flag', async () => {
79
+ mockUsage.mockResolvedValue(USAGE_RESULT);
80
+ await runUsage('--json');
81
+ expect(logSpy).toHaveBeenCalledTimes(1);
82
+ const parsed = JSON.parse(logSpy.mock.calls[0][0]);
83
+ expect(parsed).toEqual(USAGE_RESULT);
84
+ });
85
+ it('passes filter options to client', async () => {
86
+ mockUsage.mockResolvedValue({ ...USAGE_RESULT, usage: [], summary: [] });
87
+ await runUsage('--service', 'anthropic', '--from', '2026-01-01', '--limit', '5');
88
+ expect(mockUsage).toHaveBeenCalledWith({
89
+ service: 'anthropic',
90
+ from: '2026-01-01',
91
+ to: undefined,
92
+ limit: 5,
93
+ });
94
+ });
95
+ it('shows empty state when no usage entries', async () => {
96
+ mockUsage.mockResolvedValue({ usage: [], summary: [], limit: 20, offset: 0 });
97
+ await runUsage();
98
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
99
+ expect(output).toContain('No usage entries found');
100
+ });
101
+ });
102
+ //# sourceMappingURL=usage.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"usage.test.js","sourceRoot":"","sources":["../../src/commands/usage.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import type { Command } from 'commander';
2
+ /**
3
+ * Register the `proxygate withdraw-confirm` command.
4
+ *
5
+ * Confirms a completed on-chain withdrawal with the gateway.
6
+ * Used for recovery when the SDK crashed after the on-chain TX
7
+ * but before the gateway was notified.
8
+ */
9
+ export declare function registerWithdrawConfirmCommand(program: Command): void;
10
+ //# sourceMappingURL=withdraw-confirm.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw-confirm.d.ts","sourceRoot":"","sources":["../../src/commands/withdraw-confirm.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAKzC;;;;;;GAMG;AACH,wBAAgB,8BAA8B,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA2CrE"}
@@ -0,0 +1,51 @@
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 withdraw-confirm` command.
6
+ *
7
+ * Confirms a completed on-chain withdrawal with the gateway.
8
+ * Used for recovery when the SDK crashed after the on-chain TX
9
+ * but before the gateway was notified.
10
+ */
11
+ export function registerWithdrawConfirmCommand(program) {
12
+ program
13
+ .command('withdraw-confirm')
14
+ .description('Confirm an on-chain withdrawal with the gateway (recovery tool)')
15
+ .requiredOption('--tx <signature>', 'Solana transaction signature to confirm')
16
+ .addHelpText('after', '\nWhen to use this:\n' +
17
+ ' If `proxygate withdraw` crashed after the on-chain TX succeeded but\n' +
18
+ ' before the gateway was notified, use this to complete the process.\n\n' +
19
+ 'Example:\n' +
20
+ ' $ proxygate withdraw-confirm --tx 5UyT3...abc')
21
+ .action(async (opts) => {
22
+ const parentOpts = program.opts();
23
+ try {
24
+ const client = await getClient(parentOpts);
25
+ const result = await client.vault.withdrawConfirm(opts.tx);
26
+ if (parentOpts.json) {
27
+ console.log(JSON.stringify(result, null, 2));
28
+ return;
29
+ }
30
+ console.log(bold('Withdrawal Confirmed'));
31
+ console.log();
32
+ console.log(` ${green('Withdrawn:')} ${formatUsdc(result.withdrawn)}`);
33
+ console.log(` ${green('Balance:')} ${formatUsdc(result.balance)}`);
34
+ console.log(` ${green('TX:')} ${result.tx_signature}`);
35
+ }
36
+ catch (err) {
37
+ if (err instanceof ProxyGateError) {
38
+ console.error(red(`Error [${err.code}]: ${err.message}`));
39
+ if (err.action)
40
+ console.error(dim(`Suggestion: ${err.action}`));
41
+ process.exit(1);
42
+ }
43
+ if (err instanceof Error) {
44
+ console.error(red(`Error: ${err.message}`));
45
+ process.exit(1);
46
+ }
47
+ throw err;
48
+ }
49
+ });
50
+ }
51
+ //# sourceMappingURL=withdraw-confirm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw-confirm.js","sourceRoot":"","sources":["../../src/commands/withdraw-confirm.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1C,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAEjE;;;;;;GAMG;AACH,MAAM,UAAU,8BAA8B,CAAC,OAAgB;IAC7D,OAAO;SACJ,OAAO,CAAC,kBAAkB,CAAC;SAC3B,WAAW,CAAC,iEAAiE,CAAC;SAC9E,cAAc,CAAC,kBAAkB,EAAE,yCAAyC,CAAC;SAC7E,WAAW,CACV,OAAO,EACP,uBAAuB;QACrB,yEAAyE;QACzE,0EAA0E;QAC1E,YAAY;QACZ,iDAAiD,CACpD;SACA,MAAM,CAAC,KAAK,EAAE,IAAoB,EAAE,EAAE;QACrC,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,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAE3D,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,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC,CAAC;YAC1C,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,CAAC,YAAY,CAAC,KAAK,UAAU,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;YACzE,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,CAAC,UAAU,CAAC,OAAO,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;YACvE,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,CAAC,KAAK,CAAC,YAAY,MAAM,CAAC,YAAY,EAAE,CAAC,CAAC;QAClE,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,IAAI,GAAG,YAAY,KAAK,EAAE,CAAC;gBACzB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBAC5C,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=withdraw-confirm.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw-confirm.test.d.ts","sourceRoot":"","sources":["../../src/commands/withdraw-confirm.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,75 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerWithdrawConfirmCommand } from './withdraw-confirm.js';
4
+ const mockWithdrawConfirm = vi.fn();
5
+ vi.mock('@proxygate/sdk', () => ({
6
+ ProxyGateClient: {
7
+ create: vi.fn().mockResolvedValue({
8
+ vault: {
9
+ withdrawConfirm: (...args) => mockWithdrawConfirm(...args),
10
+ },
11
+ }),
12
+ },
13
+ ProxyGateError: class extends Error {
14
+ code;
15
+ action;
16
+ constructor(msg, code) {
17
+ super(msg);
18
+ this.code = code;
19
+ }
20
+ },
21
+ }));
22
+ vi.mock('../config.js', () => ({
23
+ loadConfig: vi.fn().mockResolvedValue({
24
+ gatewayUrl: 'http://localhost:3001',
25
+ keypairPath: '/tmp/key.json',
26
+ }),
27
+ }));
28
+ const CONFIRM_RESULT = {
29
+ balance: 8_000_000,
30
+ withdrawn: 2_000_000,
31
+ tx_signature: '5abc123xyz789txsig',
32
+ currency: 'lamports',
33
+ };
34
+ describe('withdraw-confirm command', () => {
35
+ let logSpy;
36
+ beforeEach(() => {
37
+ vi.clearAllMocks();
38
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
39
+ vi.spyOn(console, 'error').mockImplementation(() => { });
40
+ });
41
+ afterEach(() => {
42
+ vi.restoreAllMocks();
43
+ });
44
+ const run = async (...args) => {
45
+ const program = new Command('proxygate');
46
+ program
47
+ .option('--gateway <url>')
48
+ .option('--keypair <path>')
49
+ .option('--json', 'Output raw JSON');
50
+ registerWithdrawConfirmCommand(program);
51
+ await program.parseAsync(['node', 'proxygate', 'withdraw-confirm', ...args]);
52
+ };
53
+ it('outputs formatted confirmation with TX and amounts', async () => {
54
+ mockWithdrawConfirm.mockResolvedValue(CONFIRM_RESULT);
55
+ await run('--tx', '5abc123xyz789txsig');
56
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
57
+ expect(output).toContain('Withdrawal Confirmed');
58
+ expect(output).toContain('2.000000 USDC');
59
+ expect(output).toContain('8.000000 USDC');
60
+ expect(output).toContain('5abc123xyz789txsig');
61
+ });
62
+ it('passes TX signature to client', async () => {
63
+ mockWithdrawConfirm.mockResolvedValue(CONFIRM_RESULT);
64
+ await run('--tx', 'my-tx-sig');
65
+ expect(mockWithdrawConfirm).toHaveBeenCalledWith('my-tx-sig');
66
+ });
67
+ it('outputs raw JSON with --json flag', async () => {
68
+ mockWithdrawConfirm.mockResolvedValue(CONFIRM_RESULT);
69
+ await run('--tx', '5abc123xyz789txsig', '--json');
70
+ expect(logSpy).toHaveBeenCalledTimes(1);
71
+ const parsed = JSON.parse(logSpy.mock.calls[0][0]);
72
+ expect(parsed).toEqual(CONFIRM_RESULT);
73
+ });
74
+ });
75
+ //# sourceMappingURL=withdraw-confirm.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw-confirm.test.js","sourceRoot":"","sources":["../../src/commands/withdraw-confirm.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { Command } from 'commander';
2
+ /**
3
+ * Register the `proxygate withdraw` command.
4
+ *
5
+ * Initiates a vault withdrawal with cooldown flow.
6
+ * Uses client.vault.withdraw() which handles cooldown polling internally.
7
+ */
8
+ export declare function registerWithdrawCommand(program: Command): void;
9
+ //# sourceMappingURL=withdraw.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw.d.ts","sourceRoot":"","sources":["../../src/commands/withdraw.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAKzC;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CA6E9D"}
@@ -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 withdraw` command.
6
+ *
7
+ * Initiates a vault withdrawal with cooldown flow.
8
+ * Uses client.vault.withdraw() which handles cooldown polling internally.
9
+ */
10
+ export function registerWithdrawCommand(program) {
11
+ program
12
+ .command('withdraw')
13
+ .description('Withdraw USDC from your vault back to your Solana wallet')
14
+ .option('-a, --amount <lamports>', 'Amount in USDC base units (omit to withdraw all available)')
15
+ .option('--rpc <url>', 'Solana RPC URL (default: devnet)')
16
+ .addHelpText('after', '\nExamples:\n' +
17
+ ' $ proxygate withdraw # Withdraw all available\n' +
18
+ ' $ proxygate withdraw -a 2000000 # Withdraw 2 USDC\n' +
19
+ ' $ proxygate withdraw --rpc https://api.mainnet-beta.solana.com\n\n' +
20
+ 'Withdrawals go through a cooldown period to finalize pending settlements.\n' +
21
+ 'The CLI handles polling automatically.')
22
+ .action(async (opts) => {
23
+ const parentOpts = program.opts();
24
+ try {
25
+ const client = await getClient(parentOpts);
26
+ let amount;
27
+ if (opts.amount) {
28
+ amount = parseInt(opts.amount, 10);
29
+ if (isNaN(amount) || amount <= 0) {
30
+ console.error(red('Error: --amount must be a positive integer (USDC base units)'));
31
+ process.exit(1);
32
+ }
33
+ }
34
+ const result = await client.vault.withdraw({
35
+ ...(amount !== undefined ? { amount } : {}),
36
+ ...(opts.rpc ? { rpcUrl: opts.rpc } : {}),
37
+ onProgress: (info) => {
38
+ if (!parentOpts.json) {
39
+ const secs = Math.ceil(info.remainingMs / 1000);
40
+ process.stdout.write(`\x1b[2K\r ${dim(`Cooldown: ${secs}s remaining...`)}`);
41
+ }
42
+ },
43
+ });
44
+ if (!parentOpts.json)
45
+ process.stdout.write('\x1b[2K\r'); // clear progress line
46
+ if (parentOpts.json) {
47
+ console.log(JSON.stringify(result, null, 2));
48
+ return;
49
+ }
50
+ console.log(bold('Vault Withdrawal'));
51
+ console.log();
52
+ // SDK withdraw() handles cooldown polling internally and always
53
+ // returns a 'complete' result with tx_signature and amount_withdrawn.
54
+ console.log(` ${green('Status:')} Complete`);
55
+ console.log(` ${green('TX Signature:')} ${result.tx_signature}`);
56
+ console.log(` ${green('Withdrawn:')} ${formatUsdc(result.amount_withdrawn)}`);
57
+ }
58
+ catch (err) {
59
+ if (err instanceof ProxyGateError) {
60
+ console.error(red(`Error [${err.code}]: ${err.message}`));
61
+ if (err.action)
62
+ console.error(dim(`Suggestion: ${err.action}`));
63
+ if (err.code === 'skim_flagged') {
64
+ console.error();
65
+ console.error(dim('Your wallet has been flagged for vault skim protection.\n' +
66
+ 'All unsettled entries have been forfeited. Contact support.'));
67
+ }
68
+ process.exit(1);
69
+ }
70
+ if (err instanceof Error) {
71
+ console.error(red(`Error: ${err.message}`));
72
+ process.exit(1);
73
+ }
74
+ throw err;
75
+ }
76
+ });
77
+ }
78
+ //# sourceMappingURL=withdraw.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw.js","sourceRoot":"","sources":["../../src/commands/withdraw.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=withdraw.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw.test.d.ts","sourceRoot":"","sources":["../../src/commands/withdraw.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,96 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { Command } from 'commander';
3
+ import { registerWithdrawCommand } from './withdraw.js';
4
+ const mockWithdraw = vi.fn();
5
+ vi.mock('@proxygate/sdk', () => ({
6
+ ProxyGateClient: {
7
+ create: vi.fn().mockResolvedValue({
8
+ vault: {
9
+ withdraw: (...args) => mockWithdraw(...args),
10
+ },
11
+ }),
12
+ },
13
+ ProxyGateError: class extends Error {
14
+ code;
15
+ constructor(msg, code) {
16
+ super(msg);
17
+ this.code = code;
18
+ }
19
+ },
20
+ }));
21
+ vi.mock('../config.js', () => ({
22
+ loadConfig: vi.fn().mockResolvedValue({
23
+ gatewayUrl: 'http://localhost:3001',
24
+ keypairPath: '/tmp/key.json',
25
+ }),
26
+ }));
27
+ const WITHDRAW_COMPLETE_RESULT = {
28
+ status: 'complete',
29
+ tx_signature: '5abc123xyz789txsig',
30
+ amount_withdrawn: 2_000_000,
31
+ };
32
+ describe('withdraw command', () => {
33
+ let logSpy;
34
+ let errorSpy;
35
+ beforeEach(() => {
36
+ vi.clearAllMocks();
37
+ logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
38
+ errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
39
+ });
40
+ afterEach(() => {
41
+ vi.restoreAllMocks();
42
+ });
43
+ const runWithdraw = async (...args) => {
44
+ const program = new Command('proxygate');
45
+ program
46
+ .option('--gateway <url>')
47
+ .option('--keypair <path>')
48
+ .option('--json', 'Output raw JSON');
49
+ registerWithdrawCommand(program);
50
+ await program.parseAsync(['node', 'proxygate', 'withdraw', ...args]);
51
+ };
52
+ it('outputs formatted complete result with TX signature and amount', async () => {
53
+ mockWithdraw.mockResolvedValue(WITHDRAW_COMPLETE_RESULT);
54
+ await runWithdraw();
55
+ const output = logSpy.mock.calls.map((c) => c[0]).join('\n');
56
+ expect(output).toContain('Vault Withdrawal');
57
+ expect(output).toContain('Complete');
58
+ expect(output).toContain('5abc123xyz789txsig');
59
+ expect(output).toContain('2.000000 USDC');
60
+ });
61
+ it('outputs raw JSON with --json flag', async () => {
62
+ mockWithdraw.mockResolvedValue(WITHDRAW_COMPLETE_RESULT);
63
+ await runWithdraw('--json');
64
+ expect(logSpy).toHaveBeenCalledTimes(1);
65
+ const parsed = JSON.parse(logSpy.mock.calls[0][0]);
66
+ expect(parsed).toEqual(WITHDRAW_COMPLETE_RESULT);
67
+ });
68
+ it('passes amount when provided', async () => {
69
+ mockWithdraw.mockResolvedValue(WITHDRAW_COMPLETE_RESULT);
70
+ await runWithdraw('--amount', '2000000');
71
+ expect(mockWithdraw).toHaveBeenCalledWith(expect.objectContaining({ amount: 2_000_000 }));
72
+ });
73
+ it('calls withdraw without amount when omitted (withdraw all)', async () => {
74
+ mockWithdraw.mockResolvedValue(WITHDRAW_COMPLETE_RESULT);
75
+ await runWithdraw();
76
+ expect(mockWithdraw).toHaveBeenCalledWith(expect.objectContaining({}));
77
+ });
78
+ it('passes rpc option when provided', async () => {
79
+ mockWithdraw.mockResolvedValue(WITHDRAW_COMPLETE_RESULT);
80
+ await runWithdraw('--amount', '1000000', '--rpc', 'https://my-rpc.com');
81
+ expect(mockWithdraw).toHaveBeenCalledWith(expect.objectContaining({
82
+ amount: 1_000_000,
83
+ rpcUrl: 'https://my-rpc.com',
84
+ }));
85
+ });
86
+ it('exits with error for invalid amount', async () => {
87
+ const mockExit = vi.spyOn(process, 'exit').mockImplementation(() => {
88
+ throw new Error('process.exit');
89
+ });
90
+ await expect(runWithdraw('--amount', '-100')).rejects.toThrow('process.exit');
91
+ const errOutput = errorSpy.mock.calls.map((c) => c[0]).join('\n');
92
+ expect(errOutput).toContain('positive integer');
93
+ mockExit.mockRestore();
94
+ });
95
+ });
96
+ //# sourceMappingURL=withdraw.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"withdraw.test.js","sourceRoot":"","sources":["../../src/commands/withdraw.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzE,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,uBAAuB,EAAE,MAAM,eAAe,CAAC;AAExD,MAAM,YAAY,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;AAC7B,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,GAAG,EAAE,CAAC,CAAC;IAC/B,eAAe,EAAE;QACf,MAAM,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC;YAChC,KAAK,EAAE;gBACL,QAAQ,EAAE,CAAC,GAAG,IAAe,EAAE,EAAE,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC;aACxD;SACF,CAAC;KACH;IACD,cAAc,EAAE,KAAM,SAAQ,KAAK;QACjC,IAAI,CAAS;QACb,YAAY,GAAW,EAAE,IAAY;YACnC,KAAK,CAAC,GAAG,CAAC,CAAC;YACX,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACnB,CAAC;KACF;CACF,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,GAAG,EAAE,CAAC,CAAC;IAC7B,UAAU,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC;QACpC,UAAU,EAAE,uBAAuB;QACnC,WAAW,EAAE,eAAe;KAC7B,CAAC;CACH,CAAC,CAAC,CAAC;AAEJ,MAAM,wBAAwB,GAAG;IAC/B,MAAM,EAAE,UAAmB;IAC3B,YAAY,EAAE,oBAAoB;IAClC,gBAAgB,EAAE,SAAS;CAC5B,CAAC;AAEF,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,IAAI,MAAmC,CAAC;IACxC,IAAI,QAAqC,CAAC;IAE1C,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;QACnB,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAC/D,QAAQ,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,eAAe,EAAE,CAAC;IACvB,CAAC,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,KAAK,EAAE,GAAG,IAAc,EAAiB,EAAE;QAC7D,MAAM,OAAO,GAAG,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC;QACzC,OAAO;aACJ,MAAM,CAAC,iBAAiB,CAAC;aACzB,MAAM,CAAC,kBAAkB,CAAC;aAC1B,MAAM,CAAC,QAAQ,EAAE,iBAAiB,CAAC,CAAC;QACvC,uBAAuB,CAAC,OAAO,CAAC,CAAC;QACjC,MAAM,OAAO,CAAC,UAAU,CAAC,CAAC,MAAM,EAAE,WAAW,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC;IACvE,CAAC,CAAC;IAEF,EAAE,CAAC,gEAAgE,EAAE,KAAK,IAAI,EAAE;QAC9E,YAAY,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CAAC;QACzD,MAAM,WAAW,EAAE,CAAC;QAEpB,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAY,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACxE,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAC7C,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QACrC,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;QAC/C,MAAM,CAAC,MAAM,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mCAAmC,EAAE,KAAK,IAAI,EAAE;QACjD,YAAY,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CAAC;QACzD,MAAM,WAAW,CAAC,QAAQ,CAAC,CAAC;QAE5B,MAAM,CAAC,MAAM,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QACxC,MAAM,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC,CAAC;QACtE,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,wBAAwB,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE,KAAK,IAAI,EAAE;QAC3C,YAAY,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CAAC;QACzD,MAAM,WAAW,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;QAEzC,MAAM,CAAC,YAAY,CAAC,CAAC,oBAAoB,CAAC,MAAM,CAAC,gBAAgB,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC;IAC5F,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2DAA2D,EAAE,KAAK,IAAI,EAAE;QACzE,YAAY,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CAAC;QACzD,MAAM,WAAW,EAAE,CAAC;QAEpB,MAAM,CAAC,YAAY,CAAC,CAAC,oBAAoB,CAAC,MAAM,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC,CAAC;IACzE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iCAAiC,EAAE,KAAK,IAAI,EAAE;QAC/C,YAAY,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CAAC;QACzD,MAAM,WAAW,CAAC,UAAU,EAAE,SAAS,EAAE,OAAO,EAAE,oBAAoB,CAAC,CAAC;QAExE,MAAM,CAAC,YAAY,CAAC,CAAC,oBAAoB,CAAC,MAAM,CAAC,gBAAgB,CAAC;YAChE,MAAM,EAAE,SAAS;YACjB,MAAM,EAAE,oBAAoB;SAC7B,CAAC,CAAC,CAAC;IACN,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qCAAqC,EAAE,KAAK,IAAI,EAAE;QACnD,MAAM,QAAQ,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE;YACjE,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,MAAM,MAAM,CAAC,WAAW,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAE9E,MAAM,SAAS,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAY,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC7E,MAAM,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC;QAChD,QAAQ,CAAC,WAAW,EAAE,CAAC;IACzB,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,20 @@
1
+ /** Directory where ProxyGate CLI stores config. */
2
+ export declare const CONFIG_DIR: string;
3
+ /** Full path to the CLI config file. */
4
+ export declare const CONFIG_PATH: string;
5
+ /** CLI configuration stored in ~/.proxygate/config.json. */
6
+ export interface CliConfig {
7
+ gatewayUrl: string;
8
+ keypairPath: string;
9
+ }
10
+ /**
11
+ * Load CLI config from disk.
12
+ * Returns null if the config file does not exist or cannot be parsed.
13
+ */
14
+ export declare function loadConfig(): Promise<CliConfig | null>;
15
+ /**
16
+ * Save CLI config to disk.
17
+ * Creates ~/.proxygate/ directory if it does not exist.
18
+ */
19
+ export declare function saveConfig(config: CliConfig): Promise<void>;
20
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAIA,mDAAmD;AACnD,eAAO,MAAM,UAAU,EAAE,MAAsC,CAAC;AAEhE,wCAAwC;AACxC,eAAO,MAAM,WAAW,EAAE,MAAwC,CAAC;AAEnE,4DAA4D;AAC5D,MAAM,WAAW,SAAS;IACxB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED;;;GAGG;AACH,wBAAsB,UAAU,IAAI,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC,CAoB5D;AAED;;;GAGG;AACH,wBAAsB,UAAU,CAAC,MAAM,EAAE,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC,CAGjE"}
package/dist/config.js ADDED
@@ -0,0 +1,38 @@
1
+ import { readFile, writeFile, mkdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { homedir } from 'node:os';
4
+ /** Directory where ProxyGate CLI stores config. */
5
+ export const CONFIG_DIR = join(homedir(), '.proxygate');
6
+ /** Full path to the CLI config file. */
7
+ export const CONFIG_PATH = join(CONFIG_DIR, 'config.json');
8
+ /**
9
+ * Load CLI config from disk.
10
+ * Returns null if the config file does not exist or cannot be parsed.
11
+ */
12
+ export async function loadConfig() {
13
+ try {
14
+ const raw = await readFile(CONFIG_PATH, 'utf-8');
15
+ const parsed = JSON.parse(raw);
16
+ if (typeof parsed === 'object' &&
17
+ parsed !== null &&
18
+ 'gatewayUrl' in parsed &&
19
+ 'keypairPath' in parsed &&
20
+ typeof parsed.gatewayUrl === 'string' &&
21
+ typeof parsed.keypairPath === 'string') {
22
+ return parsed;
23
+ }
24
+ return null;
25
+ }
26
+ catch {
27
+ return null;
28
+ }
29
+ }
30
+ /**
31
+ * Save CLI config to disk.
32
+ * Creates ~/.proxygate/ directory if it does not exist.
33
+ */
34
+ export async function saveConfig(config) {
35
+ await mkdir(CONFIG_DIR, { recursive: true });
36
+ await writeFile(CONFIG_PATH, JSON.stringify(config, null, 2) + '\n', 'utf-8');
37
+ }
38
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AAC9D,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAElC,mDAAmD;AACnD,MAAM,CAAC,MAAM,UAAU,GAAW,IAAI,CAAC,OAAO,EAAE,EAAE,YAAY,CAAC,CAAC;AAEhE,wCAAwC;AACxC,MAAM,CAAC,MAAM,WAAW,GAAW,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;AAQnE;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,UAAU;IAC9B,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;QACjD,MAAM,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAExC,IACE,OAAO,MAAM,KAAK,QAAQ;YAC1B,MAAM,KAAK,IAAI;YACf,YAAY,IAAI,MAAM;YACtB,aAAa,IAAI,MAAM;YACvB,OAAQ,MAAoB,CAAC,UAAU,KAAK,QAAQ;YACpD,OAAQ,MAAoB,CAAC,WAAW,KAAK,QAAQ,EACrD,CAAC;YACD,OAAO,MAAmB,CAAC;QAC7B,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,MAAiB;IAChD,MAAM,KAAK,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC7C,MAAM,SAAS,CAAC,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC;AAChF,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=config.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.test.d.ts","sourceRoot":"","sources":["../src/config.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,54 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest';
2
+ vi.mock('node:fs/promises', () => ({
3
+ readFile: vi.fn(),
4
+ writeFile: vi.fn(),
5
+ mkdir: vi.fn(),
6
+ }));
7
+ import { readFile, writeFile, mkdir } from 'node:fs/promises';
8
+ import { loadConfig, saveConfig, CONFIG_DIR, CONFIG_PATH } from './config.js';
9
+ const mockReadFile = vi.mocked(readFile);
10
+ const mockWriteFile = vi.mocked(writeFile);
11
+ const mockMkdir = vi.mocked(mkdir);
12
+ beforeEach(() => {
13
+ vi.clearAllMocks();
14
+ });
15
+ describe('loadConfig', () => {
16
+ it('returns null when file does not exist (ENOENT)', async () => {
17
+ mockReadFile.mockRejectedValue(Object.assign(new Error('ENOENT'), { code: 'ENOENT' }));
18
+ expect(await loadConfig()).toBeNull();
19
+ });
20
+ it('returns null on malformed JSON', async () => {
21
+ mockReadFile.mockResolvedValue('not valid json{{{');
22
+ expect(await loadConfig()).toBeNull();
23
+ });
24
+ it('returns null when required fields are missing', async () => {
25
+ mockReadFile.mockResolvedValue(JSON.stringify({ gatewayUrl: 'http://localhost' }));
26
+ expect(await loadConfig()).toBeNull();
27
+ });
28
+ it('returns valid CliConfig when file is correct', async () => {
29
+ const config = { gatewayUrl: 'http://localhost:3001', keypairPath: '/home/user/.config/solana/id.json' };
30
+ mockReadFile.mockResolvedValue(JSON.stringify(config));
31
+ const result = await loadConfig();
32
+ expect(result).toEqual(config);
33
+ });
34
+ });
35
+ describe('saveConfig', () => {
36
+ it('creates directory and writes JSON', async () => {
37
+ mockMkdir.mockResolvedValue(undefined);
38
+ mockWriteFile.mockResolvedValue(undefined);
39
+ const config = { gatewayUrl: 'http://localhost:3001', keypairPath: '/keys/id.json' };
40
+ await saveConfig(config);
41
+ expect(mockMkdir).toHaveBeenCalledWith(CONFIG_DIR, { recursive: true });
42
+ expect(mockWriteFile).toHaveBeenCalledWith(CONFIG_PATH, JSON.stringify(config, null, 2) + '\n', 'utf-8');
43
+ });
44
+ it('writes expected format (2-space indent + trailing newline)', async () => {
45
+ mockMkdir.mockResolvedValue(undefined);
46
+ mockWriteFile.mockResolvedValue(undefined);
47
+ const config = { gatewayUrl: 'https://gw.proxygate.io', keypairPath: '~/.config/solana/id.json' };
48
+ await saveConfig(config);
49
+ const written = mockWriteFile.mock.calls[0][1];
50
+ expect(written).toContain(' "gatewayUrl"');
51
+ expect(written.endsWith('\n')).toBe(true);
52
+ });
53
+ });
54
+ //# sourceMappingURL=config.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.test.js","sourceRoot":"","sources":["../src/config.test.ts"],"names":[],"mappings":"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"}