hedgequantx 2.4.2 → 2.4.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.
@@ -0,0 +1,599 @@
1
+ /**
2
+ * Rithmic Manual Protobuf Decoder
3
+ * Required because protobufjs can't handle field IDs > 100000
4
+ */
5
+
6
+ // Field IDs for LastTrade (template_id = 150)
7
+ // From last_trade.proto - field numbers are the proto field IDs
8
+ const LAST_TRADE_FIELDS = {
9
+ TEMPLATE_ID: 154467,
10
+ SYMBOL: 110100,
11
+ EXCHANGE: 110101,
12
+ TRADE_PRICE: 100006,
13
+ TRADE_SIZE: 100178,
14
+ AGGRESSOR: 112003, // TransactionType enum: 1=BUY, 2=SELL
15
+ SSBOE: 150100,
16
+ USECS: 150101,
17
+ };
18
+
19
+ // Debug: log ALL fields to find aggressor
20
+ let debugFieldsLogged = 0;
21
+ let debugAllFieldsLogged = 0;
22
+
23
+ // Field IDs for BestBidOffer (template_id = 151)
24
+ const BBO_FIELDS = {
25
+ TEMPLATE_ID: 154467,
26
+ SYMBOL: 110100,
27
+ EXCHANGE: 110101,
28
+ BID_PRICE: 100022,
29
+ BID_SIZE: 100030,
30
+ ASK_PRICE: 100025,
31
+ ASK_SIZE: 100031,
32
+ SSBOE: 150100,
33
+ USECS: 150101,
34
+ };
35
+
36
+ // Field IDs for AccountPnLPositionUpdate (template_id = 450)
37
+ const ACCOUNT_PNL_FIELDS = {
38
+ TEMPLATE_ID: 154467,
39
+ IS_SNAPSHOT: 110121,
40
+ FCM_ID: 154013,
41
+ IB_ID: 154014,
42
+ ACCOUNT_ID: 154008,
43
+ FILL_BUY_QTY: 154041,
44
+ FILL_SELL_QTY: 154042,
45
+ BUY_QTY: 154260,
46
+ SELL_QTY: 154261,
47
+ OPEN_POSITION_PNL: 156961,
48
+ OPEN_POSITION_QUANTITY: 156962,
49
+ CLOSED_POSITION_PNL: 156963,
50
+ CLOSED_POSITION_QUANTITY: 156964,
51
+ NET_QUANTITY: 156967,
52
+ MIN_ACCOUNT_BALANCE: 156968,
53
+ ACCOUNT_BALANCE: 156970,
54
+ CASH_ON_HAND: 156971,
55
+ MIN_MARGIN_BALANCE: 156976,
56
+ MARGIN_BALANCE: 156977,
57
+ AVAILABLE_BUYING_POWER: 157015,
58
+ USED_BUYING_POWER: 157014,
59
+ DAY_OPEN_PNL: 157954,
60
+ DAY_CLOSED_PNL: 157955,
61
+ DAY_PNL: 157956,
62
+ PERCENT_MAX_LOSS: 156965,
63
+ SSBOE: 150100,
64
+ USECS: 150101,
65
+ };
66
+
67
+ // Field IDs for InstrumentPnLPositionUpdate (template_id = 451)
68
+ const INSTRUMENT_PNL_FIELDS = {
69
+ TEMPLATE_ID: 154467,
70
+ IS_SNAPSHOT: 110121,
71
+ FCM_ID: 154013,
72
+ IB_ID: 154014,
73
+ ACCOUNT_ID: 154008,
74
+ SYMBOL: 110100,
75
+ EXCHANGE: 110101,
76
+ PRODUCT_CODE: 100749,
77
+ INSTRUMENT_TYPE: 110116,
78
+ FILL_BUY_QTY: 154041,
79
+ FILL_SELL_QTY: 154042,
80
+ BUY_QTY: 154260,
81
+ SELL_QTY: 154261,
82
+ AVG_OPEN_FILL_PRICE: 154434,
83
+ OPEN_POSITION_PNL: 156961,
84
+ OPEN_POSITION_QUANTITY: 156962,
85
+ CLOSED_POSITION_PNL: 156963,
86
+ CLOSED_POSITION_QUANTITY: 156964,
87
+ NET_QUANTITY: 156967,
88
+ DAY_OPEN_PNL: 157954,
89
+ DAY_CLOSED_PNL: 157955,
90
+ DAY_PNL: 157956,
91
+ SSBOE: 150100,
92
+ USECS: 150101,
93
+ };
94
+
95
+ /**
96
+ * Read a varint from buffer starting at offset
97
+ * Uses BigInt internally to handle large field IDs correctly
98
+ * Returns [value, newOffset]
99
+ */
100
+ function readVarint(buffer, offset) {
101
+ let result = BigInt(0);
102
+ let shift = BigInt(0);
103
+ let pos = offset;
104
+
105
+ while (pos < buffer.length) {
106
+ const byte = buffer[pos++];
107
+ result |= BigInt(byte & 0x7f) << shift;
108
+ if ((byte & 0x80) === 0) {
109
+ return [Number(result), pos];
110
+ }
111
+ shift += BigInt(7);
112
+ if (shift > BigInt(63)) {
113
+ throw new Error('Varint too large');
114
+ }
115
+ }
116
+
117
+ throw new Error('Incomplete varint');
118
+ }
119
+
120
+ /**
121
+ * Read a length-delimited field (string/bytes)
122
+ * Returns [string, newOffset]
123
+ */
124
+ function readLengthDelimited(buffer, offset) {
125
+ const [length, newOffset] = readVarint(buffer, offset);
126
+ const value = buffer.slice(newOffset, newOffset + length).toString('utf8');
127
+ return [value, newOffset + length];
128
+ }
129
+
130
+ /**
131
+ * Skip a field based on wire type
132
+ */
133
+ function skipField(buffer, offset, wireType) {
134
+ switch (wireType) {
135
+ case 0: // Varint
136
+ const [, newOffset] = readVarint(buffer, offset);
137
+ return newOffset;
138
+ case 1: // 64-bit fixed
139
+ return offset + 8;
140
+ case 2: // Length-delimited
141
+ const [length, lenOffset] = readVarint(buffer, offset);
142
+ return lenOffset + length;
143
+ case 5: // 32-bit fixed
144
+ return offset + 4;
145
+ default:
146
+ throw new Error(`Unknown wire type: ${wireType}`);
147
+ }
148
+ }
149
+
150
+ /**
151
+ * Get template ID from buffer (first field)
152
+ */
153
+ function getTemplateId(buffer) {
154
+ let offset = 0;
155
+ while (offset < buffer.length) {
156
+ try {
157
+ const [tag, newOffset] = readVarint(buffer, offset);
158
+ const fieldNumber = tag >>> 3;
159
+ const wireType = tag & 0x7;
160
+ offset = newOffset;
161
+
162
+ if (fieldNumber === 154467 && wireType === 0) {
163
+ const [templateId] = readVarint(buffer, offset);
164
+ return templateId;
165
+ }
166
+ offset = skipField(buffer, offset, wireType);
167
+ } catch {
168
+ return -1;
169
+ }
170
+ }
171
+ return -1;
172
+ }
173
+
174
+ /**
175
+ * Decode LastTrade message (template_id = 150)
176
+ */
177
+ function decodeLastTrade(buffer) {
178
+ const result = {};
179
+ let offset = 0;
180
+ const allFields = []; // Debug: track all fields
181
+
182
+ // Debug first few calls
183
+ if (debugAllFieldsLogged < 3) {
184
+ console.log(`[DECODER] decodeLastTrade called, buffer size: ${buffer.length}`);
185
+ }
186
+
187
+ while (offset < buffer.length) {
188
+ try {
189
+ const [tag, newOffset] = readVarint(buffer, offset);
190
+ const fieldNumber = tag >>> 3;
191
+ const wireType = tag & 0x7;
192
+ offset = newOffset;
193
+
194
+ // Debug: log ALL fields for first few messages
195
+ if (debugAllFieldsLogged < 2) {
196
+ allFields.push({ fieldNumber, wireType });
197
+ }
198
+
199
+ switch (fieldNumber) {
200
+ case LAST_TRADE_FIELDS.TEMPLATE_ID:
201
+ [result.templateId, offset] = readVarint(buffer, offset);
202
+ break;
203
+ case LAST_TRADE_FIELDS.SYMBOL:
204
+ if (wireType === 2) {
205
+ [result.symbol, offset] = readLengthDelimited(buffer, offset);
206
+ } else {
207
+ offset = skipField(buffer, offset, wireType);
208
+ }
209
+ break;
210
+ case LAST_TRADE_FIELDS.EXCHANGE:
211
+ if (wireType === 2) {
212
+ [result.exchange, offset] = readLengthDelimited(buffer, offset);
213
+ } else {
214
+ offset = skipField(buffer, offset, wireType);
215
+ }
216
+ break;
217
+ case LAST_TRADE_FIELDS.TRADE_PRICE:
218
+ if (wireType === 1) {
219
+ // 64-bit double little-endian
220
+ result.tradePrice = buffer.readDoubleLE(offset);
221
+ offset += 8;
222
+ } else {
223
+ offset = skipField(buffer, offset, wireType);
224
+ }
225
+ break;
226
+ case LAST_TRADE_FIELDS.TRADE_SIZE:
227
+ if (wireType === 0) {
228
+ [result.tradeSize, offset] = readVarint(buffer, offset);
229
+ } else {
230
+ offset = skipField(buffer, offset, wireType);
231
+ }
232
+ break;
233
+ case LAST_TRADE_FIELDS.AGGRESSOR:
234
+ if (wireType === 0) {
235
+ [result.aggressor, offset] = readVarint(buffer, offset);
236
+ // Debug: log when aggressor is found
237
+ if (debugAllFieldsLogged < 5) {
238
+ console.log(`[DECODER] AGGRESSOR FOUND! value=${result.aggressor}`);
239
+ }
240
+ } else {
241
+ offset = skipField(buffer, offset, wireType);
242
+ }
243
+ break;
244
+ case LAST_TRADE_FIELDS.SSBOE:
245
+ if (wireType === 0) {
246
+ [result.ssboe, offset] = readVarint(buffer, offset);
247
+ } else {
248
+ offset = skipField(buffer, offset, wireType);
249
+ }
250
+ break;
251
+ case LAST_TRADE_FIELDS.USECS:
252
+ if (wireType === 0) {
253
+ [result.usecs, offset] = readVarint(buffer, offset);
254
+ } else {
255
+ offset = skipField(buffer, offset, wireType);
256
+ }
257
+ break;
258
+ default:
259
+ offset = skipField(buffer, offset, wireType);
260
+ }
261
+ } catch {
262
+ break;
263
+ }
264
+ }
265
+
266
+ // Debug: log all fields found in first messages
267
+ if (debugAllFieldsLogged < 2 && result.tradePrice > 0) {
268
+ console.log(`[DECODER:LastTrade] ALL FIELDS: ${JSON.stringify(allFields)}`);
269
+ console.log(`[DECODER:LastTrade] Result: price=${result.tradePrice}, aggressor=${result.aggressor}`);
270
+ debugAllFieldsLogged++;
271
+ }
272
+
273
+ return result;
274
+ }
275
+
276
+ /**
277
+ * Decode BestBidOffer message (template_id = 151)
278
+ */
279
+ function decodeBestBidOffer(buffer) {
280
+ const result = {};
281
+ let offset = 0;
282
+
283
+ while (offset < buffer.length) {
284
+ try {
285
+ const [tag, newOffset] = readVarint(buffer, offset);
286
+ const fieldNumber = tag >>> 3;
287
+ const wireType = tag & 0x7;
288
+ offset = newOffset;
289
+
290
+ switch (fieldNumber) {
291
+ case BBO_FIELDS.TEMPLATE_ID:
292
+ [result.templateId, offset] = readVarint(buffer, offset);
293
+ break;
294
+ case BBO_FIELDS.SYMBOL:
295
+ if (wireType === 2) {
296
+ [result.symbol, offset] = readLengthDelimited(buffer, offset);
297
+ } else {
298
+ offset = skipField(buffer, offset, wireType);
299
+ }
300
+ break;
301
+ case BBO_FIELDS.EXCHANGE:
302
+ if (wireType === 2) {
303
+ [result.exchange, offset] = readLengthDelimited(buffer, offset);
304
+ } else {
305
+ offset = skipField(buffer, offset, wireType);
306
+ }
307
+ break;
308
+ case BBO_FIELDS.BID_PRICE:
309
+ if (wireType === 1) {
310
+ result.bidPrice = buffer.readDoubleLE(offset);
311
+ offset += 8;
312
+ } else {
313
+ offset = skipField(buffer, offset, wireType);
314
+ }
315
+ break;
316
+ case BBO_FIELDS.BID_SIZE:
317
+ if (wireType === 0) {
318
+ [result.bidSize, offset] = readVarint(buffer, offset);
319
+ } else {
320
+ offset = skipField(buffer, offset, wireType);
321
+ }
322
+ break;
323
+ case BBO_FIELDS.ASK_PRICE:
324
+ if (wireType === 1) {
325
+ result.askPrice = buffer.readDoubleLE(offset);
326
+ offset += 8;
327
+ } else {
328
+ offset = skipField(buffer, offset, wireType);
329
+ }
330
+ break;
331
+ case BBO_FIELDS.ASK_SIZE:
332
+ if (wireType === 0) {
333
+ [result.askSize, offset] = readVarint(buffer, offset);
334
+ } else {
335
+ offset = skipField(buffer, offset, wireType);
336
+ }
337
+ break;
338
+ case BBO_FIELDS.SSBOE:
339
+ if (wireType === 0) {
340
+ [result.ssboe, offset] = readVarint(buffer, offset);
341
+ } else {
342
+ offset = skipField(buffer, offset, wireType);
343
+ }
344
+ break;
345
+ case BBO_FIELDS.USECS:
346
+ if (wireType === 0) {
347
+ [result.usecs, offset] = readVarint(buffer, offset);
348
+ } else {
349
+ offset = skipField(buffer, offset, wireType);
350
+ }
351
+ break;
352
+ default:
353
+ offset = skipField(buffer, offset, wireType);
354
+ }
355
+ } catch {
356
+ break;
357
+ }
358
+ }
359
+
360
+ return result;
361
+ }
362
+
363
+ /**
364
+ * Decode AccountPnLPositionUpdate message (template_id = 450)
365
+ */
366
+ function decodeAccountPnL(buffer) {
367
+ const result = {};
368
+ let offset = 0;
369
+
370
+ while (offset < buffer.length) {
371
+ try {
372
+ const [tag, newOffset] = readVarint(buffer, offset);
373
+ const fieldNumber = tag >>> 3;
374
+ const wireType = tag & 0x7;
375
+ offset = newOffset;
376
+
377
+ switch (fieldNumber) {
378
+ case ACCOUNT_PNL_FIELDS.TEMPLATE_ID:
379
+ [result.templateId, offset] = readVarint(buffer, offset);
380
+ break;
381
+ case ACCOUNT_PNL_FIELDS.IS_SNAPSHOT:
382
+ const [isSnap, snapOffset] = readVarint(buffer, offset);
383
+ result.isSnapshot = isSnap !== 0;
384
+ offset = snapOffset;
385
+ break;
386
+ case ACCOUNT_PNL_FIELDS.FCM_ID:
387
+ [result.fcmId, offset] = readLengthDelimited(buffer, offset);
388
+ break;
389
+ case ACCOUNT_PNL_FIELDS.IB_ID:
390
+ [result.ibId, offset] = readLengthDelimited(buffer, offset);
391
+ break;
392
+ case ACCOUNT_PNL_FIELDS.ACCOUNT_ID:
393
+ [result.accountId, offset] = readLengthDelimited(buffer, offset);
394
+ break;
395
+ case ACCOUNT_PNL_FIELDS.FILL_BUY_QTY:
396
+ [result.fillBuyQty, offset] = readVarint(buffer, offset);
397
+ break;
398
+ case ACCOUNT_PNL_FIELDS.FILL_SELL_QTY:
399
+ [result.fillSellQty, offset] = readVarint(buffer, offset);
400
+ break;
401
+ case ACCOUNT_PNL_FIELDS.BUY_QTY:
402
+ [result.buyQty, offset] = readVarint(buffer, offset);
403
+ break;
404
+ case ACCOUNT_PNL_FIELDS.SELL_QTY:
405
+ [result.sellQty, offset] = readVarint(buffer, offset);
406
+ break;
407
+ case ACCOUNT_PNL_FIELDS.ACCOUNT_BALANCE:
408
+ [result.accountBalance, offset] = readLengthDelimited(buffer, offset);
409
+ break;
410
+ case ACCOUNT_PNL_FIELDS.CASH_ON_HAND:
411
+ [result.cashOnHand, offset] = readLengthDelimited(buffer, offset);
412
+ break;
413
+ case ACCOUNT_PNL_FIELDS.MARGIN_BALANCE:
414
+ [result.marginBalance, offset] = readLengthDelimited(buffer, offset);
415
+ break;
416
+ case ACCOUNT_PNL_FIELDS.MIN_MARGIN_BALANCE:
417
+ [result.minMarginBalance, offset] = readLengthDelimited(buffer, offset);
418
+ break;
419
+ case ACCOUNT_PNL_FIELDS.AVAILABLE_BUYING_POWER:
420
+ [result.availableBuyingPower, offset] = readLengthDelimited(buffer, offset);
421
+ break;
422
+ case ACCOUNT_PNL_FIELDS.USED_BUYING_POWER:
423
+ [result.usedBuyingPower, offset] = readLengthDelimited(buffer, offset);
424
+ break;
425
+ case ACCOUNT_PNL_FIELDS.OPEN_POSITION_PNL:
426
+ [result.openPositionPnl, offset] = readLengthDelimited(buffer, offset);
427
+ break;
428
+ case ACCOUNT_PNL_FIELDS.OPEN_POSITION_QUANTITY:
429
+ [result.openPositionQuantity, offset] = readVarint(buffer, offset);
430
+ break;
431
+ case ACCOUNT_PNL_FIELDS.CLOSED_POSITION_PNL:
432
+ [result.closedPositionPnl, offset] = readLengthDelimited(buffer, offset);
433
+ break;
434
+ case ACCOUNT_PNL_FIELDS.CLOSED_POSITION_QUANTITY:
435
+ [result.closedPositionQuantity, offset] = readVarint(buffer, offset);
436
+ break;
437
+ case ACCOUNT_PNL_FIELDS.NET_QUANTITY:
438
+ [result.netQuantity, offset] = readVarint(buffer, offset);
439
+ break;
440
+ case ACCOUNT_PNL_FIELDS.DAY_OPEN_PNL:
441
+ [result.dayOpenPnl, offset] = readLengthDelimited(buffer, offset);
442
+ break;
443
+ case ACCOUNT_PNL_FIELDS.DAY_CLOSED_PNL:
444
+ [result.dayClosedPnl, offset] = readLengthDelimited(buffer, offset);
445
+ break;
446
+ case ACCOUNT_PNL_FIELDS.DAY_PNL:
447
+ [result.dayPnl, offset] = readLengthDelimited(buffer, offset);
448
+ break;
449
+ case ACCOUNT_PNL_FIELDS.PERCENT_MAX_LOSS:
450
+ [result.percentMaxLoss, offset] = readLengthDelimited(buffer, offset);
451
+ break;
452
+ case ACCOUNT_PNL_FIELDS.SSBOE:
453
+ [result.ssboe, offset] = readVarint(buffer, offset);
454
+ break;
455
+ case ACCOUNT_PNL_FIELDS.USECS:
456
+ [result.usecs, offset] = readVarint(buffer, offset);
457
+ break;
458
+ default:
459
+ offset = skipField(buffer, offset, wireType);
460
+ }
461
+ } catch {
462
+ break;
463
+ }
464
+ }
465
+
466
+ return result;
467
+ }
468
+
469
+ /**
470
+ * Decode InstrumentPnLPositionUpdate message (template_id = 451)
471
+ */
472
+ function decodeInstrumentPnL(buffer) {
473
+ const result = {};
474
+ let offset = 0;
475
+
476
+ while (offset < buffer.length) {
477
+ try {
478
+ const [tag, newOffset] = readVarint(buffer, offset);
479
+ const fieldNumber = tag >>> 3;
480
+ const wireType = tag & 0x7;
481
+ offset = newOffset;
482
+
483
+ switch (fieldNumber) {
484
+ case INSTRUMENT_PNL_FIELDS.TEMPLATE_ID:
485
+ [result.templateId, offset] = readVarint(buffer, offset);
486
+ break;
487
+ case INSTRUMENT_PNL_FIELDS.IS_SNAPSHOT:
488
+ const [isSnap, snapOffset] = readVarint(buffer, offset);
489
+ result.isSnapshot = isSnap !== 0;
490
+ offset = snapOffset;
491
+ break;
492
+ case INSTRUMENT_PNL_FIELDS.FCM_ID:
493
+ [result.fcmId, offset] = readLengthDelimited(buffer, offset);
494
+ break;
495
+ case INSTRUMENT_PNL_FIELDS.IB_ID:
496
+ [result.ibId, offset] = readLengthDelimited(buffer, offset);
497
+ break;
498
+ case INSTRUMENT_PNL_FIELDS.ACCOUNT_ID:
499
+ [result.accountId, offset] = readLengthDelimited(buffer, offset);
500
+ break;
501
+ case INSTRUMENT_PNL_FIELDS.SYMBOL:
502
+ [result.symbol, offset] = readLengthDelimited(buffer, offset);
503
+ break;
504
+ case INSTRUMENT_PNL_FIELDS.EXCHANGE:
505
+ [result.exchange, offset] = readLengthDelimited(buffer, offset);
506
+ break;
507
+ case INSTRUMENT_PNL_FIELDS.PRODUCT_CODE:
508
+ [result.productCode, offset] = readLengthDelimited(buffer, offset);
509
+ break;
510
+ case INSTRUMENT_PNL_FIELDS.INSTRUMENT_TYPE:
511
+ [result.instrumentType, offset] = readLengthDelimited(buffer, offset);
512
+ break;
513
+ case INSTRUMENT_PNL_FIELDS.FILL_BUY_QTY:
514
+ [result.fillBuyQty, offset] = readVarint(buffer, offset);
515
+ break;
516
+ case INSTRUMENT_PNL_FIELDS.FILL_SELL_QTY:
517
+ [result.fillSellQty, offset] = readVarint(buffer, offset);
518
+ break;
519
+ case INSTRUMENT_PNL_FIELDS.BUY_QTY:
520
+ [result.buyQty, offset] = readVarint(buffer, offset);
521
+ break;
522
+ case INSTRUMENT_PNL_FIELDS.SELL_QTY:
523
+ [result.sellQty, offset] = readVarint(buffer, offset);
524
+ break;
525
+ case INSTRUMENT_PNL_FIELDS.AVG_OPEN_FILL_PRICE:
526
+ if (wireType === 1) {
527
+ result.avgOpenFillPrice = buffer.readDoubleLE(offset);
528
+ offset += 8;
529
+ } else {
530
+ offset = skipField(buffer, offset, wireType);
531
+ }
532
+ break;
533
+ case INSTRUMENT_PNL_FIELDS.OPEN_POSITION_PNL:
534
+ [result.openPositionPnl, offset] = readLengthDelimited(buffer, offset);
535
+ break;
536
+ case INSTRUMENT_PNL_FIELDS.OPEN_POSITION_QUANTITY:
537
+ [result.openPositionQuantity, offset] = readVarint(buffer, offset);
538
+ break;
539
+ case INSTRUMENT_PNL_FIELDS.CLOSED_POSITION_PNL:
540
+ [result.closedPositionPnl, offset] = readLengthDelimited(buffer, offset);
541
+ break;
542
+ case INSTRUMENT_PNL_FIELDS.CLOSED_POSITION_QUANTITY:
543
+ [result.closedPositionQuantity, offset] = readVarint(buffer, offset);
544
+ break;
545
+ case INSTRUMENT_PNL_FIELDS.NET_QUANTITY:
546
+ [result.netQuantity, offset] = readVarint(buffer, offset);
547
+ break;
548
+ case INSTRUMENT_PNL_FIELDS.DAY_OPEN_PNL:
549
+ if (wireType === 1) {
550
+ result.dayOpenPnl = buffer.readDoubleLE(offset);
551
+ offset += 8;
552
+ } else {
553
+ offset = skipField(buffer, offset, wireType);
554
+ }
555
+ break;
556
+ case INSTRUMENT_PNL_FIELDS.DAY_CLOSED_PNL:
557
+ if (wireType === 1) {
558
+ result.dayClosedPnl = buffer.readDoubleLE(offset);
559
+ offset += 8;
560
+ } else {
561
+ offset = skipField(buffer, offset, wireType);
562
+ }
563
+ break;
564
+ case INSTRUMENT_PNL_FIELDS.DAY_PNL:
565
+ if (wireType === 1) {
566
+ result.dayPnl = buffer.readDoubleLE(offset);
567
+ offset += 8;
568
+ } else {
569
+ offset = skipField(buffer, offset, wireType);
570
+ }
571
+ break;
572
+ case INSTRUMENT_PNL_FIELDS.SSBOE:
573
+ [result.ssboe, offset] = readVarint(buffer, offset);
574
+ break;
575
+ case INSTRUMENT_PNL_FIELDS.USECS:
576
+ [result.usecs, offset] = readVarint(buffer, offset);
577
+ break;
578
+ default:
579
+ offset = skipField(buffer, offset, wireType);
580
+ }
581
+ } catch {
582
+ break;
583
+ }
584
+ }
585
+
586
+ return result;
587
+ }
588
+
589
+ module.exports = {
590
+ getTemplateId,
591
+ decodeLastTrade,
592
+ decodeBestBidOffer,
593
+ decodeAccountPnL,
594
+ decodeInstrumentPnL,
595
+ LAST_TRADE_FIELDS,
596
+ BBO_FIELDS,
597
+ ACCOUNT_PNL_FIELDS,
598
+ INSTRUMENT_PNL_FIELDS,
599
+ };
@@ -0,0 +1,61 @@
1
+ /**
2
+ * M1 Module
3
+ */
4
+ const EventEmitter = require('events');
5
+ const { v4: uuidv4 } = require('uuid');
6
+ const OS = { B: 0, A: 1 };
7
+ const SS = { W: 1, M: 2, S: 3, VS: 4, E: 5 };
8
+
9
+ class VC {
10
+ constructor() { this.d = new Map(); this.psv = new Map(); }
11
+ gk(c) { return `${c}_${new Date().toISOString().split('T')[0]}`; }
12
+ init(c) { const k = this.gk(c); if (!this.d.has(k)) { this.d.set(k, { v: 0, cv: 0, ctpv: 0, ub1: 0, ub2: 0, ub3: 0, lb1: 0, lb2: 0, lb3: 0, sd: 0, tc: 0, hod: -Infinity, lod: Infinity, lu: Date.now() }); this.psv.set(k, 0); } }
13
+ pt(c, h, l, cl, vol, ts = Date.now()) { const k = this.gk(c); let d = this.d.get(k); if (!d) { this.init(c); d = this.d.get(k); } const tp = (h + l + cl) / 3; d.cv += vol; d.ctpv += tp * vol; if (d.cv > 0) d.v = d.ctpv / d.cv; const p = this.psv.get(k) || 0; this.psv.set(k, p + tp * tp * vol); this.cs(k); if (h > d.hod) d.hod = h; if (l < d.lod) d.lod = l; d.tc++; d.lu = ts; }
14
+ cs(k) { const d = this.d.get(k); if (!d || d.cv === 0) return; const p = this.psv.get(k) || 0; const ms = p / d.cv; const sm = d.v * d.v; const va = ms - sm; d.sd = Math.sqrt(Math.max(0, va)); d.ub1 = d.v + d.sd; d.ub2 = d.v + 2 * d.sd; d.ub3 = d.v + 3 * d.sd; d.lb1 = d.v - d.sd; d.lb2 = d.v - 2 * d.sd; d.lb3 = d.v - 3 * d.sd; }
15
+ an(c, cp, ts = 0.25) { const k = this.gk(c); const d = this.d.get(k); if (!d || d.v === 0) return null; const pd = cp - d.v; const dt = pd / ts; const dv = d.sd > 0 ? pd / d.sd : 0; let pos; if (Math.abs(dv) < 0.1) pos = 'AT'; else if (dv > 0) pos = 'ABOVE'; else pos = 'BELOW'; const ad = Math.abs(dv); let bl; if (ad < 1) bl = 0; else if (ad < 2) bl = 1; else if (ad < 3) bl = 2; else bl = 3; let mrp; if (ad >= 3) mrp = 0.99; else if (ad >= 2) mrp = 0.95; else if (ad >= 1.5) mrp = 0.85; else if (ad >= 1) mrp = 0.68; else mrp = 0.50; return { currentPrice: cp, vwap: d.v, deviation: dv, deviationTicks: dt, position: pos, bandLevel: bl, target: d.v, probability: mrp, isFavorableForMeanReversion: ad >= 1.5 }; }
16
+ get(c) { return this.d.get(this.gk(c)) || null; }
17
+ reset(c) { const k = this.gk(c); this.d.delete(k); this.psv.delete(k); }
18
+ }
19
+
20
+ class S1 extends EventEmitter {
21
+ constructor(cfg = {}) { super(); this.ts = cfg.tickSize || 0.25; this.mrr = 1.2; this.ap = 14; this.ah = []; this.ba = 2.5; this.rt = []; this.ws = 0; this.ls = 0; this.lsd = null; this.bh = new Map(); this.vc = new VC(); this.lst = 0; this.cd = 30000; this.st = { s: 0, t: 0, w: 0, l: 0, p: 0 }; }
22
+ // tickSize for tick calculations, P&L comes from API
23
+ initialize(c, ts = 0.25) { this.ts = ts; this.bh.set(c, []); this.vc.init(c); }
24
+ processBar(c, b) { let bs = this.bh.get(c); if (!bs) { bs = []; this.bh.set(c, bs); } bs.push(b); if (bs.length > 500) bs.shift(); if (bs.length < 50) return null; const sd = new Date(b.timestamp).toISOString().split('T')[0]; if (this.lsd !== sd) this.lsd = sd; this.vc.pt(c, b.high, b.low, b.close, b.volume, b.timestamp); const va = this.vc.an(c, b.close, this.ts); if (!va) return null; const vs = this.sv(va, b.close); const of = this.aof(bs); const os = this.sof(of); const dd = this.adb(bs); const ds = this.sdd(dd); const md = this.am(bs); const ms = this.sm(md); const pd = this.avp(bs, c, b.close); const ps = this.svp(pd, b.close); const { dir, st, ofc } = this.dd(va, of, dd, pd, b.close); if (dir === 'none') return null; if (st !== 'vr') return null; if (!ofc) return null; const bc = vs * 0.25 + os * 0.30 + ds * 0.20 + ms * 0.10 + ps * 0.15; const cf = Math.min(1.0, bc + 0.05); const pm = this.gap(bs); if (cf < pm.ct) return null; if (Date.now() - this.lst < this.cd) return null; const ep = b.close; const spt = pm.spt; const tgt = pm.tgt; let sl, tp; if (dir === 'long') { sl = ep - spt * this.ts; tp = ep + tgt * this.ts; const vt = va.vwap; if (vt > ep && vt < tp) { const vr = (vt - ep) / this.ts; if (vr >= spt * 1.5) tp = vt; } } else { sl = ep + spt * this.ts; tp = ep - tgt * this.ts; const vt = va.vwap; if (vt < ep && vt > tp) { const vr = (ep - vt) / this.ts; if (vr >= spt * 1.5) tp = vt; } } const rt = Math.abs(ep - sl) / this.ts; const rwt = Math.abs(tp - ep) / this.ts; if (rwt / rt < this.mrr) return null; let p1, p2; if (dir === 'long') { p1 = ep + pm.p1t * this.ts; p2 = ep + pm.p2t * this.ts; } else { p1 = ep - pm.p1t * this.ts; p2 = ep - pm.p2t * this.ts; } let str = SS.M; if (cf >= 0.75) str = SS.S; else if (cf >= 0.85) str = SS.VS; else if (cf < 0.55) str = SS.W; const wp = 0.5 + (cf - 0.5) * 0.3; const aw = Math.abs(tp - ep); const al = Math.abs(ep - sl); const eg = wp * aw - (1 - wp) * al; this.lst = Date.now(); this.st.s++; const sig = { id: uuidv4(), timestamp: Date.now(), symbol: c.split('.')[0] || c, contractId: c, side: dir === 'long' ? OS.B : OS.A, direction: dir, strategy: 'S1', strength: str, edge: eg, confidence: cf, entryPrice: ep, stopLoss: sl, takeProfit: tp, riskReward: rwt / rt, stopTicks: spt, targetTicks: tgt, trailTriggerTicks: pm.ttt, trailDistanceTicks: pm.tdt, partial1Price: p1, partial2Price: p2, score1: vs, score2: os, domScore: ds, microstructureScore: ms, volumeProfileScore: ps, confirmed: ofc, zScore: va.deviation, regime: pm.rg, expires: Date.now() + 60000 }; this.emit('signal', sig); return sig; }
25
+ processTick(t) { const { contractId, price, bid, ask, volume, side, timestamp } = t; const b = { timestamp: timestamp || Date.now(), open: price, high: price, low: price, close: price, volume: volume || 1, trades: 1, delta: side === 'buy' ? (volume || 1) : -(volume || 1), vwap: price }; return this.processBar(contractId, b); }
26
+ onTick(t) { return this.processTick(t); }
27
+ recordTradeResult(p) { this.rt.push({ netPnl: p, timestamp: Date.now() }); if (this.rt.length > 100) this.rt.shift(); if (p > 0) { this.ws++; this.ls = 0; this.st.w++; } else { this.ls++; this.ws = 0; this.st.l++; } this.st.p += p; this.st.t++; }
28
+ sv(v, cp) { const z = Math.abs(v.deviation); if (z < 0.5) return 0.3; else if (z < 1.0) return 0.6; else if (z < 2.0) return 0.9; else if (z < 2.5) return 0.7; else return 0.4; }
29
+ sof(o) { let s = 0.5; s += o.ir * 0.3; if (o.dd) s += 0.2; if (o.ad) s += 0.15; return Math.min(1.0, s); }
30
+ sdd(d) { let s = 0.5; s += Math.abs(d.pi) * 0.25; if (d.sb || d.sa) s += 0.15; if (d.al !== null) s += 0.1; return Math.min(1.0, s); }
31
+ sm(m) { const ns = 1.0 - m.nr; const ls = 0.0001 < m.kl && m.kl < 0.001 ? 0.7 : 0.5; return ns * 0.6 + ls * 0.4; }
32
+ svp(p, cp) { let s = 0.5; const pd = Math.abs(cp - p.poc) / this.ts; if (pd < 10) s += 0.2; if (p.val <= cp && cp <= p.vah) s += 0.15; for (const h of p.hvn) { if (Math.abs(cp - h) < 5 * this.ts) { s += 0.15; break; } } return Math.min(1.0, s); }
33
+ aof(bs) { if (bs.length < 2) return { d: 0, cd: 0, dd: false, ad: false, ir: 0, as: 'neutral' }; const cb = bs[bs.length - 1]; const pb = bs[bs.length - 2]; const br = cb.high - cb.low; let bv, sv; if (br > 0) { const cp = (cb.close - cb.low) / br; bv = cb.volume * cp; sv = cb.volume * (1 - cp); } else { bv = cb.volume * 0.5; sv = cb.volume * 0.5; } const d = bv - sv; const cd = cb.delta || d; let dv = false; if (bs.length >= 5) { const rb = bs.slice(-5); const pc = rb[4].close - rb[0].close; const ds = rb.reduce((s, b) => s + (b.delta || 0), 0); if ((pc > 0 && ds < -cb.volume * 0.5) || (pc < 0 && ds > cb.volume * 0.5)) dv = true; } const lb = Math.min(20, bs.length); const rbs = bs.slice(-lb); const ar = rbs.reduce((s, b) => s + (b.high - b.low), 0) / lb; const av = rbs.reduce((s, b) => s + b.volume, 0) / lb; const ab = cb.volume > av * 1.5 && br < ar * 0.6; const tv = bv + sv; const im = tv > 0 ? Math.abs(bv - sv) / tv : 0; let ag; if (bv > sv * 1.3) ag = 'buy'; else if (sv > bv * 1.3) ag = 'sell'; else ag = 'neutral'; return { d, cd, dd: dv, ad: ab, ir: im, as: ag }; }
34
+ adb(bs) { const lb = Math.min(10, bs.length); if (lb < 2) return { bp: 0.5, ap: 0.5, pi: 0, sb: false, sa: false, al: null }; const rbs = bs.slice(-lb); let bss = [], ass = []; for (const b of rbs) { const br = b.high - b.low; if (br > 0) { const cp = (b.close - b.low) / br; bss.push(cp * b.volume); ass.push((1 - cp) * b.volume); } } const t = bss.reduce((a, b) => a + b, 0) + ass.reduce((a, b) => a + b, 0); const bp = t > 0 ? bss.reduce((a, b) => a + b, 0) / t : 0.5; const ap = t > 0 ? ass.reduce((a, b) => a + b, 0) / t : 0.5; const pi = bp - ap; const lf = rbs.slice(-5); let hc = 0, lc = 0; for (const b of lf) { const br = b.high - b.low; if (br > 0) { const cp = (b.close - b.low) / br; if (cp > 0.7) hc++; if (cp < 0.3) lc++; } } let al = null; const ar = rbs.reduce((s, b) => s + (b.high - b.low), 0) / lb; const av = rbs.reduce((s, b) => s + b.volume, 0) / lb; for (const b of rbs) { const br = b.high - b.low; if (b.volume > av * 1.5 && br < ar * 0.5) { al = (b.high + b.low) / 2; break; } } return { bp, ap, pi, sb: hc >= 3, sa: lc >= 3, al }; }
35
+ am(bs) { const lb = Math.min(50, bs.length); if (lb < 10) return { kl: 0, rs: 0, es: 0, rv: 0, nr: 0.5 }; const rbs = bs.slice(-lb); const pc = [], vs = []; for (let i = 1; i < rbs.length; i++) { pc.push(rbs[i].close - rbs[i - 1].close); vs.push(rbs[i].volume); } let kl = 0; if (pc.length > 5) { const mp = pc.reduce((a, b) => a + b, 0) / pc.length; const mv = vs.reduce((a, b) => a + b, 0) / vs.length; let cv = 0, vv = 0; for (let i = 0; i < pc.length; i++) { cv += (pc[i] - mp) * (vs[i] - mv); vv += Math.pow(vs[i] - mv, 2); } cv /= pc.length; vv /= pc.length; kl = vv > 0 ? Math.abs(cv / vv) : 0; } let rs = 0; if (pc.length > 2) { const c1 = pc.slice(1), c0 = pc.slice(0, -1); const m1 = c1.reduce((a, b) => a + b, 0) / c1.length; const m0 = c0.reduce((a, b) => a + b, 0) / c0.length; let ac = 0; for (let i = 0; i < c1.length; i++) ac += (c1[i] - m1) * (c0[i] - m0); ac /= c1.length; rs = Math.sqrt(Math.max(0, -ac)) * 2; } const es = rbs.reduce((s, b) => s + (b.high - b.low), 0) / lb; const rt = []; for (let i = 1; i < rbs.length; i++) { if (rbs[i - 1].close > 0) rt.push(Math.log(rbs[i].close / rbs[i - 1].close)); } let rv = 0; if (rt.length > 0) { const mr = rt.reduce((a, b) => a + b, 0) / rt.length; const va = rt.reduce((s, r) => s + Math.pow(r - mr, 2), 0) / rt.length; rv = Math.sqrt(va) * Math.sqrt(252 * 390); } const sv = pc.length >= 10 ? Math.sqrt(pc.slice(-10).reduce((s, p) => s + p * p, 0) / 10) : 0; const lv = pc.length > 0 ? Math.sqrt(pc.reduce((s, p) => s + p * p, 0) / pc.length) : 1; const nr = lv > 0 ? Math.min(1, sv / lv) : 0.5; return { kl, rs, es, rv, nr }; }
36
+ avp(bs, c, cp) { const lb = Math.min(100, bs.length); if (lb < 10) return { poc: cp, vah: cp, val: cp, hvn: [], lvn: [] }; const rbs = bs.slice(-lb); const vap = new Map(); for (const b of rbs) { const br = b.high - b.low; const nl = Math.max(1, Math.round(br / this.ts)); const vpl = b.volume / nl; let p = b.low; while (p <= b.high) { const r = Math.round(p / this.ts) * this.ts; vap.set(r, (vap.get(r) || 0) + vpl); p += this.ts; } } if (vap.size === 0) return { poc: cp, vah: cp, val: cp, hvn: [], lvn: [] }; let poc = cp, mv = 0; for (const [p, v] of vap) { if (v > mv) { mv = v; poc = p; } } const tv = Array.from(vap.values()).reduce((a, b) => a + b, 0); const tgv = tv * 0.7; const sp = Array.from(vap.keys()).sort((a, b) => a - b); const pi = sp.indexOf(poc); let vv = vap.get(poc) || 0, li = pi, hi = pi; while (vv < tgv && (li > 0 || hi < sp.length - 1)) { const lv = li > 0 ? (vap.get(sp[li - 1]) || 0) : 0; const hv = hi < sp.length - 1 ? (vap.get(sp[hi + 1]) || 0) : 0; if (lv >= hv && li > 0) { li--; vv += vap.get(sp[li]) || 0; } else if (hi < sp.length - 1) { hi++; vv += vap.get(sp[hi]) || 0; } else break; } const val = sp[li], vah = sp[hi]; const av = tv / vap.size; const hvn = [], lvn = []; for (const [p, v] of vap) { if (v > av * 1.5) hvn.push(p); if (v < av * 0.5) lvn.push(p); } return { poc, vah, val, hvn, lvn }; }
37
+ dd(v, o, d, p, cp) { let ls = 0, ss = 0, st = 'mixed', ofc = false; if (v.deviation < -0.8) { ls += 2; st = 'vr'; if (o.as === 'buy' || o.dd) ofc = true; } else if (v.deviation > 0.8) { ss += 2; st = 'vr'; if (o.as === 'sell' || o.dd) ofc = true; } if (o.as === 'buy' && o.ir > 0.35) ls += 1; else if (o.as === 'sell' && o.ir > 0.35) ss += 1; if (o.dd) { if (o.cd < 0) { ls += 2; st = 'dv'; ofc = true; } else { ss += 2; st = 'dv'; ofc = true; } } if (d.pi > 0.35) ls += 1; else if (d.pi < -0.35) ss += 1; if (d.sb) ls += 1; if (d.sa) ss += 1; if (cp < p.poc - 5 * this.ts) ls += 1; else if (cp > p.poc + 5 * this.ts) ss += 1; if (ls >= 2 && ls > ss) return { dir: 'long', st, ofc }; else if (ss >= 2 && ss > ls) return { dir: 'short', st, ofc }; else return { dir: 'none', st: 'none', ofc: false }; }
38
+ catr(bs) { if (bs.length < this.ap + 1) return this.ba; const tv = []; for (let i = bs.length - this.ap; i < bs.length; i++) { const b = bs[i]; const pc = bs[i - 1].close; const tr = Math.max(b.high - b.low, Math.abs(b.high - pc), Math.abs(b.low - pc)); tv.push(tr); } const a = tv.reduce((a, b) => a + b, 0) / tv.length; this.ah.push(a); if (this.ah.length > 500) this.ah.shift(); return a; }
39
+ gap(bs) { const a = this.catr(bs); const at = a / this.ts; let ap = 0.5; if (this.ah.length >= 20) ap = this.ah.filter(x => x <= a).length / this.ah.length; let rg, vm; if (ap < 0.25) { rg = 'low'; vm = 0.9; } else if (ap < 0.70) { rg = 'normal'; vm = 1.0; } else if (ap < 0.90) { rg = 'high'; vm = 1.15; } else { rg = 'extreme'; vm = 1.25; } const pm = this.gpm(); const bs6 = 6; let spt = Math.round(bs6 * vm); spt = Math.max(5, Math.min(8, spt)); const bt8 = 8; let tgt = Math.round(bt8 * vm); tgt = Math.max(6, Math.min(10, tgt)); if (tgt / spt < 1.2) tgt = Math.round(spt * 1.33); const ttt = Math.max(3, Math.min(5, Math.round(tgt * 0.5))); const tdt = Math.max(2, Math.min(3, Math.round(spt * 0.35))); const bc = 0.45; const ra = { low: 0.05, normal: 0.0, high: 0.03, extreme: 0.10 }; let ct = bc + ra[rg]; ct = Math.min(0.65, ct * pm); const p1t = Math.round(tgt * 0.5); const p2t = Math.round(tgt * 0.75); return { spt, tgt, ct, ttt, tdt, p1t, p2t, rg, at, vm }; }
40
+ gpm() { if (this.rt.length === 0) return 1.0; const r = this.rt.slice(-20); const w = r.filter(t => t.netPnl > 0).length; const wr = w / r.length; let m = 1.0; if (this.ls >= 2) { m += 0.05 * this.ls; m = Math.min(m, 1.3); } if (r.length >= 5 && wr < 0.40) m *= 1.1; if (this.ws >= 4 && wr > 0.50) m *= 0.95; return Math.max(0.9, Math.min(1.3, m)); }
41
+ getBarHistory(c) { return this.bh.get(c) || []; }
42
+ reset(c) { this.bh.set(c, []); this.lsd = null; this.vc.reset(c); }
43
+ getStats() { return this.st; }
44
+ getAnalysisState(c, cp) { const bs = this.bh.get(c) || []; if (bs.length < 10) return { ready: false, message: 'Collecting...' }; const va = this.vc.an(c, cp, this.ts); const of = this.aof(bs); const pm = this.gap(bs); return { ready: true, vwap: va?.vwap || 0, zScore: va?.deviation || 0, orderFlow: of.as, ratio: of.ir, regime: pm.rg, stopTicks: pm.spt, targetTicks: pm.tgt, barsProcessed: bs.length }; }
45
+ }
46
+
47
+ class M1 extends EventEmitter {
48
+ constructor(cfg = {}) { super(); this.cfg = cfg; this.s = new S1(cfg); this.s.on('signal', (sig) => { this.emit('signal', { side: sig.direction === 'long' ? 'buy' : 'sell', action: 'open', reason: `Z=${sig.zScore.toFixed(2)}, cf=${(sig.confidence * 100).toFixed(0)}%`, ...sig }); }); }
49
+ processTick(t) { return this.s.processTick(t); }
50
+ onTick(t) { return this.processTick(t); }
51
+ processBar(c, b) { return this.s.processBar(c, b); }
52
+ onTrade(t) { this.s.processTick({ contractId: t.contractId || t.symbol, price: t.price, volume: t.size || t.volume || 1, side: t.side, timestamp: t.timestamp || Date.now() }); }
53
+ initialize(c, ts, tv) { this.s.initialize(c, ts, tv); }
54
+ getAnalysisState(c, p) { return this.s.getAnalysisState(c, p); }
55
+ recordTradeResult(p) { this.s.recordTradeResult(p); }
56
+ reset(c) { this.s.reset(c); this.emit('log', { type: 'info', message: 'Reset' }); }
57
+ getStats() { return this.s.getStats(); }
58
+ generateSignal(p) { return null; }
59
+ }
60
+
61
+ module.exports = { S1, M1, VC, OS, SS };