@solana-compass/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 (120) hide show
  1. package/README.md +327 -0
  2. package/bin/sol.mjs +6 -0
  3. package/dist/commands/config.d.ts +2 -0
  4. package/dist/commands/config.js +73 -0
  5. package/dist/commands/config.js.map +1 -0
  6. package/dist/commands/lend.d.ts +2 -0
  7. package/dist/commands/lend.js +262 -0
  8. package/dist/commands/lend.js.map +1 -0
  9. package/dist/commands/lp.d.ts +2 -0
  10. package/dist/commands/lp.js +158 -0
  11. package/dist/commands/lp.js.map +1 -0
  12. package/dist/commands/network.d.ts +2 -0
  13. package/dist/commands/network.js +126 -0
  14. package/dist/commands/network.js.map +1 -0
  15. package/dist/commands/portfolio.d.ts +2 -0
  16. package/dist/commands/portfolio.js +187 -0
  17. package/dist/commands/portfolio.js.map +1 -0
  18. package/dist/commands/stake.d.ts +2 -0
  19. package/dist/commands/stake.js +143 -0
  20. package/dist/commands/stake.js.map +1 -0
  21. package/dist/commands/token.d.ts +2 -0
  22. package/dist/commands/token.js +551 -0
  23. package/dist/commands/token.js.map +1 -0
  24. package/dist/commands/tx.d.ts +2 -0
  25. package/dist/commands/tx.js +131 -0
  26. package/dist/commands/tx.js.map +1 -0
  27. package/dist/commands/wallet.d.ts +2 -0
  28. package/dist/commands/wallet.js +403 -0
  29. package/dist/commands/wallet.js.map +1 -0
  30. package/dist/core/config-manager.d.ts +31 -0
  31. package/dist/core/config-manager.js +79 -0
  32. package/dist/core/config-manager.js.map +1 -0
  33. package/dist/core/kamino-compat.d.ts +17 -0
  34. package/dist/core/kamino-compat.js +38 -0
  35. package/dist/core/kamino-compat.js.map +1 -0
  36. package/dist/core/lend-service.d.ts +41 -0
  37. package/dist/core/lend-service.js +331 -0
  38. package/dist/core/lend-service.js.map +1 -0
  39. package/dist/core/lp-service.d.ts +30 -0
  40. package/dist/core/lp-service.js +21 -0
  41. package/dist/core/lp-service.js.map +1 -0
  42. package/dist/core/onramp-service.d.ts +15 -0
  43. package/dist/core/onramp-service.js +57 -0
  44. package/dist/core/onramp-service.js.map +1 -0
  45. package/dist/core/portfolio-service.d.ts +55 -0
  46. package/dist/core/portfolio-service.js +272 -0
  47. package/dist/core/portfolio-service.js.map +1 -0
  48. package/dist/core/price-service.d.ts +8 -0
  49. package/dist/core/price-service.js +116 -0
  50. package/dist/core/price-service.js.map +1 -0
  51. package/dist/core/rpc.d.ts +5 -0
  52. package/dist/core/rpc.js +69 -0
  53. package/dist/core/rpc.js.map +1 -0
  54. package/dist/core/stake-service.d.ts +42 -0
  55. package/dist/core/stake-service.js +319 -0
  56. package/dist/core/stake-service.js.map +1 -0
  57. package/dist/core/swap-service.d.ts +31 -0
  58. package/dist/core/swap-service.js +142 -0
  59. package/dist/core/swap-service.js.map +1 -0
  60. package/dist/core/token-registry.d.ts +23 -0
  61. package/dist/core/token-registry.js +174 -0
  62. package/dist/core/token-registry.js.map +1 -0
  63. package/dist/core/token-service.d.ts +20 -0
  64. package/dist/core/token-service.js +92 -0
  65. package/dist/core/token-service.js.map +1 -0
  66. package/dist/core/transaction.d.ts +55 -0
  67. package/dist/core/transaction.js +196 -0
  68. package/dist/core/transaction.js.map +1 -0
  69. package/dist/core/wallet-manager.d.ts +20 -0
  70. package/dist/core/wallet-manager.js +142 -0
  71. package/dist/core/wallet-manager.js.map +1 -0
  72. package/dist/db/database.d.ts +3 -0
  73. package/dist/db/database.js +44 -0
  74. package/dist/db/database.js.map +1 -0
  75. package/dist/db/migrations/001_initial.d.ts +1 -0
  76. package/dist/db/migrations/001_initial.js +116 -0
  77. package/dist/db/migrations/001_initial.js.map +1 -0
  78. package/dist/db/migrations/002_tx_prices.d.ts +1 -0
  79. package/dist/db/migrations/002_tx_prices.js +5 -0
  80. package/dist/db/migrations/002_tx_prices.js.map +1 -0
  81. package/dist/db/repos/price-repo.d.ts +11 -0
  82. package/dist/db/repos/price-repo.js +14 -0
  83. package/dist/db/repos/price-repo.js.map +1 -0
  84. package/dist/db/repos/snapshot-repo.d.ts +27 -0
  85. package/dist/db/repos/snapshot-repo.js +32 -0
  86. package/dist/db/repos/snapshot-repo.js.map +1 -0
  87. package/dist/db/repos/token-repo.d.ts +17 -0
  88. package/dist/db/repos/token-repo.js +53 -0
  89. package/dist/db/repos/token-repo.js.map +1 -0
  90. package/dist/db/repos/transaction-repo.d.ts +22 -0
  91. package/dist/db/repos/transaction-repo.js +28 -0
  92. package/dist/db/repos/transaction-repo.js.map +1 -0
  93. package/dist/db/repos/wallet-repo.d.ts +18 -0
  94. package/dist/db/repos/wallet-repo.js +44 -0
  95. package/dist/db/repos/wallet-repo.js.map +1 -0
  96. package/dist/index.d.ts +1 -0
  97. package/dist/index.js +84 -0
  98. package/dist/index.js.map +1 -0
  99. package/dist/output/formatter.d.ts +27 -0
  100. package/dist/output/formatter.js +76 -0
  101. package/dist/output/formatter.js.map +1 -0
  102. package/dist/output/portfolio-renderer.d.ts +3 -0
  103. package/dist/output/portfolio-renderer.js +205 -0
  104. package/dist/output/portfolio-renderer.js.map +1 -0
  105. package/dist/output/table.d.ts +8 -0
  106. package/dist/output/table.js +22 -0
  107. package/dist/output/table.js.map +1 -0
  108. package/dist/utils/fs.d.ts +7 -0
  109. package/dist/utils/fs.js +39 -0
  110. package/dist/utils/fs.js.map +1 -0
  111. package/dist/utils/retry.d.ts +17 -0
  112. package/dist/utils/retry.js +55 -0
  113. package/dist/utils/retry.js.map +1 -0
  114. package/dist/utils/solana.d.ts +9 -0
  115. package/dist/utils/solana.js +26 -0
  116. package/dist/utils/solana.js.map +1 -0
  117. package/dist/utils/token-list.d.ts +9 -0
  118. package/dist/utils/token-list.js +31 -0
  119. package/dist/utils/token-list.js.map +1 -0
  120. package/package.json +44 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"wallet-repo.js","sourceRoot":"","sources":["../../../src/db/repos/wallet-repo.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AAUvC,MAAM,UAAU,YAAY,CAAC,IAAY,EAAE,OAAe,EAAE,QAAgB;IAC1E,KAAK,EAAE,CAAC,OAAO,CACb,iEAAiE,CAClE,CAAC,GAAG,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;AACjC,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,IAAY;IACpC,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC,sCAAsC,CAAC,CAAC,GAAG,CAAC,IAAI,CAA0B,CAAC;AACpG,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,OAAe;IAChD,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC,yCAAyC,CAAC,CAAC,GAAG,CAAC,OAAO,CAA0B,CAAC;AAC1G,CAAC;AAED,MAAM,UAAU,WAAW;IACzB,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC,2CAA2C,CAAC,CAAC,GAAG,EAAiB,CAAC;AAC3F,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,KAAa;IAC9C,OAAO,KAAK,EAAE,CAAC,OAAO,CAAC;;;;;GAKtB,CAAC,CAAC,GAAG,CAAC,KAAK,CAAgB,CAAC;AAC/B,CAAC;AAED,MAAM,UAAU,YAAY,CAAC,IAAY;IACvC,MAAM,MAAM,GAAG,KAAK,EAAE,CAAC,OAAO,CAAC,oCAAoC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC/E,OAAO,MAAM,CAAC,OAAO,GAAG,CAAC,CAAC;AAC5B,CAAC;AAED,MAAM,UAAU,QAAQ,CAAC,UAAkB,EAAE,KAAa;IACxD,KAAK,EAAE,CAAC,OAAO,CACb,wEAAwE,CACzE,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;AAC3B,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,UAAkB,EAAE,KAAa;IAC3D,KAAK,EAAE,CAAC,OAAO,CACb,+DAA+D,CAChE,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;AAC3B,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,UAAkB;IAC1C,MAAM,IAAI,GAAG,KAAK,EAAE,CAAC,OAAO,CAC1B,sEAAsE,CACvE,CAAC,GAAG,CAAC,UAAU,CAAwB,CAAC;IACzC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AAChC,CAAC;AAED,MAAM,UAAU,WAAW;IACzB,MAAM,GAAG,GAAG,KAAK,EAAE,CAAC,OAAO,CAAC,uCAAuC,CAAC,CAAC,GAAG,EAAuB,CAAC;IAChG,OAAO,GAAG,CAAC,KAAK,CAAC;AACnB,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,MAAM,GAAG,GAAG,KAAK,EAAE,CAAC,OAAO,CAAC,sDAAsD,CAAC,CAAC,GAAG,EAAkC,CAAC;IAC1H,OAAO,GAAG,EAAE,IAAI,CAAC;AACnB,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
package/dist/index.js ADDED
@@ -0,0 +1,84 @@
1
+ import { Command } from 'commander';
2
+ import { setJsonMode, setVerboseMode } from './output/formatter.js';
3
+ import { setRpcOverride } from './core/rpc.js';
4
+ import { registerConfigCommand } from './commands/config.js';
5
+ import { registerWalletCommand } from './commands/wallet.js';
6
+ import { registerTokenCommand } from './commands/token.js';
7
+ import { registerStakeCommand } from './commands/stake.js';
8
+ import { registerLendCommand } from './commands/lend.js';
9
+ import { registerPortfolioCommand } from './commands/portfolio.js';
10
+ import { registerTxCommand } from './commands/tx.js';
11
+ import { registerNetworkCommand } from './commands/network.js';
12
+ import { closeDb } from './db/database.js';
13
+ const program = new Command();
14
+ program
15
+ .name('sol')
16
+ .description('Solana CLI for Humans and LLM Agents')
17
+ .version('0.1.0')
18
+ .option('--json', 'Output structured JSON')
19
+ .option('--rpc <url>', 'Override RPC endpoint')
20
+ .option('--verbose', 'Verbose output')
21
+ .hook('preAction', (thisCommand) => {
22
+ const opts = thisCommand.optsWithGlobals();
23
+ if (opts.json)
24
+ setJsonMode(true);
25
+ if (opts.verbose)
26
+ setVerboseMode(true);
27
+ if (opts.rpc)
28
+ setRpcOverride(opts.rpc);
29
+ });
30
+ // Register all command groups
31
+ registerConfigCommand(program);
32
+ registerWalletCommand(program);
33
+ registerTokenCommand(program);
34
+ registerStakeCommand(program);
35
+ registerLendCommand(program);
36
+ registerPortfolioCommand(program);
37
+ registerTxCommand(program);
38
+ registerNetworkCommand(program);
39
+ // Custom help: show subcommands grouped under each parent
40
+ program.configureHelp({
41
+ formatHelp(cmd, helper) {
42
+ const lines = [];
43
+ lines.push(`${helper.commandDescription(cmd)}\n`);
44
+ lines.push(`Usage: ${helper.commandUsage(cmd)}\n`);
45
+ // Global options
46
+ const opts = helper.visibleOptions(cmd);
47
+ if (opts.length) {
48
+ lines.push('Options:');
49
+ const optWidth = Math.max(...opts.map(o => helper.optionTerm(o).length));
50
+ for (const opt of opts) {
51
+ const term = helper.optionTerm(opt).padEnd(optWidth + 2);
52
+ lines.push(` ${term} ${helper.optionDescription(opt)}`);
53
+ }
54
+ lines.push('');
55
+ }
56
+ // Commands with subcommands
57
+ lines.push('Commands:');
58
+ for (const group of helper.visibleCommands(cmd)) {
59
+ const children = group.commands;
60
+ if (children.length === 0) {
61
+ // Leaf command (e.g. portfolio, tx, network)
62
+ const usage = group.usage().replace('[options] ', '').replace('[options]', '').trim();
63
+ const term = usage ? `${group.name()} ${usage}` : group.name();
64
+ lines.push(` ${term.padEnd(38)} ${group.description()}`);
65
+ }
66
+ else {
67
+ lines.push(` ${group.name()}`);
68
+ for (const child of children) {
69
+ const usage = child.usage().replace('[options] ', '').replace('[options]', '').trim();
70
+ const term = usage ? `${child.name()} ${usage}` : child.name();
71
+ lines.push(` ${term.padEnd(36)} ${child.description()}`);
72
+ }
73
+ }
74
+ }
75
+ return lines.join('\n') + '\n';
76
+ },
77
+ });
78
+ // Graceful cleanup
79
+ process.on('exit', () => { closeDb(); });
80
+ program.parseAsync(process.argv).catch((err) => {
81
+ console.error(`Fatal: ${err.message}`);
82
+ process.exitCode = 1;
83
+ });
84
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACpE,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAC7D,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAC7D,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;AACzD,OAAO,EAAE,wBAAwB,EAAE,MAAM,yBAAyB,CAAC;AACnE,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAE,sBAAsB,EAAE,MAAM,uBAAuB,CAAC;AAC/D,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAE3C,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,KAAK,CAAC;KACX,WAAW,CAAC,sCAAsC,CAAC;KACnD,OAAO,CAAC,OAAO,CAAC;KAChB,MAAM,CAAC,QAAQ,EAAE,wBAAwB,CAAC;KAC1C,MAAM,CAAC,aAAa,EAAE,uBAAuB,CAAC;KAC9C,MAAM,CAAC,WAAW,EAAE,gBAAgB,CAAC;KACrC,IAAI,CAAC,WAAW,EAAE,CAAC,WAAW,EAAE,EAAE;IACjC,MAAM,IAAI,GAAG,WAAW,CAAC,eAAe,EAAE,CAAC;IAC3C,IAAI,IAAI,CAAC,IAAI;QAAE,WAAW,CAAC,IAAI,CAAC,CAAC;IACjC,IAAI,IAAI,CAAC,OAAO;QAAE,cAAc,CAAC,IAAI,CAAC,CAAC;IACvC,IAAI,IAAI,CAAC,GAAG;QAAE,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AACzC,CAAC,CAAC,CAAC;AAEL,8BAA8B;AAC9B,qBAAqB,CAAC,OAAO,CAAC,CAAC;AAC/B,qBAAqB,CAAC,OAAO,CAAC,CAAC;AAC/B,oBAAoB,CAAC,OAAO,CAAC,CAAC;AAC9B,oBAAoB,CAAC,OAAO,CAAC,CAAC;AAC9B,mBAAmB,CAAC,OAAO,CAAC,CAAC;AAC7B,wBAAwB,CAAC,OAAO,CAAC,CAAC;AAClC,iBAAiB,CAAC,OAAO,CAAC,CAAC;AAC3B,sBAAsB,CAAC,OAAO,CAAC,CAAC;AAEhC,0DAA0D;AAC1D,OAAO,CAAC,aAAa,CAAC;IACpB,UAAU,CAAC,GAAG,EAAE,MAAM;QACpB,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAClD,KAAK,CAAC,IAAI,CAAC,UAAU,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEnD,iBAAiB;QACjB,MAAM,IAAI,GAAG,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;QACxC,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YACvB,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;YACzE,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;gBACvB,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC;gBACzD,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,KAAK,MAAM,CAAC,iBAAiB,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC5D,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,4BAA4B;QAC5B,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACxB,KAAK,MAAM,KAAK,IAAI,MAAM,CAAC,eAAe,CAAC,GAAG,CAAC,EAAE,CAAC;YAChD,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAC;YAChC,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC1B,6CAA6C;gBAC7C,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,EAAE,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;gBACtF,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,IAAI,EAAE,IAAI,KAAK,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;gBAC/D,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;YAC7D,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,IAAI,CAAC,KAAK,KAAK,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;gBAChC,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE,CAAC;oBAC7B,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,EAAE,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;oBACtF,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,IAAI,EAAE,IAAI,KAAK,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;oBAC/D,KAAK,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;gBAC/D,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC;IACjC,CAAC;CACF,CAAC,CAAC;AAEH,mBAAmB;AACnB,OAAO,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;AAEzC,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;IAC7C,OAAO,CAAC,KAAK,CAAC,UAAU,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;IACvC,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;AACvB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,27 @@
1
+ export interface CommandResult<T = unknown> {
2
+ ok: boolean;
3
+ data?: T;
4
+ error?: string;
5
+ message?: string;
6
+ meta?: {
7
+ elapsed_ms: number;
8
+ [key: string]: unknown;
9
+ };
10
+ }
11
+ export declare function success<T>(data: T, meta?: Record<string, unknown>): CommandResult<T>;
12
+ export declare function failure(error: string, message: string, data?: unknown): CommandResult;
13
+ export declare function setJsonMode(enabled: boolean): void;
14
+ export declare function setVerboseMode(enabled: boolean): void;
15
+ export declare function isJsonMode(): boolean;
16
+ export declare function isVerboseMode(): boolean;
17
+ export declare function output<T>(result: CommandResult<T>): void;
18
+ export declare function outputRaw(text: string): void;
19
+ export declare function warn(message: string): void;
20
+ export declare function verbose(message: string): void;
21
+ /** Format a USD price with enough decimals to show meaningful digits.
22
+ * $80.60, $0.0061, $0.00000012 — never shows $0.0000 for non-zero prices. */
23
+ export declare function fmtPrice(price: number): string;
24
+ export declare function timed<T>(fn: () => T | Promise<T>): Promise<{
25
+ result: T;
26
+ elapsed_ms: number;
27
+ }>;
@@ -0,0 +1,76 @@
1
+ export function success(data, meta) {
2
+ return { ok: true, data, meta: { elapsed_ms: 0, ...meta } };
3
+ }
4
+ export function failure(error, message, data) {
5
+ return { ok: false, error, message, ...(data ? { data } : {}) };
6
+ }
7
+ let jsonMode = false;
8
+ let verboseMode = false;
9
+ export function setJsonMode(enabled) { jsonMode = enabled; }
10
+ export function setVerboseMode(enabled) { verboseMode = enabled; }
11
+ export function isJsonMode() { return jsonMode; }
12
+ export function isVerboseMode() { return verboseMode; }
13
+ export function output(result) {
14
+ if (jsonMode) {
15
+ console.log(JSON.stringify(result, null, 2));
16
+ }
17
+ else if (result.ok) {
18
+ // Human-readable output — handled by individual commands
19
+ // This is a fallback for commands that don't format their own output
20
+ if (result.data !== undefined) {
21
+ if (typeof result.data === 'string') {
22
+ console.log(result.data);
23
+ }
24
+ else {
25
+ console.log(JSON.stringify(result.data, null, 2));
26
+ }
27
+ }
28
+ }
29
+ else {
30
+ console.error(`Error: ${result.message || result.error}`);
31
+ }
32
+ }
33
+ export function outputRaw(text) {
34
+ if (!jsonMode) {
35
+ console.log(text);
36
+ }
37
+ }
38
+ export function warn(message) {
39
+ if (!jsonMode) {
40
+ console.error(`Warning: ${message}`);
41
+ }
42
+ }
43
+ export function verbose(message) {
44
+ if (verboseMode && !jsonMode) {
45
+ console.error(`[verbose] ${message}`);
46
+ }
47
+ }
48
+ /** Format a USD price with enough decimals to show meaningful digits.
49
+ * $80.60, $0.0061, $0.00000012 — never shows $0.0000 for non-zero prices. */
50
+ export function fmtPrice(price) {
51
+ if (price >= 1)
52
+ return price.toFixed(2);
53
+ if (price >= 0.01)
54
+ return price.toFixed(4);
55
+ // For tiny prices, show 2 significant digits
56
+ if (price > 0) {
57
+ const digits = -Math.floor(Math.log10(price)) + 1;
58
+ return price.toFixed(Math.min(digits, 12));
59
+ }
60
+ return '0.00';
61
+ }
62
+ export function timed(fn) {
63
+ const start = performance.now();
64
+ const maybePromise = fn();
65
+ if (maybePromise instanceof Promise) {
66
+ return maybePromise.then(result => ({
67
+ result,
68
+ elapsed_ms: Math.round(performance.now() - start),
69
+ }));
70
+ }
71
+ return Promise.resolve({
72
+ result: maybePromise,
73
+ elapsed_ms: Math.round(performance.now() - start),
74
+ });
75
+ }
76
+ //# sourceMappingURL=formatter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"formatter.js","sourceRoot":"","sources":["../../src/output/formatter.ts"],"names":[],"mappings":"AAQA,MAAM,UAAU,OAAO,CAAI,IAAO,EAAE,IAA8B;IAChE,OAAO,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,UAAU,EAAE,CAAC,EAAE,GAAG,IAAI,EAAE,EAAE,CAAC;AAC9D,CAAC;AAED,MAAM,UAAU,OAAO,CAAC,KAAa,EAAE,OAAe,EAAE,IAAc;IACpE,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC;AAClE,CAAC;AAED,IAAI,QAAQ,GAAG,KAAK,CAAC;AACrB,IAAI,WAAW,GAAG,KAAK,CAAC;AAExB,MAAM,UAAU,WAAW,CAAC,OAAgB,IAAI,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC;AACrE,MAAM,UAAU,cAAc,CAAC,OAAgB,IAAI,WAAW,GAAG,OAAO,CAAC,CAAC,CAAC;AAC3E,MAAM,UAAU,UAAU,KAAK,OAAO,QAAQ,CAAC,CAAC,CAAC;AACjD,MAAM,UAAU,aAAa,KAAK,OAAO,WAAW,CAAC,CAAC,CAAC;AAEvD,MAAM,UAAU,MAAM,CAAI,MAAwB;IAChD,IAAI,QAAQ,EAAE,CAAC;QACb,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAC/C,CAAC;SAAM,IAAI,MAAM,CAAC,EAAE,EAAE,CAAC;QACrB,yDAAyD;QACzD,qEAAqE;QACrE,IAAI,MAAM,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC9B,IAAI,OAAO,MAAM,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;gBACpC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC3B,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YACpD,CAAC;QACH,CAAC;IACH,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,KAAK,CAAC,UAAU,MAAM,CAAC,OAAO,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;IAC5D,CAAC;AACH,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,IAAY;IACpC,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACpB,CAAC;AACH,CAAC;AAED,MAAM,UAAU,IAAI,CAAC,OAAe;IAClC,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO,CAAC,KAAK,CAAC,YAAY,OAAO,EAAE,CAAC,CAAC;IACvC,CAAC;AACH,CAAC;AAED,MAAM,UAAU,OAAO,CAAC,OAAe;IACrC,IAAI,WAAW,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC7B,OAAO,CAAC,KAAK,CAAC,aAAa,OAAO,EAAE,CAAC,CAAC;IACxC,CAAC;AACH,CAAC;AAED;8EAC8E;AAC9E,MAAM,UAAU,QAAQ,CAAC,KAAa;IACpC,IAAI,KAAK,IAAI,CAAC;QAAE,OAAO,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IACxC,IAAI,KAAK,IAAI,IAAI;QAAE,OAAO,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IAC3C,6CAA6C;IAC7C,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;QACd,MAAM,MAAM,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC;QAClD,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC;IAC7C,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,KAAK,CAAI,EAAwB;IAC/C,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;IAChC,MAAM,YAAY,GAAG,EAAE,EAAE,CAAC;IAC1B,IAAI,YAAY,YAAY,OAAO,EAAE,CAAC;QACpC,OAAO,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YAClC,MAAM;YACN,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;SAClD,CAAC,CAAC,CAAC;IACN,CAAC;IACD,OAAO,OAAO,CAAC,OAAO,CAAC;QACrB,MAAM,EAAE,YAAY;QACpB,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;KAClD,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { PortfolioReport, CompareResult } from '../core/portfolio-service.js';
2
+ export declare function renderPortfolio(report: PortfolioReport): string;
3
+ export declare function renderCompare(result: CompareResult, label: string): string;
@@ -0,0 +1,205 @@
1
+ import { table } from './table.js';
2
+ const BAR_WIDTH = 30;
3
+ export function renderPortfolio(report) {
4
+ const lines = [];
5
+ // Header
6
+ const walletCount = report.wallets.length;
7
+ const walletLabel = walletCount === 1 ? '1 wallet' : `${walletCount} wallets`;
8
+ lines.push(`Portfolio — ${walletLabel} — $${fmt(report.totalValueUsd)}`);
9
+ lines.push('');
10
+ // Tokens section — group by symbol across wallets
11
+ const tokenPositions = report.positions.filter(p => p.type === 'token' && p.amount > 0);
12
+ if (tokenPositions.length > 0) {
13
+ const grouped = new Map();
14
+ for (const p of tokenPositions) {
15
+ const existing = grouped.get(p.symbol);
16
+ if (existing) {
17
+ existing.amount += p.amount;
18
+ if (p.valueUsd != null)
19
+ existing.valueUsd = (existing.valueUsd ?? 0) + p.valueUsd;
20
+ }
21
+ else {
22
+ grouped.set(p.symbol, { symbol: p.symbol, amount: p.amount, valueUsd: p.valueUsd });
23
+ }
24
+ }
25
+ const rows = [...grouped.values()].sort((a, b) => (b.valueUsd ?? 0) - (a.valueUsd ?? 0));
26
+ lines.push('Tokens');
27
+ lines.push(table(rows.map(r => ({
28
+ token: r.symbol,
29
+ balance: fmtAmount(r.amount),
30
+ value: r.valueUsd != null ? `$${fmt(r.valueUsd)}` : '—',
31
+ pct: r.valueUsd != null && report.totalValueUsd > 0
32
+ ? `${((r.valueUsd / report.totalValueUsd) * 100).toFixed(1)}%`
33
+ : '',
34
+ })), [
35
+ { key: 'token', header: 'Token' },
36
+ { key: 'balance', header: 'Balance', align: 'right' },
37
+ { key: 'value', header: 'Value', align: 'right' },
38
+ { key: 'pct', header: '%', align: 'right' },
39
+ ]));
40
+ lines.push('');
41
+ }
42
+ // Staking section
43
+ const stakePositions = report.positions.filter(p => p.type === 'stake');
44
+ if (stakePositions.length > 0) {
45
+ lines.push('Staking');
46
+ lines.push(table(stakePositions.map(p => {
47
+ const addr = String(p.extra?.stakeAccount ?? '');
48
+ const short = addr.length > 12 ? `${addr.slice(0, 4)}...${addr.slice(-6)}` : addr;
49
+ const validator = String(p.extra?.validator ?? '—');
50
+ const validatorShort = validator.length > 12 ? `${validator.slice(0, 7)}...${validator.slice(-5)}` : validator;
51
+ const excess = p.extra?.claimableExcess ?? 0;
52
+ return {
53
+ account: short,
54
+ staked: `${fmtAmount(p.amount)} SOL`,
55
+ validator: validatorShort,
56
+ mev: excess > 0 ? `${fmtAmount(excess)} SOL *` : '—',
57
+ };
58
+ }), [
59
+ { key: 'account', header: 'Account' },
60
+ { key: 'staked', header: 'Staked', align: 'right' },
61
+ { key: 'validator', header: 'Validator' },
62
+ { key: 'mev', header: 'MEV', align: 'right' },
63
+ ]));
64
+ lines.push('');
65
+ }
66
+ // Lending section
67
+ const lendPositions = report.positions.filter(p => p.type === 'lend');
68
+ if (lendPositions.length > 0) {
69
+ lines.push('Lending (Kamino)');
70
+ const deposits = lendPositions.filter(p => p.extra?.side === 'deposit');
71
+ const borrows = lendPositions.filter(p => p.extra?.side === 'borrow');
72
+ const allLendRows = [
73
+ ...deposits.map(p => ({
74
+ type: 'Deposit',
75
+ token: p.symbol,
76
+ amount: fmtAmount(p.amount),
77
+ value: p.valueUsd != null ? `$${fmt(p.valueUsd)}` : '—',
78
+ apy: p.extra?.apy != null ? `${(p.extra.apy * 100).toFixed(2)}%` : '—',
79
+ })),
80
+ ...borrows.map(p => ({
81
+ type: 'Borrow',
82
+ token: p.symbol,
83
+ amount: fmtAmount(p.amount),
84
+ value: p.valueUsd != null ? `-$${fmt(Math.abs(p.valueUsd))}` : '—',
85
+ apy: p.extra?.apy != null ? `${(p.extra.apy * 100).toFixed(2)}%` : '—',
86
+ })),
87
+ ];
88
+ lines.push(table(allLendRows, [
89
+ { key: 'type', header: 'Type' },
90
+ { key: 'token', header: 'Token' },
91
+ { key: 'amount', header: 'Amount', align: 'right' },
92
+ { key: 'value', header: 'Value', align: 'right' },
93
+ { key: 'apy', header: 'APY', align: 'right' },
94
+ ]));
95
+ const depositTotal = deposits.reduce((s, p) => s + (p.valueUsd ?? 0), 0);
96
+ const borrowTotal = borrows.reduce((s, p) => s + Math.abs(p.valueUsd ?? 0), 0);
97
+ if (deposits.length > 0 && borrows.length > 0) {
98
+ lines.push(` Net: $${fmt(depositTotal - borrowTotal)}`);
99
+ }
100
+ // Health factor warning
101
+ const healthFactors = borrows.map(p => p.extra?.healthFactor).filter(h => h != null && h > 0);
102
+ if (healthFactors.length > 0) {
103
+ const minHealth = Math.min(...healthFactors);
104
+ if (minHealth < 1.1) {
105
+ lines.push(` Warning: health factor ${minHealth.toFixed(2)} — consider repaying or adding collateral.`);
106
+ }
107
+ }
108
+ lines.push('');
109
+ }
110
+ // Allocation bars
111
+ if (report.allocation.length > 0) {
112
+ lines.push('Allocation');
113
+ const maxSymLen = Math.max(...report.allocation.map(a => a.symbol.length), 4);
114
+ for (const a of report.allocation) {
115
+ const filled = Math.round((a.pct / 100) * BAR_WIDTH);
116
+ const empty = BAR_WIDTH - filled;
117
+ const bar = '\u2588'.repeat(filled) + '\u2591'.repeat(empty);
118
+ const pctStr = `${a.pct.toFixed(1)}%`.padStart(6);
119
+ lines.push(`${a.symbol.padEnd(maxSymLen)} ${bar} ${pctStr}`);
120
+ }
121
+ lines.push('');
122
+ }
123
+ // Per-wallet breakdown (only if multiple wallets)
124
+ if (report.wallets.length > 1) {
125
+ const walletTotals = new Map();
126
+ for (const p of report.positions) {
127
+ if (p.valueUsd != null) {
128
+ walletTotals.set(p.wallet, (walletTotals.get(p.wallet) ?? 0) + p.valueUsd);
129
+ }
130
+ }
131
+ const walletRows = [...walletTotals.entries()]
132
+ .sort((a, b) => b[1] - a[1])
133
+ .map(([name, value]) => ({
134
+ wallet: name,
135
+ value: `$${fmt(value)}`,
136
+ pct: report.totalValueUsd > 0
137
+ ? `${((value / report.totalValueUsd) * 100).toFixed(1)}%`
138
+ : '',
139
+ }));
140
+ lines.push('Wallets');
141
+ lines.push(table(walletRows, [
142
+ { key: 'wallet', header: 'Wallet' },
143
+ { key: 'value', header: 'Value', align: 'right' },
144
+ { key: 'pct', header: '%', align: 'right' },
145
+ ]));
146
+ lines.push('');
147
+ }
148
+ // Footer
149
+ lines.push(`Total: $${fmt(report.totalValueUsd)} across ${walletLabel}`);
150
+ // Signposts
151
+ if (report.claimableMev > 0) {
152
+ lines.push(`* ${fmtAmount(report.claimableMev)} SOL claimable MEV. Run \`sol stake claim-mev\` to compound.`);
153
+ }
154
+ if (report.lastSnapshot) {
155
+ lines.push(`Last snapshot: ${report.lastSnapshot.ago}. Run \`sol portfolio compare\` to see changes.`);
156
+ }
157
+ else {
158
+ lines.push('No snapshots yet. Run `sol portfolio snapshot` to save current state.');
159
+ }
160
+ return lines.join('\n');
161
+ }
162
+ export function renderCompare(result, label) {
163
+ const lines = [];
164
+ lines.push(`${label}\n`);
165
+ if (result.diffs.length === 0) {
166
+ lines.push('No significant changes.');
167
+ }
168
+ else {
169
+ lines.push(table(result.diffs.map(d => ({
170
+ wallet: d.wallet,
171
+ symbol: d.symbol,
172
+ before: `$${fmt(d.valueBefore)}`,
173
+ after: `$${fmt(d.valueAfter)}`,
174
+ change: `${d.change >= 0 ? '+' : ''}$${fmt(d.change)}`,
175
+ pct: d.changePct != null ? `${d.changePct >= 0 ? '+' : ''}${d.changePct.toFixed(1)}%` : '—',
176
+ })), [
177
+ { key: 'wallet', header: 'Wallet' },
178
+ { key: 'symbol', header: 'Token' },
179
+ { key: 'before', header: 'Before', align: 'right' },
180
+ { key: 'after', header: 'After', align: 'right' },
181
+ { key: 'change', header: 'Change', align: 'right' },
182
+ { key: 'pct', header: '%', align: 'right' },
183
+ ]));
184
+ }
185
+ const sign = result.totalChange >= 0 ? '+' : '';
186
+ const pctStr = result.totalChangePct != null
187
+ ? ` (${sign}${result.totalChangePct.toFixed(1)}%)`
188
+ : '';
189
+ lines.push(`\nTotal: $${fmt(result.totalBefore)} → $${fmt(result.totalAfter)} (${sign}$${fmt(result.totalChange)})${pctStr}`);
190
+ return lines.join('\n');
191
+ }
192
+ // ── Helpers ───────────────────────────────────────────────
193
+ function fmt(n) {
194
+ return n.toFixed(2);
195
+ }
196
+ function fmtAmount(n) {
197
+ if (n >= 1_000_000)
198
+ return n.toFixed(0);
199
+ if (n >= 1)
200
+ return n.toFixed(4);
201
+ if (n >= 0.0001)
202
+ return n.toFixed(6);
203
+ return n.toFixed(9);
204
+ }
205
+ //# sourceMappingURL=portfolio-renderer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"portfolio-renderer.js","sourceRoot":"","sources":["../../src/output/portfolio-renderer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface Column {
2
+ key: string;
3
+ header: string;
4
+ align?: 'left' | 'right';
5
+ width?: number;
6
+ format?: (value: unknown) => string;
7
+ }
8
+ export declare function table(rows: Record<string, unknown>[], columns: Column[]): string;
@@ -0,0 +1,22 @@
1
+ export function table(rows, columns) {
2
+ if (rows.length === 0)
3
+ return '(no data)';
4
+ const formatted = rows.map(row => columns.map(col => {
5
+ const val = row[col.key];
6
+ return col.format ? col.format(val) : String(val ?? '');
7
+ }));
8
+ const widths = columns.map((col, i) => {
9
+ const dataMax = formatted.reduce((max, row) => Math.max(max, row[i].length), 0);
10
+ return Math.max(col.header.length, col.width ?? 0, dataMax);
11
+ });
12
+ const header = columns.map((col, i) => pad(col.header, widths[i], col.align)).join(' ');
13
+ const separator = widths.map(w => '─'.repeat(w)).join('──');
14
+ const body = formatted.map(row => columns.map((col, i) => pad(row[i], widths[i], col.align)).join(' ')).join('\n');
15
+ return `${header}\n${separator}\n${body}`;
16
+ }
17
+ function pad(str, width, align = 'left') {
18
+ if (align === 'right')
19
+ return str.padStart(width);
20
+ return str.padEnd(width);
21
+ }
22
+ //# sourceMappingURL=table.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"table.js","sourceRoot":"","sources":["../../src/output/table.ts"],"names":[],"mappings":"AAQA,MAAM,UAAU,KAAK,CAAC,IAA+B,EAAE,OAAiB;IACtE,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,WAAW,CAAC;IAE1C,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAC/B,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;QAChB,MAAM,GAAG,GAAG,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACzB,OAAO,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,IAAI,EAAE,CAAC,CAAC;IAC1D,CAAC,CAAC,CACH,CAAC;IAEF,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;QACpC,MAAM,OAAO,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAChF,OAAO,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC;IAC9D,CAAC,CAAC,CAAC;IAEH,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACzF,MAAM,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAE5D,MAAM,IAAI,GAAG,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAC/B,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACtE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAEb,OAAO,GAAG,MAAM,KAAK,SAAS,KAAK,IAAI,EAAE,CAAC;AAC5C,CAAC;AAED,SAAS,GAAG,CAAC,GAAW,EAAE,KAAa,EAAE,QAA0B,MAAM;IACvE,IAAI,KAAK,KAAK,OAAO;QAAE,OAAO,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IAClD,OAAO,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;AAC3B,CAAC"}
@@ -0,0 +1,7 @@
1
+ export declare function getWalletsDir(): string;
2
+ export declare function ensureWalletsDir(): void;
3
+ export declare function writeKeyFile(name: string, keypairBytes: Uint8Array): string;
4
+ export declare function readKeyFile(filePath: string): Uint8Array;
5
+ export declare function softDeleteKeyFile(filePath: string): void;
6
+ export declare function keyFileExists(name: string): boolean;
7
+ export declare function listKeyFiles(): string[];
@@ -0,0 +1,39 @@
1
+ import { readFileSync, writeFileSync, renameSync, existsSync, mkdirSync, readdirSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ import { getSolDir } from '../core/config-manager.js';
4
+ const WALLETS_DIR = join(getSolDir(), 'wallets');
5
+ export function getWalletsDir() {
6
+ return WALLETS_DIR;
7
+ }
8
+ export function ensureWalletsDir() {
9
+ if (!existsSync(WALLETS_DIR)) {
10
+ mkdirSync(WALLETS_DIR, { recursive: true });
11
+ }
12
+ }
13
+ export function writeKeyFile(name, keypairBytes) {
14
+ ensureWalletsDir();
15
+ const filePath = join(WALLETS_DIR, `${name}.json`);
16
+ const jsonArray = JSON.stringify(Array.from(keypairBytes));
17
+ writeFileSync(filePath, jsonArray, { mode: 0o600 });
18
+ return filePath;
19
+ }
20
+ export function readKeyFile(filePath) {
21
+ const raw = readFileSync(filePath, 'utf-8');
22
+ const arr = JSON.parse(raw);
23
+ return new Uint8Array(arr);
24
+ }
25
+ export function softDeleteKeyFile(filePath) {
26
+ if (existsSync(filePath)) {
27
+ renameSync(filePath, filePath + '.deleted');
28
+ }
29
+ }
30
+ export function keyFileExists(name) {
31
+ return existsSync(join(WALLETS_DIR, `${name}.json`));
32
+ }
33
+ export function listKeyFiles() {
34
+ ensureWalletsDir();
35
+ return readdirSync(WALLETS_DIR)
36
+ .filter(f => f.endsWith('.json'))
37
+ .map(f => f.replace('.json', ''));
38
+ }
39
+ //# sourceMappingURL=fs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fs.js","sourceRoot":"","sources":["../../src/utils/fs.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,UAAU,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AACtG,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AAEtD,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,EAAE,EAAE,SAAS,CAAC,CAAC;AAEjD,MAAM,UAAU,aAAa;IAC3B,OAAO,WAAW,CAAC;AACrB,CAAC;AAED,MAAM,UAAU,gBAAgB;IAC9B,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC7B,SAAS,CAAC,WAAW,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC9C,CAAC;AACH,CAAC;AAED,MAAM,UAAU,YAAY,CAAC,IAAY,EAAE,YAAwB;IACjE,gBAAgB,EAAE,CAAC;IACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,GAAG,IAAI,OAAO,CAAC,CAAC;IACnD,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;IAC3D,aAAa,CAAC,QAAQ,EAAE,SAAS,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;IACpD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,QAAgB;IAC1C,MAAM,GAAG,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAC5C,MAAM,GAAG,GAAa,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACtC,OAAO,IAAI,UAAU,CAAC,GAAG,CAAC,CAAC;AAC7B,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,QAAgB;IAChD,IAAI,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QACzB,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,UAAU,CAAC,CAAC;IAC9C,CAAC;AACH,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,IAAY;IACxC,OAAO,UAAU,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,IAAI,OAAO,CAAC,CAAC,CAAC;AACvD,CAAC;AAED,MAAM,UAAU,YAAY;IAC1B,gBAAgB,EAAE,CAAC;IACnB,OAAO,WAAW,CAAC,WAAW,CAAC;SAC5B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;SAChC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AACtC,CAAC"}
@@ -0,0 +1,17 @@
1
+ export interface RetryOptions {
2
+ maxRetries?: number;
3
+ baseDelay?: number;
4
+ maxDelay?: number;
5
+ jitter?: boolean;
6
+ shouldRetry?: (error: unknown) => boolean;
7
+ onRetry?: (error: unknown, attempt: number) => void;
8
+ }
9
+ export declare function withRetry<T>(fn: () => Promise<T>, opts?: RetryOptions): Promise<T>;
10
+ export declare function isRetryableHttpError(error: unknown): boolean;
11
+ export declare class RateLimiter {
12
+ private timestamps;
13
+ private readonly windowMs;
14
+ private readonly maxRequests;
15
+ constructor(maxRequests: number, windowMs: number);
16
+ acquire(): Promise<void>;
17
+ }
@@ -0,0 +1,55 @@
1
+ export async function withRetry(fn, opts = {}) {
2
+ const { maxRetries = 3, baseDelay = 500, maxDelay = 5000, jitter = true, shouldRetry = () => true, onRetry, } = opts;
3
+ let lastError;
4
+ for (let attempt = 0; attempt <= maxRetries; attempt++) {
5
+ try {
6
+ return await fn();
7
+ }
8
+ catch (error) {
9
+ lastError = error;
10
+ if (attempt >= maxRetries || !shouldRetry(error)) {
11
+ throw error;
12
+ }
13
+ const delay = Math.min(baseDelay * Math.pow(2, attempt), maxDelay);
14
+ const finalDelay = jitter ? delay * (0.5 + Math.random() * 0.5) : delay;
15
+ onRetry?.(error, attempt + 1);
16
+ await sleep(finalDelay);
17
+ }
18
+ }
19
+ throw lastError;
20
+ }
21
+ function sleep(ms) {
22
+ return new Promise(resolve => setTimeout(resolve, ms));
23
+ }
24
+ export function isRetryableHttpError(error) {
25
+ if (error instanceof Error) {
26
+ const msg = error.message.toLowerCase();
27
+ if (msg.includes('429') || msg.includes('rate limit'))
28
+ return true;
29
+ if (msg.includes('5') && msg.includes('server error'))
30
+ return true;
31
+ if (msg.includes('timeout') || msg.includes('econnreset') || msg.includes('enotfound'))
32
+ return true;
33
+ }
34
+ return false;
35
+ }
36
+ export class RateLimiter {
37
+ timestamps = [];
38
+ windowMs;
39
+ maxRequests;
40
+ constructor(maxRequests, windowMs) {
41
+ this.maxRequests = maxRequests;
42
+ this.windowMs = windowMs;
43
+ }
44
+ async acquire() {
45
+ const now = Date.now();
46
+ this.timestamps = this.timestamps.filter(t => now - t < this.windowMs);
47
+ if (this.timestamps.length >= this.maxRequests) {
48
+ const oldest = this.timestamps[0];
49
+ const waitMs = this.windowMs - (now - oldest) + 10;
50
+ await sleep(waitMs);
51
+ }
52
+ this.timestamps.push(Date.now());
53
+ }
54
+ }
55
+ //# sourceMappingURL=retry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retry.js","sourceRoot":"","sources":["../../src/utils/retry.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ export declare function isValidAddress(address: string): boolean;
2
+ export declare function lamportsToSol(lamports: bigint | number): number;
3
+ export declare function solToLamports(sol: number): bigint;
4
+ export declare function tokenAmountToUi(rawAmount: bigint | string, decimals: number): number;
5
+ export declare function uiToTokenAmount(uiAmount: number, decimals: number): bigint;
6
+ export declare function shortenAddress(address: string, chars?: number): string;
7
+ export declare function explorerUrl(signature: string, type?: 'tx' | 'address'): string;
8
+ export declare const SOL_MINT = "So11111111111111111111111111111111111111112";
9
+ export declare const SOL_DECIMALS = 9;