hedgequantx 2.6.163 → 2.7.1

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 (146) hide show
  1. package/README.md +15 -88
  2. package/bin/cli.js +0 -11
  3. package/dist/lib/api.jsc +0 -0
  4. package/dist/lib/api2.jsc +0 -0
  5. package/dist/lib/core.jsc +0 -0
  6. package/dist/lib/core2.jsc +0 -0
  7. package/dist/lib/data.js +1 -1
  8. package/dist/lib/data.jsc +0 -0
  9. package/dist/lib/data2.jsc +0 -0
  10. package/dist/lib/decoder.jsc +0 -0
  11. package/dist/lib/m/mod1.jsc +0 -0
  12. package/dist/lib/m/mod2.jsc +0 -0
  13. package/dist/lib/n/r1.jsc +0 -0
  14. package/dist/lib/n/r2.jsc +0 -0
  15. package/dist/lib/n/r3.jsc +0 -0
  16. package/dist/lib/n/r4.jsc +0 -0
  17. package/dist/lib/n/r5.jsc +0 -0
  18. package/dist/lib/n/r6.jsc +0 -0
  19. package/dist/lib/n/r7.jsc +0 -0
  20. package/dist/lib/o/util1.jsc +0 -0
  21. package/dist/lib/o/util2.jsc +0 -0
  22. package/package.json +8 -5
  23. package/src/app.js +40 -162
  24. package/src/config/constants.js +31 -33
  25. package/src/config/propfirms.js +13 -217
  26. package/src/config/settings.js +0 -43
  27. package/src/lib/api.js +198 -0
  28. package/src/lib/api2.js +353 -0
  29. package/src/lib/core.js +539 -0
  30. package/src/lib/core2.js +341 -0
  31. package/src/lib/data.js +555 -0
  32. package/src/lib/data2.js +492 -0
  33. package/src/lib/decoder.js +599 -0
  34. package/src/lib/m/s1.js +804 -0
  35. package/src/lib/m/s2.js +34 -0
  36. package/src/lib/n/r1.js +454 -0
  37. package/src/lib/n/r2.js +514 -0
  38. package/src/lib/n/r3.js +631 -0
  39. package/src/lib/n/r4.js +401 -0
  40. package/src/lib/n/r5.js +335 -0
  41. package/src/lib/n/r6.js +425 -0
  42. package/src/lib/n/r7.js +530 -0
  43. package/src/lib/o/l1.js +44 -0
  44. package/src/lib/o/l2.js +427 -0
  45. package/src/lib/python-bridge.js +206 -0
  46. package/src/menus/connect.js +14 -176
  47. package/src/menus/dashboard.js +65 -110
  48. package/src/pages/accounts.js +18 -18
  49. package/src/pages/algo/copy-trading.js +210 -240
  50. package/src/pages/algo/index.js +41 -104
  51. package/src/pages/algo/one-account.js +386 -33
  52. package/src/pages/algo/ui.js +312 -151
  53. package/src/pages/orders.js +3 -3
  54. package/src/pages/positions.js +3 -3
  55. package/src/pages/stats/chart.js +74 -0
  56. package/src/pages/stats/display.js +228 -0
  57. package/src/pages/stats/index.js +236 -0
  58. package/src/pages/stats/metrics.js +213 -0
  59. package/src/pages/user.js +6 -6
  60. package/src/services/hqx-server/constants.js +55 -0
  61. package/src/services/hqx-server/index.js +401 -0
  62. package/src/services/hqx-server/latency.js +81 -0
  63. package/src/services/index.js +12 -3
  64. package/src/services/rithmic/accounts.js +7 -32
  65. package/src/services/rithmic/connection.js +1 -204
  66. package/src/services/rithmic/contracts.js +116 -99
  67. package/src/services/rithmic/handlers.js +21 -196
  68. package/src/services/rithmic/index.js +63 -120
  69. package/src/services/rithmic/market.js +31 -0
  70. package/src/services/rithmic/orders.js +5 -111
  71. package/src/services/rithmic/protobuf.js +384 -138
  72. package/src/services/session.js +22 -173
  73. package/src/ui/box.js +10 -18
  74. package/src/ui/index.js +1 -3
  75. package/src/ui/menu.js +1 -1
  76. package/src/utils/prompts.js +2 -2
  77. package/dist/lib/m/s1.js +0 -1
  78. package/src/menus/ai-agent-connect.js +0 -181
  79. package/src/menus/ai-agent-models.js +0 -219
  80. package/src/menus/ai-agent-oauth.js +0 -292
  81. package/src/menus/ai-agent-ui.js +0 -141
  82. package/src/menus/ai-agent.js +0 -484
  83. package/src/pages/algo/algo-config.js +0 -195
  84. package/src/pages/algo/algo-multi.js +0 -801
  85. package/src/pages/algo/algo-utils.js +0 -58
  86. package/src/pages/algo/copy-engine.js +0 -449
  87. package/src/pages/algo/custom-strategy.js +0 -459
  88. package/src/pages/algo/logger.js +0 -245
  89. package/src/pages/algo/smart-logs-data.js +0 -218
  90. package/src/pages/algo/smart-logs.js +0 -387
  91. package/src/pages/algo/ui-constants.js +0 -144
  92. package/src/pages/algo/ui-summary.js +0 -184
  93. package/src/pages/stats-calculations.js +0 -191
  94. package/src/pages/stats-ui.js +0 -381
  95. package/src/pages/stats.js +0 -339
  96. package/src/services/ai/client-analysis.js +0 -194
  97. package/src/services/ai/client-models.js +0 -333
  98. package/src/services/ai/client.js +0 -343
  99. package/src/services/ai/index.js +0 -384
  100. package/src/services/ai/oauth-anthropic.js +0 -265
  101. package/src/services/ai/oauth-gemini.js +0 -223
  102. package/src/services/ai/oauth-iflow.js +0 -269
  103. package/src/services/ai/oauth-openai.js +0 -233
  104. package/src/services/ai/oauth-qwen.js +0 -279
  105. package/src/services/ai/providers/direct-providers.js +0 -323
  106. package/src/services/ai/providers/index.js +0 -62
  107. package/src/services/ai/providers/other-providers.js +0 -104
  108. package/src/services/ai/proxy-install.js +0 -249
  109. package/src/services/ai/proxy-manager.js +0 -494
  110. package/src/services/ai/proxy-remote.js +0 -161
  111. package/src/services/ai/strategy-supervisor.js +0 -1312
  112. package/src/services/ai/supervisor-data.js +0 -195
  113. package/src/services/ai/supervisor-optimize.js +0 -215
  114. package/src/services/ai/supervisor-sync.js +0 -178
  115. package/src/services/ai/supervisor-utils.js +0 -158
  116. package/src/services/ai/supervisor.js +0 -484
  117. package/src/services/ai/validation.js +0 -250
  118. package/src/services/hqx-server-events.js +0 -110
  119. package/src/services/hqx-server-handlers.js +0 -217
  120. package/src/services/hqx-server-latency.js +0 -136
  121. package/src/services/hqx-server.js +0 -403
  122. package/src/services/position-constants.js +0 -28
  123. package/src/services/position-exit-logic.js +0 -174
  124. package/src/services/position-manager.js +0 -438
  125. package/src/services/position-momentum.js +0 -206
  126. package/src/services/projectx/accounts.js +0 -142
  127. package/src/services/projectx/index.js +0 -443
  128. package/src/services/projectx/market.js +0 -172
  129. package/src/services/projectx/stats.js +0 -110
  130. package/src/services/projectx/trading.js +0 -180
  131. package/src/services/rithmic/latency-tracker.js +0 -182
  132. package/src/services/rithmic/market-data-decoders.js +0 -229
  133. package/src/services/rithmic/market-data.js +0 -272
  134. package/src/services/rithmic/orders-fast.js +0 -246
  135. package/src/services/rithmic/proto-decoders.js +0 -403
  136. package/src/services/rithmic/specs.js +0 -146
  137. package/src/services/rithmic/trade-history.js +0 -254
  138. package/src/services/session-history.js +0 -475
  139. package/src/services/strategy/hft-signal-calc.js +0 -147
  140. package/src/services/strategy/hft-tick.js +0 -407
  141. package/src/services/strategy/recovery-math.js +0 -402
  142. package/src/services/tradovate/constants.js +0 -109
  143. package/src/services/tradovate/index.js +0 -392
  144. package/src/services/tradovate/market.js +0 -47
  145. package/src/services/tradovate/orders.js +0 -145
  146. package/src/services/tradovate/websocket.js +0 -97
@@ -1,146 +0,0 @@
1
- /**
2
- * @fileoverview Rithmic Specifications and Configurations
3
- * CME Contract Specifications and PropFirm configurations
4
- *
5
- * NO FAKE DATA - These are official exchange constants
6
- */
7
-
8
- const { RITHMIC_ENDPOINTS, RITHMIC_SYSTEMS } = require('./constants');
9
-
10
- /**
11
- * CME Contract Specifications - Official exchange tick sizes, values, and names
12
- * These are technical constants defined by the exchange, not market data.
13
- * Source: CME Group contract specifications
14
- */
15
- const CME_CONTRACT_SPECS = {
16
- // E-mini Index Futures (CME)
17
- ES: { tickSize: 0.25, tickValue: 12.50, name: 'E-mini S&P 500' },
18
- NQ: { tickSize: 0.25, tickValue: 5.00, name: 'E-mini NASDAQ-100' },
19
- RTY: { tickSize: 0.10, tickValue: 5.00, name: 'E-mini Russell 2000' },
20
- YM: { tickSize: 1.00, tickValue: 5.00, name: 'E-mini Dow' },
21
-
22
- // Micro Index Futures (CME)
23
- MES: { tickSize: 0.25, tickValue: 1.25, name: 'Micro E-mini S&P 500' },
24
- MNQ: { tickSize: 0.25, tickValue: 0.50, name: 'Micro E-mini NASDAQ-100' },
25
- M2K: { tickSize: 0.10, tickValue: 0.50, name: 'Micro E-mini Russell 2000' },
26
- MYM: { tickSize: 1.00, tickValue: 0.50, name: 'Micro E-mini Dow' },
27
-
28
- // Energy Futures (NYMEX)
29
- CL: { tickSize: 0.01, tickValue: 10.00, name: 'Crude Oil' },
30
- QM: { tickSize: 0.025, tickValue: 12.50, name: 'E-mini Crude Oil' },
31
- MCL: { tickSize: 0.01, tickValue: 1.00, name: 'Micro Crude Oil' },
32
- NG: { tickSize: 0.001, tickValue: 10.00, name: 'Natural Gas' },
33
- QG: { tickSize: 0.005, tickValue: 12.50, name: 'E-mini Natural Gas' },
34
-
35
- // Metal Futures (COMEX)
36
- GC: { tickSize: 0.10, tickValue: 10.00, name: 'Gold' },
37
- MGC: { tickSize: 0.10, tickValue: 1.00, name: 'Micro Gold' },
38
- SI: { tickSize: 0.005, tickValue: 25.00, name: 'Silver' },
39
- SIL: { tickSize: 0.005, tickValue: 2.50, name: '1000oz Silver' },
40
- HG: { tickSize: 0.0005, tickValue: 12.50, name: 'Copper' },
41
- MHG: { tickSize: 0.0005, tickValue: 1.25, name: 'Micro Copper' },
42
-
43
- // Treasury Futures (CBOT)
44
- ZB: { tickSize: 0.03125, tickValue: 31.25, name: '30-Year T-Bond' },
45
- ZN: { tickSize: 0.015625, tickValue: 15.625, name: '10-Year T-Note' },
46
- ZF: { tickSize: 0.0078125, tickValue: 7.8125, name: '5-Year T-Note' },
47
- ZT: { tickSize: 0.0078125, tickValue: 15.625, name: '2-Year T-Note' },
48
-
49
- // Agricultural Futures (CBOT)
50
- ZC: { tickSize: 0.25, tickValue: 12.50, name: 'Corn' },
51
- ZS: { tickSize: 0.25, tickValue: 12.50, name: 'Soybeans' },
52
- ZW: { tickSize: 0.25, tickValue: 12.50, name: 'Wheat' },
53
- ZL: { tickSize: 0.01, tickValue: 6.00, name: 'Soybean Oil' },
54
- ZM: { tickSize: 0.10, tickValue: 10.00, name: 'Soybean Meal' },
55
-
56
- // Currency Futures (CME)
57
- '6E': { tickSize: 0.00005, tickValue: 6.25, name: 'Euro FX' },
58
- '6J': { tickSize: 0.0000005, tickValue: 6.25, name: 'Japanese Yen' },
59
- '6B': { tickSize: 0.0001, tickValue: 6.25, name: 'British Pound' },
60
- '6A': { tickSize: 0.0001, tickValue: 10.00, name: 'Australian Dollar' },
61
- '6C': { tickSize: 0.00005, tickValue: 5.00, name: 'Canadian Dollar' },
62
- '6M': { tickSize: 0.0001, tickValue: 5.00, name: 'Mexican Peso' },
63
-
64
- // Nikkei (CME)
65
- NKD: { tickSize: 5.0, tickValue: 25.00, name: 'Nikkei 225' },
66
-
67
- // VIX Futures (CFE)
68
- VX: { tickSize: 0.05, tickValue: 50.00, name: 'VIX Futures' },
69
- };
70
-
71
- /**
72
- * PropFirm configurations
73
- */
74
- const PROPFIRM_CONFIGS = {
75
- apex: { name: 'Apex Trader Funding', systemName: 'Apex', gateway: RITHMIC_ENDPOINTS.CHICAGO },
76
- apex_rithmic: { name: 'Apex Trader Funding', systemName: 'Apex', gateway: RITHMIC_ENDPOINTS.CHICAGO },
77
- topstep_r: { name: 'Topstep (Rithmic)', systemName: RITHMIC_SYSTEMS.TOPSTEP, gateway: RITHMIC_ENDPOINTS.CHICAGO },
78
- bulenox_r: { name: 'Bulenox (Rithmic)', systemName: RITHMIC_SYSTEMS.BULENOX, gateway: RITHMIC_ENDPOINTS.CHICAGO },
79
- earn2trade: { name: 'Earn2Trade', systemName: RITHMIC_SYSTEMS.EARN_2_TRADE, gateway: RITHMIC_ENDPOINTS.CHICAGO },
80
- mescapital: { name: 'MES Capital', systemName: RITHMIC_SYSTEMS.MES_CAPITAL, gateway: RITHMIC_ENDPOINTS.CHICAGO },
81
- tradefundrr: { name: 'TradeFundrr', systemName: RITHMIC_SYSTEMS.TRADEFUNDRR, gateway: RITHMIC_ENDPOINTS.CHICAGO },
82
- thetradingpit: { name: 'The Trading Pit', systemName: RITHMIC_SYSTEMS.THE_TRADING_PIT, gateway: RITHMIC_ENDPOINTS.CHICAGO },
83
- fundedfutures: { name: 'Funded Futures Network', systemName: RITHMIC_SYSTEMS.FUNDED_FUTURES_NETWORK, gateway: RITHMIC_ENDPOINTS.CHICAGO },
84
- propshop: { name: 'PropShop Trader', systemName: RITHMIC_SYSTEMS.PROPSHOP_TRADER, gateway: RITHMIC_ENDPOINTS.CHICAGO },
85
- '4proptrader': { name: '4PropTrader', systemName: RITHMIC_SYSTEMS.FOUR_PROP_TRADER, gateway: RITHMIC_ENDPOINTS.CHICAGO },
86
- daytraders: { name: 'DayTraders.com', systemName: RITHMIC_SYSTEMS.DAY_TRADERS, gateway: RITHMIC_ENDPOINTS.CHICAGO },
87
- '10xfutures': { name: '10X Futures', systemName: RITHMIC_SYSTEMS.TEN_X_FUTURES, gateway: RITHMIC_ENDPOINTS.CHICAGO },
88
- lucidtrading: { name: 'Lucid Trading', systemName: RITHMIC_SYSTEMS.LUCID_TRADING, gateway: RITHMIC_ENDPOINTS.CHICAGO },
89
- thrivetrading: { name: 'Thrive Trading', systemName: RITHMIC_SYSTEMS.THRIVE_TRADING, gateway: RITHMIC_ENDPOINTS.CHICAGO },
90
- legendstrading: { name: 'Legends Trading', systemName: RITHMIC_SYSTEMS.LEGENDS_TRADING, gateway: RITHMIC_ENDPOINTS.CHICAGO },
91
- };
92
-
93
- /**
94
- * Get tick multiplier for P&L calculation
95
- * @param {string} symbol - Trading symbol
96
- * @returns {number} Multiplier for P&L calculation
97
- */
98
- const getTickMultiplier = (symbol) => {
99
- const sym = (symbol || '').toUpperCase();
100
- if (sym.startsWith('ES')) return 50; // E-mini S&P 500: $50 per point
101
- if (sym.startsWith('NQ')) return 20; // E-mini Nasdaq: $20 per point
102
- if (sym.startsWith('YM')) return 5; // E-mini Dow: $5 per point
103
- if (sym.startsWith('RTY')) return 50; // E-mini Russell: $50 per point
104
- if (sym.startsWith('MES')) return 5; // Micro E-mini S&P: $5 per point
105
- if (sym.startsWith('MNQ')) return 2; // Micro E-mini Nasdaq: $2 per point
106
- if (sym.startsWith('GC')) return 100; // Gold: $100 per point
107
- if (sym.startsWith('SI')) return 5000; // Silver: $5000 per point
108
- if (sym.startsWith('CL')) return 1000; // Crude Oil: $1000 per point
109
- if (sym.startsWith('NG')) return 10000; // Natural Gas: $10000 per point
110
- if (sym.startsWith('ZB') || sym.startsWith('ZN')) return 1000; // Bonds
111
- if (sym.startsWith('6E')) return 125000; // Euro FX
112
- if (sym.startsWith('6J')) return 12500000; // Japanese Yen
113
- return 1; // Default
114
- };
115
-
116
- /**
117
- * Check market hours
118
- * @returns {{isOpen: boolean, message: string}}
119
- */
120
- const checkMarketHours = () => {
121
- const now = new Date();
122
- const utcDay = now.getUTCDay();
123
- const utcHour = now.getUTCHours();
124
-
125
- const isDST = now.getTimezoneOffset() < Math.max(
126
- new Date(now.getFullYear(), 0, 1).getTimezoneOffset(),
127
- new Date(now.getFullYear(), 6, 1).getTimezoneOffset()
128
- );
129
- const ctOffset = isDST ? 5 : 6;
130
- const ctHour = (utcHour - ctOffset + 24) % 24;
131
- const ctDay = utcHour < ctOffset ? (utcDay + 6) % 7 : utcDay;
132
-
133
- if (ctDay === 6) return { isOpen: false, message: 'Market closed (Saturday)' };
134
- if (ctDay === 0 && ctHour < 17) return { isOpen: false, message: 'Market opens Sunday 5:00 PM CT' };
135
- if (ctDay === 5 && ctHour >= 16) return { isOpen: false, message: 'Market closed (Friday after 4PM CT)' };
136
- if (ctHour === 16 && ctDay >= 1 && ctDay <= 4) return { isOpen: false, message: 'Daily maintenance (4:00-5:00 PM CT)' };
137
-
138
- return { isOpen: true, message: 'Market is open' };
139
- };
140
-
141
- module.exports = {
142
- CME_CONTRACT_SPECS,
143
- PROPFIRM_CONFIGS,
144
- getTickMultiplier,
145
- checkMarketHours,
146
- };
@@ -1,254 +0,0 @@
1
- /**
2
- * @fileoverview Rithmic Trade History
3
- * Handles trade history fetching and P&L calculation
4
- *
5
- * NO FAKE DATA - Only real values from Rithmic API
6
- */
7
-
8
- const { logger } = require('../../utils/logger');
9
- const { getTickMultiplier } = require('./specs');
10
-
11
- const log = logger.scope('RithmicTradeHistory');
12
-
13
- /**
14
- * Get trade history from Rithmic API
15
- * Uses RequestShowOrderHistory to fetch historical fills
16
- * @param {Object} service - RithmicService instance
17
- * @param {string} accountId - Optional account filter
18
- * @param {number} days - Number of days to fetch (default: 30)
19
- * @returns {Promise<{success: boolean, trades: Array}>}
20
- */
21
- const getTradeHistory = async (service, accountId, days = 30) => {
22
- if (!service.orderConn || !service.loginInfo) {
23
- return { success: true, trades: service.completedTrades || [] };
24
- }
25
-
26
- return new Promise((resolve) => {
27
- const historyOrders = [];
28
- let resolved = false;
29
-
30
- // Timeout after 5 seconds
31
- const timeout = setTimeout(() => {
32
- if (!resolved) {
33
- resolved = true;
34
- cleanup();
35
- // Combine API history with session trades
36
- const allTrades = [...processHistoryToTrades(historyOrders, service), ...service.completedTrades];
37
- resolve({ success: true, trades: allTrades });
38
- }
39
- }, 5000);
40
-
41
- // Listen for order history snapshots
42
- const onOrderNotification = (data) => {
43
- try {
44
- if (data.isSnapshot || data.status === 'complete' || data.status === 'Complete') {
45
- const order = {
46
- orderId: data.orderId || data.orderTag,
47
- accountId: data.accountId,
48
- symbol: data.symbol,
49
- exchange: data.exchange,
50
- side: data.transactionType === 1 ? 0 : 1,
51
- quantity: data.quantity || data.totalFillQuantity || 0,
52
- fillPrice: data.avgFillPrice || data.lastFillPrice || 0,
53
- timestamp: data.ssboe ? data.ssboe * 1000 : Date.now(),
54
- status: data.status,
55
- };
56
-
57
- if (order.quantity > 0 && order.fillPrice > 0) {
58
- historyOrders.push(order);
59
- }
60
- }
61
- } catch (e) {
62
- // Ignore parse errors
63
- }
64
- };
65
-
66
- const onHistoryComplete = () => {
67
- if (!resolved) {
68
- resolved = true;
69
- cleanup();
70
- const allTrades = [...processHistoryToTrades(historyOrders, service), ...service.completedTrades];
71
- resolve({ success: true, trades: allTrades });
72
- }
73
- };
74
-
75
- const cleanup = () => {
76
- clearTimeout(timeout);
77
- service.removeListener('orderNotification', onOrderNotification);
78
- service.removeListener('orderHistoryComplete', onHistoryComplete);
79
- };
80
-
81
- service.on('orderNotification', onOrderNotification);
82
- service.on('orderHistoryComplete', onHistoryComplete);
83
-
84
- // Request order history for each account
85
- try {
86
- const accounts = accountId
87
- ? service.accounts.filter(a => a.accountId === accountId)
88
- : service.accounts;
89
-
90
- for (const acc of accounts) {
91
- service.orderConn.send('RequestShowOrderHistory', {
92
- templateId: 324,
93
- userMsg: ['HQX-HISTORY'],
94
- fcmId: acc.fcmId || service.loginInfo.fcmId,
95
- ibId: acc.ibId || service.loginInfo.ibId,
96
- accountId: acc.accountId,
97
- });
98
- }
99
- } catch (e) {
100
- if (!resolved) {
101
- resolved = true;
102
- cleanup();
103
- resolve({ success: false, error: e.message, trades: service.completedTrades || [] });
104
- }
105
- }
106
- });
107
- };
108
-
109
- /**
110
- * Process historical orders into trades with P&L
111
- * Matches entries and exits to calculate P&L
112
- * @param {Array} orders - Historical orders
113
- * @param {Object} service - RithmicService instance (for _getTickMultiplier)
114
- * @returns {Array} Processed trades
115
- */
116
- const processHistoryToTrades = (orders, service) => {
117
- const trades = [];
118
- const openPositions = new Map();
119
-
120
- // Sort by timestamp (oldest first)
121
- const sorted = [...orders].sort((a, b) => a.timestamp - b.timestamp);
122
-
123
- for (const order of sorted) {
124
- const key = `${order.accountId}:${order.symbol}`;
125
- const open = openPositions.get(key);
126
-
127
- if (open && open.side !== order.side) {
128
- // Closing trade - calculate P&L
129
- const closeQty = Math.min(order.quantity, open.quantity);
130
- let pnl;
131
-
132
- if (open.side === 0) {
133
- pnl = (order.fillPrice - open.price) * closeQty;
134
- } else {
135
- pnl = (open.price - order.fillPrice) * closeQty;
136
- }
137
-
138
- const tickMultiplier = getTickMultiplier(order.symbol);
139
- pnl = pnl * tickMultiplier;
140
-
141
- trades.push({
142
- id: order.orderId,
143
- accountId: order.accountId,
144
- symbol: order.symbol,
145
- exchange: order.exchange,
146
- side: open.side,
147
- size: closeQty,
148
- entryPrice: open.price,
149
- exitPrice: order.fillPrice,
150
- price: order.fillPrice,
151
- timestamp: order.timestamp,
152
- creationTimestamp: new Date(order.timestamp).toISOString(),
153
- status: 'CLOSED',
154
- profitAndLoss: pnl,
155
- pnl: pnl,
156
- fees: 0,
157
- });
158
-
159
- const remaining = open.quantity - closeQty;
160
- if (remaining <= 0) {
161
- openPositions.delete(key);
162
- } else {
163
- open.quantity = remaining;
164
- }
165
- } else if (open && open.side === order.side) {
166
- // Adding to position
167
- const totalQty = open.quantity + order.quantity;
168
- open.price = ((open.price * open.quantity) + (order.fillPrice * order.quantity)) / totalQty;
169
- open.quantity = totalQty;
170
- } else {
171
- // Opening new position
172
- openPositions.set(key, {
173
- side: order.side,
174
- quantity: order.quantity,
175
- price: order.fillPrice,
176
- timestamp: order.timestamp,
177
- });
178
- }
179
- }
180
-
181
- return trades;
182
- };
183
-
184
- /**
185
- * Setup order fill listener for real-time P&L tracking
186
- * @param {Object} service - RithmicService instance
187
- */
188
- const setupOrderFillListener = (service) => {
189
- service._openEntries = new Map();
190
-
191
- service.on('orderFilled', (fillInfo) => {
192
- const key = `${fillInfo.accountId}:${fillInfo.symbol}`;
193
- const side = fillInfo.transactionType === 1 ? 0 : 1;
194
- const qty = fillInfo.fillQuantity || fillInfo.totalFillQuantity || 0;
195
- const price = fillInfo.avgFillPrice || fillInfo.lastFillPrice || 0;
196
-
197
- const openEntry = service._openEntries.get(key);
198
- let pnl = null;
199
-
200
- if (openEntry && openEntry.side !== side) {
201
- // Closing trade
202
- const closeQty = Math.min(qty, openEntry.qty);
203
-
204
- if (openEntry.side === 0) {
205
- pnl = (price - openEntry.price) * closeQty;
206
- } else {
207
- pnl = (openEntry.price - price) * closeQty;
208
- }
209
-
210
- const remainingQty = openEntry.qty - closeQty;
211
- if (remainingQty <= 0) {
212
- service._openEntries.delete(key);
213
- } else {
214
- openEntry.qty = remainingQty;
215
- }
216
-
217
- service.completedTrades.push({
218
- id: fillInfo.orderId || fillInfo.orderTag,
219
- orderTag: fillInfo.orderTag,
220
- accountId: fillInfo.accountId,
221
- symbol: fillInfo.symbol,
222
- exchange: fillInfo.exchange,
223
- side: openEntry.side,
224
- size: closeQty,
225
- entryPrice: openEntry.price,
226
- exitPrice: price,
227
- price: price,
228
- timestamp: fillInfo.localTimestamp || Date.now(),
229
- creationTimestamp: new Date().toISOString(),
230
- status: 'CLOSED',
231
- profitAndLoss: pnl,
232
- pnl: pnl,
233
- fees: 0,
234
- });
235
- log.debug('Trade closed', { symbol: fillInfo.symbol, pnl, trades: service.completedTrades.length });
236
- } else {
237
- // Opening or adding to position
238
- if (openEntry && openEntry.side === side) {
239
- const totalQty = openEntry.qty + qty;
240
- openEntry.price = ((openEntry.price * openEntry.qty) + (price * qty)) / totalQty;
241
- openEntry.qty = totalQty;
242
- } else {
243
- service._openEntries.set(key, { side, qty, price, timestamp: Date.now() });
244
- }
245
- log.debug('Position opened/added', { symbol: fillInfo.symbol, side, qty, price });
246
- }
247
- });
248
- };
249
-
250
- module.exports = {
251
- getTradeHistory,
252
- processHistoryToTrades,
253
- setupOrderFillListener,
254
- };