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,253 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
2
+ import { existsSync, unlinkSync, readFileSync, writeFileSync, renameSync } from "fs";
3
+ import { resolve } from "path";
4
+ import { logExecution, readExecutionLog } from "../execution-log.js";
5
+ const PERP_DIR = resolve(process.env.HOME || "~", ".perp");
6
+ const LOG_FILE = resolve(PERP_DIR, "executions.jsonl");
7
+ const BACKUP_FILE = resolve(PERP_DIR, "executions.jsonl.arb-manage-backup");
8
+ function detectArbPairs(positions) {
9
+ const bySymbol = new Map();
10
+ for (const p of positions) {
11
+ const key = p.symbol.toUpperCase();
12
+ if (!bySymbol.has(key))
13
+ bySymbol.set(key, []);
14
+ bySymbol.get(key).push(p);
15
+ }
16
+ const pairs = [];
17
+ for (const [symbol, pos] of bySymbol) {
18
+ const longs = pos.filter(p => p.side === "long");
19
+ const shorts = pos.filter(p => p.side === "short");
20
+ for (const l of longs) {
21
+ for (const s of shorts) {
22
+ if (l.exchange !== s.exchange) {
23
+ pairs.push({ symbol, longExchange: l.exchange, shortExchange: s.exchange, longPos: l, shortPos: s });
24
+ }
25
+ }
26
+ }
27
+ }
28
+ return pairs;
29
+ }
30
+ // ── Tests ──
31
+ describe("Arb pair detection", () => {
32
+ it("matches positions on different exchanges with opposite sides", () => {
33
+ const positions = [
34
+ { exchange: "hyperliquid", symbol: "BTC", side: "long", size: 0.1, entryPrice: 100000, markPrice: 101000, unrealizedPnl: 100 },
35
+ { exchange: "lighter", symbol: "BTC", side: "short", size: 0.1, entryPrice: 100500, markPrice: 101000, unrealizedPnl: -50 },
36
+ ];
37
+ const pairs = detectArbPairs(positions);
38
+ expect(pairs).toHaveLength(1);
39
+ expect(pairs[0].symbol).toBe("BTC");
40
+ expect(pairs[0].longExchange).toBe("hyperliquid");
41
+ expect(pairs[0].shortExchange).toBe("lighter");
42
+ });
43
+ it("does not match positions on the same exchange", () => {
44
+ const positions = [
45
+ { exchange: "hyperliquid", symbol: "ETH", side: "long", size: 1, entryPrice: 3000, markPrice: 3100, unrealizedPnl: 100 },
46
+ { exchange: "hyperliquid", symbol: "ETH", side: "short", size: 1, entryPrice: 3050, markPrice: 3100, unrealizedPnl: -50 },
47
+ ];
48
+ const pairs = detectArbPairs(positions);
49
+ expect(pairs).toHaveLength(0);
50
+ });
51
+ it("does not match positions with same side", () => {
52
+ const positions = [
53
+ { exchange: "hyperliquid", symbol: "SOL", side: "long", size: 10, entryPrice: 150, markPrice: 155, unrealizedPnl: 50 },
54
+ { exchange: "lighter", symbol: "SOL", side: "long", size: 10, entryPrice: 151, markPrice: 155, unrealizedPnl: 40 },
55
+ ];
56
+ const pairs = detectArbPairs(positions);
57
+ expect(pairs).toHaveLength(0);
58
+ });
59
+ it("returns empty for no positions", () => {
60
+ const pairs = detectArbPairs([]);
61
+ expect(pairs).toHaveLength(0);
62
+ });
63
+ it("matches multiple arb pairs for different symbols", () => {
64
+ const positions = [
65
+ { exchange: "hyperliquid", symbol: "BTC", side: "long", size: 0.1, entryPrice: 100000, markPrice: 101000, unrealizedPnl: 100 },
66
+ { exchange: "lighter", symbol: "BTC", side: "short", size: 0.1, entryPrice: 100500, markPrice: 101000, unrealizedPnl: -50 },
67
+ { exchange: "pacifica", symbol: "ETH", side: "long", size: 1, entryPrice: 3000, markPrice: 3100, unrealizedPnl: 100 },
68
+ { exchange: "hyperliquid", symbol: "ETH", side: "short", size: 1, entryPrice: 3050, markPrice: 3100, unrealizedPnl: -50 },
69
+ ];
70
+ const pairs = detectArbPairs(positions);
71
+ expect(pairs).toHaveLength(2);
72
+ const symbols = pairs.map(p => p.symbol).sort();
73
+ expect(symbols).toEqual(["BTC", "ETH"]);
74
+ });
75
+ it("handles case-insensitive symbol matching", () => {
76
+ const positions = [
77
+ { exchange: "hyperliquid", symbol: "btc", side: "long", size: 0.1, entryPrice: 100000, markPrice: 101000, unrealizedPnl: 100 },
78
+ { exchange: "lighter", symbol: "BTC", side: "short", size: 0.1, entryPrice: 100500, markPrice: 101000, unrealizedPnl: -50 },
79
+ ];
80
+ const pairs = detectArbPairs(positions);
81
+ expect(pairs).toHaveLength(1);
82
+ });
83
+ });
84
+ describe("Arb close dry-run", () => {
85
+ it("dry-run flag prevents execution", () => {
86
+ // Simulating dry-run behavior: the flag should be passed through
87
+ const dryRun = true;
88
+ const executed = [];
89
+ // In dry-run mode, no orders should be placed
90
+ if (!dryRun) {
91
+ executed.push("sell");
92
+ executed.push("buy");
93
+ }
94
+ expect(executed).toHaveLength(0);
95
+ });
96
+ it("produces correct actions for close", () => {
97
+ const longPos = { exchange: "hyperliquid", symbol: "BTC", rawSymbol: "BTC", side: "long", size: 0.1 };
98
+ const shortPos = { exchange: "lighter", symbol: "BTC", rawSymbol: "BTC-PERP", side: "short", size: 0.1 };
99
+ const actions = [
100
+ { exchange: longPos.exchange, action: "sell", symbol: longPos.rawSymbol, size: String(longPos.size) },
101
+ { exchange: shortPos.exchange, action: "buy", symbol: shortPos.rawSymbol, size: String(shortPos.size) },
102
+ ];
103
+ expect(actions).toHaveLength(2);
104
+ expect(actions[0].action).toBe("sell"); // close long = sell
105
+ expect(actions[1].action).toBe("buy"); // close short = buy
106
+ expect(actions[0].exchange).toBe("hyperliquid");
107
+ expect(actions[1].exchange).toBe("lighter");
108
+ });
109
+ });
110
+ describe("Arb history grouping and stats", () => {
111
+ beforeEach(() => {
112
+ if (existsSync(LOG_FILE)) {
113
+ const content = readFileSync(LOG_FILE, "utf-8");
114
+ writeFileSync(BACKUP_FILE, content);
115
+ }
116
+ if (existsSync(LOG_FILE))
117
+ unlinkSync(LOG_FILE);
118
+ });
119
+ afterEach(() => {
120
+ if (existsSync(LOG_FILE))
121
+ unlinkSync(LOG_FILE);
122
+ if (existsSync(BACKUP_FILE)) {
123
+ renameSync(BACKUP_FILE, LOG_FILE);
124
+ }
125
+ });
126
+ it("groups arb entries and closes by symbol", () => {
127
+ logExecution({
128
+ type: "arb_entry", exchange: "hyperliquid+lighter",
129
+ symbol: "BTC", side: "entry", size: "0.1",
130
+ status: "success", dryRun: false,
131
+ meta: { longExchange: "hyperliquid", shortExchange: "lighter", spread: 35, markPrice: 100000 },
132
+ });
133
+ logExecution({
134
+ type: "arb_close", exchange: "hyperliquid+lighter",
135
+ symbol: "BTC", side: "close", size: "0.1",
136
+ status: "success", dryRun: false,
137
+ meta: { longExchange: "hyperliquid", shortExchange: "lighter", currentSpread: 3, unrealizedPnl: 5.5, netPnl: 4.2 },
138
+ });
139
+ const entries = readExecutionLog({ type: "arb_entry" });
140
+ const closes = readExecutionLog({ type: "arb_close" });
141
+ expect(entries).toHaveLength(1);
142
+ expect(closes).toHaveLength(1);
143
+ expect(entries[0].symbol).toBe("BTC");
144
+ expect(closes[0].symbol).toBe("BTC");
145
+ expect(entries[0].meta?.spread).toBe(35);
146
+ expect(closes[0].meta?.netPnl).toBe(4.2);
147
+ });
148
+ it("calculates trade stats from multiple entries", () => {
149
+ // Trade 1: winner
150
+ logExecution({
151
+ type: "arb_entry", exchange: "hyperliquid+lighter",
152
+ symbol: "BTC", side: "entry", size: "0.1",
153
+ status: "success", dryRun: false,
154
+ meta: { spread: 40, markPrice: 100000 },
155
+ });
156
+ logExecution({
157
+ type: "arb_close", exchange: "hyperliquid+lighter",
158
+ symbol: "BTC", side: "close", size: "0.1",
159
+ status: "success", dryRun: false,
160
+ meta: { unrealizedPnl: 10, netPnl: 8 },
161
+ });
162
+ // Trade 2: loser
163
+ logExecution({
164
+ type: "arb_entry", exchange: "pacifica+hyperliquid",
165
+ symbol: "ETH", side: "entry", size: "1.0",
166
+ status: "success", dryRun: false,
167
+ meta: { spread: 25, markPrice: 3000 },
168
+ });
169
+ logExecution({
170
+ type: "arb_close", exchange: "pacifica+hyperliquid",
171
+ symbol: "ETH", side: "close", size: "1.0",
172
+ status: "success", dryRun: false,
173
+ meta: { unrealizedPnl: -5, netPnl: -7 },
174
+ });
175
+ const allEntries = readExecutionLog()
176
+ .filter(r => r.type === "arb_entry" || r.type === "arb_close");
177
+ const entries = allEntries.filter(r => r.type === "arb_entry");
178
+ const closes = allEntries.filter(r => r.type === "arb_close");
179
+ expect(entries).toHaveLength(2);
180
+ expect(closes).toHaveLength(2);
181
+ // Verify we can compute stats
182
+ const completedNetPnls = closes
183
+ .filter(c => c.status === "success" && c.meta?.netPnl !== undefined)
184
+ .map(c => Number(c.meta.netPnl));
185
+ expect(completedNetPnls).toHaveLength(2);
186
+ const totalNetPnl = completedNetPnls.reduce((s, n) => s + n, 0);
187
+ expect(totalNetPnl).toBe(1); // 8 + (-7) = 1
188
+ const winners = completedNetPnls.filter(n => n > 0);
189
+ expect(winners).toHaveLength(1);
190
+ const winRate = (winners.length / completedNetPnls.length) * 100;
191
+ expect(winRate).toBe(50);
192
+ });
193
+ it("handles entries without matching closes (open trades)", () => {
194
+ logExecution({
195
+ type: "arb_entry", exchange: "hyperliquid+lighter",
196
+ symbol: "SOL", side: "entry", size: "10",
197
+ status: "success", dryRun: false,
198
+ meta: { spread: 30, markPrice: 150 },
199
+ });
200
+ const entries = readExecutionLog({ type: "arb_entry" });
201
+ const closes = readExecutionLog({ type: "arb_close" });
202
+ expect(entries).toHaveLength(1);
203
+ expect(closes).toHaveLength(0);
204
+ // Should be treated as an open trade
205
+ const symbol = entries[0].symbol;
206
+ const matchingClose = closes.find(c => c.symbol === symbol);
207
+ expect(matchingClose).toBeUndefined();
208
+ });
209
+ it("filters by period correctly", () => {
210
+ // Log an entry with a very old timestamp manually
211
+ const oldDate = new Date(Date.now() - 60 * 24 * 60 * 60 * 1000).toISOString();
212
+ const recentDate = new Date().toISOString();
213
+ logExecution({
214
+ type: "arb_entry", exchange: "hyperliquid+lighter",
215
+ symbol: "OLD_TRADE", side: "entry", size: "0.1",
216
+ status: "success", dryRun: false,
217
+ });
218
+ logExecution({
219
+ type: "arb_entry", exchange: "hyperliquid+lighter",
220
+ symbol: "NEW_TRADE", side: "entry", size: "0.1",
221
+ status: "success", dryRun: false,
222
+ });
223
+ // Read with 30-day filter: both are recent (logged just now), so both should appear
224
+ const since30d = new Date(Date.now() - 30 * 24 * 60 * 60 * 1000).toISOString();
225
+ const recent = readExecutionLog({ type: "arb_entry", since: since30d });
226
+ expect(recent).toHaveLength(2);
227
+ });
228
+ });
229
+ describe("Duration formatting", () => {
230
+ function formatDuration(ms) {
231
+ const hours = Math.floor(ms / (1000 * 60 * 60));
232
+ const days = Math.floor(hours / 24);
233
+ const remainingHours = hours % 24;
234
+ if (days > 0)
235
+ return `${days}d ${remainingHours}h`;
236
+ if (hours > 0)
237
+ return `${hours}h`;
238
+ const minutes = Math.floor(ms / (1000 * 60));
239
+ return `${minutes}m`;
240
+ }
241
+ it("formats minutes", () => {
242
+ expect(formatDuration(30 * 60 * 1000)).toBe("30m");
243
+ });
244
+ it("formats hours", () => {
245
+ expect(formatDuration(5 * 60 * 60 * 1000)).toBe("5h");
246
+ });
247
+ it("formats days and hours", () => {
248
+ expect(formatDuration(26 * 60 * 60 * 1000)).toBe("1d 2h");
249
+ });
250
+ it("formats zero", () => {
251
+ expect(formatDuration(0)).toBe("0m");
252
+ });
253
+ });
@@ -0,0 +1 @@
1
+ export {};