perp-cli 0.3.3

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 (325) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +293 -0
  3. package/dist/__tests__/alert-logic.test.d.ts +1 -0
  4. package/dist/__tests__/alert-logic.test.js +107 -0
  5. package/dist/__tests__/arb-auto-3dex.test.d.ts +1 -0
  6. package/dist/__tests__/arb-auto-3dex.test.js +397 -0
  7. package/dist/__tests__/arb-history-stats.test.d.ts +1 -0
  8. package/dist/__tests__/arb-history-stats.test.js +176 -0
  9. package/dist/__tests__/arb-logic.test.d.ts +1 -0
  10. package/dist/__tests__/arb-logic.test.js +84 -0
  11. package/dist/__tests__/arb-manage.test.d.ts +1 -0
  12. package/dist/__tests__/arb-manage.test.js +253 -0
  13. package/dist/__tests__/arb-new-features.test.d.ts +1 -0
  14. package/dist/__tests__/arb-new-features.test.js +457 -0
  15. package/dist/__tests__/arb-sizing.test.d.ts +1 -0
  16. package/dist/__tests__/arb-sizing.test.js +48 -0
  17. package/dist/__tests__/arb-state.test.d.ts +1 -0
  18. package/dist/__tests__/arb-state.test.js +284 -0
  19. package/dist/__tests__/arb-userflow.test.d.ts +1 -0
  20. package/dist/__tests__/arb-userflow.test.js +945 -0
  21. package/dist/__tests__/arb-utils.test.d.ts +1 -0
  22. package/dist/__tests__/arb-utils.test.js +264 -0
  23. package/dist/__tests__/bot-conditions.test.d.ts +1 -0
  24. package/dist/__tests__/bot-conditions.test.js +341 -0
  25. package/dist/__tests__/client-id-tracker.test.d.ts +1 -0
  26. package/dist/__tests__/client-id-tracker.test.js +137 -0
  27. package/dist/__tests__/commands/new-atomic-commands.test.d.ts +1 -0
  28. package/dist/__tests__/commands/new-atomic-commands.test.js +502 -0
  29. package/dist/__tests__/commands/order-intent.test.d.ts +1 -0
  30. package/dist/__tests__/commands/order-intent.test.js +600 -0
  31. package/dist/__tests__/commands/trade-commands.test.d.ts +1 -0
  32. package/dist/__tests__/commands/trade-commands.test.js +821 -0
  33. package/dist/__tests__/config.test.d.ts +1 -0
  34. package/dist/__tests__/config.test.js +86 -0
  35. package/dist/__tests__/cross-chain-margin.test.d.ts +1 -0
  36. package/dist/__tests__/cross-chain-margin.test.js +287 -0
  37. package/dist/__tests__/dex-asset-map.test.d.ts +1 -0
  38. package/dist/__tests__/dex-asset-map.test.js +191 -0
  39. package/dist/__tests__/errors.test.d.ts +1 -0
  40. package/dist/__tests__/errors.test.js +110 -0
  41. package/dist/__tests__/event-stream.test.d.ts +1 -0
  42. package/dist/__tests__/event-stream.test.js +276 -0
  43. package/dist/__tests__/exchanges/interface.test.d.ts +1 -0
  44. package/dist/__tests__/exchanges/interface.test.js +132 -0
  45. package/dist/__tests__/exchanges/mock-adapter.d.ts +69 -0
  46. package/dist/__tests__/exchanges/mock-adapter.js +137 -0
  47. package/dist/__tests__/execution-log.test.d.ts +1 -0
  48. package/dist/__tests__/execution-log.test.js +106 -0
  49. package/dist/__tests__/funding-calc.test.d.ts +1 -0
  50. package/dist/__tests__/funding-calc.test.js +71 -0
  51. package/dist/__tests__/funding-history.test.d.ts +1 -0
  52. package/dist/__tests__/funding-history.test.js +343 -0
  53. package/dist/__tests__/funding-rates.test.d.ts +1 -0
  54. package/dist/__tests__/funding-rates.test.js +342 -0
  55. package/dist/__tests__/funding.test.d.ts +1 -0
  56. package/dist/__tests__/funding.test.js +173 -0
  57. package/dist/__tests__/gap-logic.test.d.ts +1 -0
  58. package/dist/__tests__/gap-logic.test.js +43 -0
  59. package/dist/__tests__/hip3-dex.test.d.ts +1 -0
  60. package/dist/__tests__/hip3-dex.test.js +234 -0
  61. package/dist/__tests__/integration/agent-features.integration.test.d.ts +1 -0
  62. package/dist/__tests__/integration/agent-features.integration.test.js +553 -0
  63. package/dist/__tests__/integration/atomic-commands.integration.test.d.ts +13 -0
  64. package/dist/__tests__/integration/atomic-commands.integration.test.js +246 -0
  65. package/dist/__tests__/integration/bridge-simulation.integration.test.d.ts +1 -0
  66. package/dist/__tests__/integration/bridge-simulation.integration.test.js +453 -0
  67. package/dist/__tests__/integration/bridge-strict.integration.test.d.ts +1 -0
  68. package/dist/__tests__/integration/bridge-strict.integration.test.js +812 -0
  69. package/dist/__tests__/integration/bridge.integration.test.d.ts +1 -0
  70. package/dist/__tests__/integration/bridge.integration.test.js +309 -0
  71. package/dist/__tests__/integration/cli-e2e.integration.test.d.ts +1 -0
  72. package/dist/__tests__/integration/cli-e2e.integration.test.js +202 -0
  73. package/dist/__tests__/integration/dex-arb.integration.test.d.ts +1 -0
  74. package/dist/__tests__/integration/dex-arb.integration.test.js +116 -0
  75. package/dist/__tests__/integration/envelope-consistency.integration.test.d.ts +13 -0
  76. package/dist/__tests__/integration/envelope-consistency.integration.test.js +205 -0
  77. package/dist/__tests__/integration/hip3-dex.integration.test.d.ts +1 -0
  78. package/dist/__tests__/integration/hip3-dex.integration.test.js +147 -0
  79. package/dist/__tests__/integration/hyperliquid.integration.test.d.ts +1 -0
  80. package/dist/__tests__/integration/hyperliquid.integration.test.js +79 -0
  81. package/dist/__tests__/integration/lighter.integration.test.d.ts +1 -0
  82. package/dist/__tests__/integration/lighter.integration.test.js +53 -0
  83. package/dist/__tests__/integration/new-commands-e2e.integration.test.d.ts +9 -0
  84. package/dist/__tests__/integration/new-commands-e2e.integration.test.js +236 -0
  85. package/dist/__tests__/integration/order-verification.integration.test.d.ts +1 -0
  86. package/dist/__tests__/integration/order-verification.integration.test.js +321 -0
  87. package/dist/__tests__/integration/pacifica.integration.test.d.ts +1 -0
  88. package/dist/__tests__/integration/pacifica.integration.test.js +75 -0
  89. package/dist/__tests__/integration/response-shapes.integration.test.d.ts +1 -0
  90. package/dist/__tests__/integration/response-shapes.integration.test.js +278 -0
  91. package/dist/__tests__/liquidity.test.d.ts +1 -0
  92. package/dist/__tests__/liquidity.test.js +225 -0
  93. package/dist/__tests__/plan-executor.test.d.ts +1 -0
  94. package/dist/__tests__/plan-executor.test.js +314 -0
  95. package/dist/__tests__/position-history.test.d.ts +1 -0
  96. package/dist/__tests__/position-history.test.js +367 -0
  97. package/dist/__tests__/retry.test.d.ts +1 -0
  98. package/dist/__tests__/retry.test.js +310 -0
  99. package/dist/__tests__/risk-assessment.test.d.ts +1 -0
  100. package/dist/__tests__/risk-assessment.test.js +145 -0
  101. package/dist/__tests__/security-adversarial.test.d.ts +1 -0
  102. package/dist/__tests__/security-adversarial.test.js +574 -0
  103. package/dist/__tests__/strategies.test.d.ts +1 -0
  104. package/dist/__tests__/strategies.test.js +539 -0
  105. package/dist/__tests__/trade-execution.test.d.ts +1 -0
  106. package/dist/__tests__/trade-execution.test.js +129 -0
  107. package/dist/__tests__/trade-validator.test.d.ts +1 -0
  108. package/dist/__tests__/trade-validator.test.js +655 -0
  109. package/dist/__tests__/utils.test.d.ts +1 -0
  110. package/dist/__tests__/utils.test.js +76 -0
  111. package/dist/api/public/hyperliquid.d.ts +18 -0
  112. package/dist/api/public/hyperliquid.js +82 -0
  113. package/dist/api/public/index.d.ts +8 -0
  114. package/dist/api/public/index.js +8 -0
  115. package/dist/api/public/lighter.d.ts +24 -0
  116. package/dist/api/public/lighter.js +100 -0
  117. package/dist/api/public/pacifica.d.ts +17 -0
  118. package/dist/api/public/pacifica.js +54 -0
  119. package/dist/api/public/urls.d.ts +12 -0
  120. package/dist/api/public/urls.js +33 -0
  121. package/dist/arb/history-stats.d.ts +44 -0
  122. package/dist/arb/history-stats.js +135 -0
  123. package/dist/arb/index.d.ts +4 -0
  124. package/dist/arb/index.js +4 -0
  125. package/dist/arb/sizing.d.ts +23 -0
  126. package/dist/arb/sizing.js +96 -0
  127. package/dist/arb/state.d.ts +51 -0
  128. package/dist/arb/state.js +112 -0
  129. package/dist/arb/utils.d.ts +81 -0
  130. package/dist/arb/utils.js +267 -0
  131. package/dist/arb-history-stats.d.ts +5 -0
  132. package/dist/arb-history-stats.js +5 -0
  133. package/dist/arb-sizing.d.ts +5 -0
  134. package/dist/arb-sizing.js +5 -0
  135. package/dist/arb-state.d.ts +5 -0
  136. package/dist/arb-state.js +5 -0
  137. package/dist/arb-utils.d.ts +5 -0
  138. package/dist/arb-utils.js +5 -0
  139. package/dist/bot/conditions.d.ts +32 -0
  140. package/dist/bot/conditions.js +141 -0
  141. package/dist/bot/config.d.ts +76 -0
  142. package/dist/bot/config.js +160 -0
  143. package/dist/bot/engine.d.ts +8 -0
  144. package/dist/bot/engine.js +519 -0
  145. package/dist/bot/presets.d.ts +11 -0
  146. package/dist/bot/presets.js +296 -0
  147. package/dist/bridge-engine.d.ts +133 -0
  148. package/dist/bridge-engine.js +1487 -0
  149. package/dist/cache.d.ts +25 -0
  150. package/dist/cache.js +99 -0
  151. package/dist/cli-spec.d.ts +50 -0
  152. package/dist/cli-spec.js +75 -0
  153. package/dist/client-id-tracker.d.ts +25 -0
  154. package/dist/client-id-tracker.js +76 -0
  155. package/dist/commands/account.d.ts +3 -0
  156. package/dist/commands/account.js +425 -0
  157. package/dist/commands/agent.d.ts +3 -0
  158. package/dist/commands/agent.js +386 -0
  159. package/dist/commands/alert.d.ts +2 -0
  160. package/dist/commands/alert.js +421 -0
  161. package/dist/commands/analytics.d.ts +3 -0
  162. package/dist/commands/analytics.js +311 -0
  163. package/dist/commands/arb/index.d.ts +3 -0
  164. package/dist/commands/arb/index.js +921 -0
  165. package/dist/commands/arb-auto.d.ts +54 -0
  166. package/dist/commands/arb-auto.js +1328 -0
  167. package/dist/commands/arb-manage.d.ts +5 -0
  168. package/dist/commands/arb-manage.js +5 -0
  169. package/dist/commands/arb.d.ts +2 -0
  170. package/dist/commands/arb.js +347 -0
  171. package/dist/commands/backtest.d.ts +2 -0
  172. package/dist/commands/backtest.js +327 -0
  173. package/dist/commands/bot.d.ts +3 -0
  174. package/dist/commands/bot.js +412 -0
  175. package/dist/commands/bridge.d.ts +2 -0
  176. package/dist/commands/bridge.js +396 -0
  177. package/dist/commands/dashboard.d.ts +3 -0
  178. package/dist/commands/dashboard.js +176 -0
  179. package/dist/commands/deposit.d.ts +4 -0
  180. package/dist/commands/deposit.js +573 -0
  181. package/dist/commands/dex.d.ts +3 -0
  182. package/dist/commands/dex.js +114 -0
  183. package/dist/commands/env.d.ts +2 -0
  184. package/dist/commands/env.js +136 -0
  185. package/dist/commands/funding.d.ts +2 -0
  186. package/dist/commands/funding.js +347 -0
  187. package/dist/commands/gap.d.ts +2 -0
  188. package/dist/commands/gap.js +305 -0
  189. package/dist/commands/health.d.ts +2 -0
  190. package/dist/commands/health.js +67 -0
  191. package/dist/commands/history.d.ts +2 -0
  192. package/dist/commands/history.js +235 -0
  193. package/dist/commands/init.d.ts +15 -0
  194. package/dist/commands/init.js +266 -0
  195. package/dist/commands/jobs.d.ts +2 -0
  196. package/dist/commands/jobs.js +133 -0
  197. package/dist/commands/manage.d.ts +4 -0
  198. package/dist/commands/manage.js +309 -0
  199. package/dist/commands/market.d.ts +3 -0
  200. package/dist/commands/market.js +225 -0
  201. package/dist/commands/plan.d.ts +3 -0
  202. package/dist/commands/plan.js +95 -0
  203. package/dist/commands/portfolio.d.ts +3 -0
  204. package/dist/commands/portfolio.js +169 -0
  205. package/dist/commands/rebalance.d.ts +3 -0
  206. package/dist/commands/rebalance.js +293 -0
  207. package/dist/commands/risk.d.ts +3 -0
  208. package/dist/commands/risk.js +169 -0
  209. package/dist/commands/run.d.ts +3 -0
  210. package/dist/commands/run.js +202 -0
  211. package/dist/commands/settings.d.ts +2 -0
  212. package/dist/commands/settings.js +102 -0
  213. package/dist/commands/stream.d.ts +5 -0
  214. package/dist/commands/stream.js +123 -0
  215. package/dist/commands/trade.d.ts +3 -0
  216. package/dist/commands/trade.js +1273 -0
  217. package/dist/commands/wallet.d.ts +14 -0
  218. package/dist/commands/wallet.js +602 -0
  219. package/dist/commands/withdraw.d.ts +3 -0
  220. package/dist/commands/withdraw.js +187 -0
  221. package/dist/config.d.ts +5 -0
  222. package/dist/config.js +68 -0
  223. package/dist/cross-chain-margin.d.ts +46 -0
  224. package/dist/cross-chain-margin.js +107 -0
  225. package/dist/dashboard/server.d.ts +80 -0
  226. package/dist/dashboard/server.js +340 -0
  227. package/dist/dashboard/ui.d.ts +4 -0
  228. package/dist/dashboard/ui.js +538 -0
  229. package/dist/dashboard/ws-feeds.d.ts +29 -0
  230. package/dist/dashboard/ws-feeds.js +660 -0
  231. package/dist/dex-asset-map.d.ts +80 -0
  232. package/dist/dex-asset-map.js +201 -0
  233. package/dist/errors.d.ts +109 -0
  234. package/dist/errors.js +84 -0
  235. package/dist/event-stream.d.ts +25 -0
  236. package/dist/event-stream.js +168 -0
  237. package/dist/exchanges/hyperliquid.d.ts +212 -0
  238. package/dist/exchanges/hyperliquid.js +931 -0
  239. package/dist/exchanges/interface.d.ts +95 -0
  240. package/dist/exchanges/interface.js +5 -0
  241. package/dist/exchanges/lighter.d.ts +159 -0
  242. package/dist/exchanges/lighter.js +793 -0
  243. package/dist/exchanges/pacifica.d.ts +51 -0
  244. package/dist/exchanges/pacifica.js +248 -0
  245. package/dist/execution-log.d.ts +36 -0
  246. package/dist/execution-log.js +102 -0
  247. package/dist/funding/history.d.ts +63 -0
  248. package/dist/funding/history.js +266 -0
  249. package/dist/funding/index.d.ts +3 -0
  250. package/dist/funding/index.js +3 -0
  251. package/dist/funding/normalize.d.ts +39 -0
  252. package/dist/funding/normalize.js +66 -0
  253. package/dist/funding/rates.d.ts +45 -0
  254. package/dist/funding/rates.js +172 -0
  255. package/dist/funding-history.d.ts +5 -0
  256. package/dist/funding-history.js +5 -0
  257. package/dist/funding-rates.d.ts +5 -0
  258. package/dist/funding-rates.js +5 -0
  259. package/dist/funding.d.ts +5 -0
  260. package/dist/funding.js +5 -0
  261. package/dist/index.d.ts +2 -0
  262. package/dist/index.js +458 -0
  263. package/dist/jobs.d.ts +37 -0
  264. package/dist/jobs.js +152 -0
  265. package/dist/liquidity.d.ts +34 -0
  266. package/dist/liquidity.js +100 -0
  267. package/dist/mcp-server.d.ts +9 -0
  268. package/dist/mcp-server.js +1206 -0
  269. package/dist/pacifica/client.d.ts +111 -0
  270. package/dist/pacifica/client.js +310 -0
  271. package/dist/pacifica/constants.d.ts +27 -0
  272. package/dist/pacifica/constants.js +47 -0
  273. package/dist/pacifica/deposit.d.ts +14 -0
  274. package/dist/pacifica/deposit.js +78 -0
  275. package/dist/pacifica/index.d.ts +6 -0
  276. package/dist/pacifica/index.js +11 -0
  277. package/dist/pacifica/signing.d.ts +49 -0
  278. package/dist/pacifica/signing.js +97 -0
  279. package/dist/pacifica/types/account.d.ts +42 -0
  280. package/dist/pacifica/types/account.js +1 -0
  281. package/dist/pacifica/types/index.d.ts +6 -0
  282. package/dist/pacifica/types/index.js +6 -0
  283. package/dist/pacifica/types/lake.d.ts +18 -0
  284. package/dist/pacifica/types/lake.js +1 -0
  285. package/dist/pacifica/types/market.d.ts +64 -0
  286. package/dist/pacifica/types/market.js +1 -0
  287. package/dist/pacifica/types/order.d.ts +92 -0
  288. package/dist/pacifica/types/order.js +1 -0
  289. package/dist/pacifica/types/position.d.ts +25 -0
  290. package/dist/pacifica/types/position.js +1 -0
  291. package/dist/pacifica/types/ws.d.ts +34 -0
  292. package/dist/pacifica/types/ws.js +41 -0
  293. package/dist/pacifica/ws-client.d.ts +42 -0
  294. package/dist/pacifica/ws-client.js +180 -0
  295. package/dist/plan-executor.d.ts +48 -0
  296. package/dist/plan-executor.js +280 -0
  297. package/dist/position-history.d.ts +68 -0
  298. package/dist/position-history.js +222 -0
  299. package/dist/rebalance.d.ts +64 -0
  300. package/dist/rebalance.js +142 -0
  301. package/dist/retry.d.ts +74 -0
  302. package/dist/retry.js +129 -0
  303. package/dist/risk.d.ts +48 -0
  304. package/dist/risk.js +156 -0
  305. package/dist/settings.d.ts +19 -0
  306. package/dist/settings.js +45 -0
  307. package/dist/shared-api.d.ts +5 -0
  308. package/dist/shared-api.js +5 -0
  309. package/dist/strategies/dca.d.ts +25 -0
  310. package/dist/strategies/dca.js +114 -0
  311. package/dist/strategies/funding-arb.d.ts +15 -0
  312. package/dist/strategies/funding-arb.js +281 -0
  313. package/dist/strategies/grid.d.ts +34 -0
  314. package/dist/strategies/grid.js +185 -0
  315. package/dist/strategies/trailing-stop.d.ts +17 -0
  316. package/dist/strategies/trailing-stop.js +121 -0
  317. package/dist/strategies/twap.d.ts +20 -0
  318. package/dist/strategies/twap.js +78 -0
  319. package/dist/trade-validator.d.ts +39 -0
  320. package/dist/trade-validator.js +154 -0
  321. package/dist/utils.d.ts +38 -0
  322. package/dist/utils.js +110 -0
  323. package/package.json +63 -0
  324. package/skills/perp-cli/SKILL.md +149 -0
  325. package/skills/perp-cli/references/commands.md +143 -0
@@ -0,0 +1,311 @@
1
+ import chalk from "chalk";
2
+ import { printJson, jsonOk, makeTable, formatUsd, formatPnl, withJsonErrors } from "../utils.js";
3
+ import { readExecutionLog, getExecutionStats } from "../execution-log.js";
4
+ const EXCHANGES = ["pacifica", "hyperliquid", "lighter"];
5
+ function parseSince(since) {
6
+ if (!since)
7
+ return undefined;
8
+ const match = since.match(/^(\d+)(h|d|w)$/);
9
+ if (match) {
10
+ const [, num, unit] = match;
11
+ const ms = { h: 3600000, d: 86400000, w: 604800000 }[unit] ?? 86400000;
12
+ return new Date(Date.now() - parseInt(num) * ms).toISOString();
13
+ }
14
+ return since;
15
+ }
16
+ export function registerAnalyticsCommands(program, getAdapterForExchange, isJson) {
17
+ const analytics = program.command("analytics").description("Trading performance analytics");
18
+ // ── analytics summary ──
19
+ analytics
20
+ .command("summary")
21
+ .description("Execution log summary statistics")
22
+ .option("--since <period>", "Period: 24h, 7d, 30d, or ISO date")
23
+ .action(async (opts) => {
24
+ await withJsonErrors(isJson(), async () => {
25
+ const since = parseSince(opts.since);
26
+ const stats = getExecutionStats(since);
27
+ const records = readExecutionLog({ since });
28
+ // Calculate volume
29
+ let totalVolume = 0;
30
+ for (const r of records) {
31
+ if (r.notional)
32
+ totalVolume += r.notional;
33
+ else if (r.price && r.size)
34
+ totalVolume += Number(r.price) * Number(r.size);
35
+ }
36
+ const result = { ...stats, totalVolume, period: opts.since ?? "all time" };
37
+ if (isJson())
38
+ return printJson(jsonOk(result));
39
+ console.log(chalk.cyan.bold(`\n Trading Summary ${opts.since ? `(${opts.since})` : "(all time)"}\n`));
40
+ console.log(` Total Trades: ${stats.totalTrades}`);
41
+ console.log(` Success Rate: ${stats.successRate.toFixed(1)}%`);
42
+ console.log(` Total Volume: $${formatUsd(totalVolume)}`);
43
+ if (Object.keys(stats.byExchange).length > 0) {
44
+ console.log(chalk.white.bold("\n By Exchange:"));
45
+ for (const [ex, count] of Object.entries(stats.byExchange)) {
46
+ console.log(` ${ex.padEnd(14)} ${count} trades`);
47
+ }
48
+ }
49
+ if (Object.keys(stats.byType).length > 0) {
50
+ console.log(chalk.white.bold("\n By Type:"));
51
+ for (const [type, count] of Object.entries(stats.byType)) {
52
+ console.log(` ${type.replace(/_/g, " ").padEnd(14)} ${count}`);
53
+ }
54
+ }
55
+ if (stats.recentErrors.length > 0) {
56
+ console.log(chalk.red.bold("\n Recent Errors:"));
57
+ for (const err of stats.recentErrors) {
58
+ console.log(` ${chalk.red(err)}`);
59
+ }
60
+ }
61
+ console.log();
62
+ });
63
+ });
64
+ // ── analytics pnl ──
65
+ analytics
66
+ .command("pnl")
67
+ .description("Realized P&L from exchange trade history")
68
+ .option("-e, --exchange <exchanges>", "Comma-separated exchanges")
69
+ .option("--since <period>", "Period: 24h, 7d, 30d")
70
+ .option("-n, --limit <n>", "Trade history limit per exchange", "100")
71
+ .action(async (opts) => {
72
+ await withJsonErrors(isJson(), async () => {
73
+ const exchanges = opts.exchange
74
+ ? opts.exchange.split(",").map(e => e.trim())
75
+ : [...EXCHANGES];
76
+ const limit = parseInt(opts.limit);
77
+ const sinceMs = opts.since ? new Date(parseSince(opts.since)).getTime() : 0;
78
+ const allTrades = [];
79
+ await Promise.allSettled(exchanges.map(async (ex) => {
80
+ try {
81
+ const adapter = await getAdapterForExchange(ex);
82
+ const trades = await adapter.getTradeHistory(limit);
83
+ for (const t of trades) {
84
+ if (sinceMs && t.time < sinceMs)
85
+ continue;
86
+ allTrades.push({
87
+ exchange: ex,
88
+ symbol: t.symbol,
89
+ side: t.side,
90
+ price: Number(t.price),
91
+ size: Number(t.size),
92
+ fee: Number(t.fee),
93
+ time: t.time,
94
+ });
95
+ }
96
+ }
97
+ catch { /* skip unavailable exchanges */ }
98
+ }));
99
+ // Aggregate by symbol
100
+ const bySymbol = new Map();
101
+ const byExchange = new Map();
102
+ let totalVolume = 0;
103
+ let totalFees = 0;
104
+ for (const t of allTrades) {
105
+ const notional = t.price * t.size;
106
+ totalVolume += notional;
107
+ totalFees += t.fee;
108
+ const sym = bySymbol.get(t.symbol) ?? { volume: 0, fees: 0, trades: 0 };
109
+ sym.volume += notional;
110
+ sym.fees += t.fee;
111
+ sym.trades++;
112
+ bySymbol.set(t.symbol, sym);
113
+ const ex = byExchange.get(t.exchange) ?? { volume: 0, fees: 0, trades: 0 };
114
+ ex.volume += notional;
115
+ ex.fees += t.fee;
116
+ ex.trades++;
117
+ byExchange.set(t.exchange, ex);
118
+ }
119
+ const result = {
120
+ totalTrades: allTrades.length,
121
+ totalVolume,
122
+ totalFees,
123
+ netAfterFees: -totalFees, // realized PnL would need entry/exit matching; fees are definite cost
124
+ bySymbol: Object.fromEntries(bySymbol),
125
+ byExchange: Object.fromEntries(byExchange),
126
+ };
127
+ if (isJson())
128
+ return printJson(jsonOk(result));
129
+ console.log(chalk.cyan.bold(`\n Realized Trading P&L ${opts.since ? `(${opts.since})` : ""}\n`));
130
+ console.log(` Total Trades: ${allTrades.length}`);
131
+ console.log(` Total Volume: $${formatUsd(totalVolume)}`);
132
+ console.log(` Total Fees: ${chalk.red(`-$${formatUsd(totalFees)}`)}`);
133
+ if (bySymbol.size > 0) {
134
+ console.log(chalk.white.bold("\n By Symbol:"));
135
+ const symRows = [...bySymbol.entries()]
136
+ .sort((a, b) => b[1].volume - a[1].volume)
137
+ .map(([sym, d]) => [
138
+ chalk.white.bold(sym),
139
+ String(d.trades),
140
+ `$${formatUsd(d.volume)}`,
141
+ chalk.red(`-$${formatUsd(d.fees)}`),
142
+ ]);
143
+ console.log(makeTable(["Symbol", "Trades", "Volume", "Fees"], symRows));
144
+ }
145
+ if (byExchange.size > 0) {
146
+ console.log(chalk.white.bold(" By Exchange:"));
147
+ const exRows = [...byExchange.entries()].map(([ex, d]) => [
148
+ chalk.white.bold(ex),
149
+ String(d.trades),
150
+ `$${formatUsd(d.volume)}`,
151
+ chalk.red(`-$${formatUsd(d.fees)}`),
152
+ ]);
153
+ console.log(makeTable(["Exchange", "Trades", "Volume", "Fees"], exRows));
154
+ }
155
+ });
156
+ });
157
+ // ── analytics funding ──
158
+ analytics
159
+ .command("funding")
160
+ .description("Funding payment history across exchanges")
161
+ .option("-e, --exchange <exchanges>", "Comma-separated exchanges")
162
+ .option("-s, --symbol <symbol>", "Filter by symbol")
163
+ .option("-n, --limit <n>", "Funding history limit per exchange", "50")
164
+ .action(async (opts) => {
165
+ await withJsonErrors(isJson(), async () => {
166
+ const exchanges = opts.exchange
167
+ ? opts.exchange.split(",").map(e => e.trim())
168
+ : [...EXCHANGES];
169
+ const allFunding = [];
170
+ await Promise.allSettled(exchanges.map(async (ex) => {
171
+ try {
172
+ const adapter = await getAdapterForExchange(ex);
173
+ const payments = await adapter.getFundingPayments(parseInt(opts.limit));
174
+ for (const p of payments) {
175
+ if (opts.symbol && !p.symbol.toUpperCase().includes(opts.symbol.toUpperCase()))
176
+ continue;
177
+ allFunding.push({
178
+ exchange: ex,
179
+ symbol: p.symbol,
180
+ payment: Number(p.payment),
181
+ time: p.time,
182
+ });
183
+ }
184
+ }
185
+ catch { /* skip */ }
186
+ }));
187
+ // Aggregate
188
+ const bySymbol = new Map();
189
+ const byExchange = new Map();
190
+ let totalFunding = 0;
191
+ for (const f of allFunding) {
192
+ totalFunding += f.payment;
193
+ bySymbol.set(f.symbol, (bySymbol.get(f.symbol) ?? 0) + f.payment);
194
+ byExchange.set(f.exchange, (byExchange.get(f.exchange) ?? 0) + f.payment);
195
+ }
196
+ const result = {
197
+ totalPayments: allFunding.length,
198
+ totalFunding,
199
+ bySymbol: Object.fromEntries(bySymbol),
200
+ byExchange: Object.fromEntries(byExchange),
201
+ };
202
+ if (isJson())
203
+ return printJson(jsonOk(result));
204
+ console.log(chalk.cyan.bold("\n Funding Payment Summary\n"));
205
+ console.log(` Total Payments: ${allFunding.length}`);
206
+ console.log(` Net Funding: ${formatPnl(totalFunding)}`);
207
+ if (bySymbol.size > 0) {
208
+ console.log(chalk.white.bold("\n By Symbol:"));
209
+ const rows = [...bySymbol.entries()]
210
+ .sort((a, b) => b[1] - a[1])
211
+ .map(([sym, amt]) => [chalk.white.bold(sym), formatPnl(amt)]);
212
+ console.log(makeTable(["Symbol", "Net Funding"], rows));
213
+ }
214
+ if (byExchange.size > 0) {
215
+ console.log(chalk.white.bold(" By Exchange:"));
216
+ const rows = [...byExchange.entries()]
217
+ .map(([ex, amt]) => [chalk.white.bold(ex), formatPnl(amt)]);
218
+ console.log(makeTable(["Exchange", "Net Funding"], rows));
219
+ }
220
+ });
221
+ });
222
+ // ── analytics report ──
223
+ analytics
224
+ .command("report")
225
+ .description("Full performance report (summary + pnl + funding)")
226
+ .option("-e, --exchange <exchanges>", "Comma-separated exchanges")
227
+ .option("--since <period>", "Period: 24h, 7d, 30d")
228
+ .action(async (opts) => {
229
+ await withJsonErrors(isJson(), async () => {
230
+ const since = parseSince(opts.since);
231
+ const stats = getExecutionStats(since);
232
+ const records = readExecutionLog({ since });
233
+ const exchanges = opts.exchange
234
+ ? opts.exchange.split(",").map(e => e.trim())
235
+ : [...EXCHANGES];
236
+ // Volume from log
237
+ let logVolume = 0;
238
+ for (const r of records) {
239
+ if (r.notional)
240
+ logVolume += r.notional;
241
+ }
242
+ // Fetch live data from exchanges
243
+ let totalFunding = 0;
244
+ let totalFees = 0;
245
+ let totalEquity = 0;
246
+ let totalUPnl = 0;
247
+ const symbolSet = new Set();
248
+ await Promise.allSettled(exchanges.map(async (ex) => {
249
+ try {
250
+ const adapter = await getAdapterForExchange(ex);
251
+ const [balance, trades, funding] = await Promise.all([
252
+ adapter.getBalance(),
253
+ adapter.getTradeHistory(50).catch(() => []),
254
+ adapter.getFundingPayments(50).catch(() => []),
255
+ ]);
256
+ totalEquity += Number(balance.equity);
257
+ totalUPnl += Number(balance.unrealizedPnl);
258
+ for (const t of trades) {
259
+ totalFees += Number(t.fee);
260
+ symbolSet.add(t.symbol);
261
+ }
262
+ for (const f of funding)
263
+ totalFunding += Number(f.payment);
264
+ }
265
+ catch { /* skip */ }
266
+ }));
267
+ const report = {
268
+ period: opts.since ?? "all time",
269
+ execution: {
270
+ totalTrades: stats.totalTrades,
271
+ successRate: stats.successRate,
272
+ volume: logVolume,
273
+ byExchange: stats.byExchange,
274
+ byType: stats.byType,
275
+ },
276
+ portfolio: {
277
+ totalEquity,
278
+ unrealizedPnl: totalUPnl,
279
+ },
280
+ costs: {
281
+ totalFees,
282
+ totalFunding,
283
+ netFunding: totalFunding,
284
+ },
285
+ insights: {
286
+ uniqueSymbols: symbolSet.size,
287
+ mostActiveExchange: Object.entries(stats.byExchange).sort((a, b) => b[1] - a[1])[0]?.[0] ?? "none",
288
+ avgTradesPerDay: stats.totalTrades > 0
289
+ ? (stats.totalTrades / Math.max(1, (Date.now() - new Date(records[records.length - 1]?.timestamp ?? Date.now()).getTime()) / 86400000)).toFixed(1)
290
+ : "0",
291
+ },
292
+ };
293
+ if (isJson())
294
+ return printJson(jsonOk(report));
295
+ console.log(chalk.cyan.bold(`\n Performance Report ${opts.since ? `(${opts.since})` : ""}\n`));
296
+ console.log(chalk.white.bold(" Execution"));
297
+ console.log(` Trades: ${stats.totalTrades} (${stats.successRate.toFixed(0)}% success)`);
298
+ console.log(` Volume: $${formatUsd(logVolume)}`);
299
+ console.log(` Symbols Traded: ${symbolSet.size}`);
300
+ console.log(` Avg/Day: ${report.insights.avgTradesPerDay}`);
301
+ console.log(chalk.white.bold("\n Portfolio"));
302
+ console.log(` Total Equity: $${formatUsd(totalEquity)}`);
303
+ console.log(` Unrealized PnL: ${formatPnl(totalUPnl)}`);
304
+ console.log(chalk.white.bold("\n Costs & Income"));
305
+ console.log(` Trading Fees: ${chalk.red(`-$${formatUsd(totalFees)}`)}`);
306
+ console.log(` Funding Income: ${formatPnl(totalFunding)}`);
307
+ console.log(` Net: ${formatPnl(totalFunding - totalFees)}`);
308
+ console.log();
309
+ });
310
+ });
311
+ }
@@ -0,0 +1,3 @@
1
+ import { Command } from "commander";
2
+ import type { ExchangeAdapter } from "../../exchanges/interface.js";
3
+ export declare function registerArbManageCommands(program: Command, getAdapterForExchange: (exchange: string) => Promise<ExchangeAdapter>, isJson: () => boolean): void;