fixparser-plugin-mcp 9.1.7-63c797c5 → 9.1.7-67ba45e9

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.
@@ -129,7 +129,7 @@ var toolSchemas = {
129
129
  }
130
130
  },
131
131
  executeOrder: {
132
- description: "Executes a verified order. verifyOrder must be called before executeOrder. user has to explicitly allow executeOrder.",
132
+ description: "Executes a verified order. verifyOrder must be called before executeOrder.",
133
133
  schema: {
134
134
  type: "object",
135
135
  properties: {
@@ -282,10 +282,63 @@ import QuickChart from "quickchart-js";
282
282
  var createMarketDataRequestHandler = (parser, pendingRequests) => {
283
283
  return async (args) => {
284
284
  try {
285
+ parser.logger.log({
286
+ level: "info",
287
+ message: `Sending market data request for symbols: ${args.symbols.join(", ")}`
288
+ });
285
289
  const response = new Promise((resolve) => {
286
290
  pendingRequests.set(args.mdReqID, resolve);
291
+ parser.logger.log({
292
+ level: "info",
293
+ message: `Registered callback for market data request ID: ${args.mdReqID}`
294
+ });
287
295
  });
288
- const entryTypes = args.mdEntryTypes || [MDEntryType.Bid, MDEntryType.Offer, MDEntryType.TradeVolume];
296
+ const entryTypes = args.mdEntryTypes || [
297
+ MDEntryType.Bid,
298
+ MDEntryType.Offer,
299
+ MDEntryType.Trade,
300
+ MDEntryType.IndexValue,
301
+ MDEntryType.OpeningPrice,
302
+ MDEntryType.ClosingPrice,
303
+ MDEntryType.SettlementPrice,
304
+ MDEntryType.TradingSessionHighPrice,
305
+ MDEntryType.TradingSessionLowPrice,
306
+ MDEntryType.VWAP,
307
+ MDEntryType.Imbalance,
308
+ MDEntryType.TradeVolume,
309
+ MDEntryType.OpenInterest,
310
+ MDEntryType.CompositeUnderlyingPrice,
311
+ MDEntryType.SimulatedSellPrice,
312
+ MDEntryType.SimulatedBuyPrice,
313
+ MDEntryType.MarginRate,
314
+ MDEntryType.MidPrice,
315
+ MDEntryType.EmptyBook,
316
+ MDEntryType.SettleHighPrice,
317
+ MDEntryType.SettleLowPrice,
318
+ MDEntryType.PriorSettlePrice,
319
+ MDEntryType.SessionHighBid,
320
+ MDEntryType.SessionLowOffer,
321
+ MDEntryType.EarlyPrices,
322
+ MDEntryType.AuctionClearingPrice,
323
+ MDEntryType.SwapValueFactor,
324
+ MDEntryType.DailyValueAdjustmentForLongPositions,
325
+ MDEntryType.CumulativeValueAdjustmentForLongPositions,
326
+ MDEntryType.DailyValueAdjustmentForShortPositions,
327
+ MDEntryType.CumulativeValueAdjustmentForShortPositions,
328
+ MDEntryType.FixingPrice,
329
+ MDEntryType.CashRate,
330
+ MDEntryType.RecoveryRate,
331
+ MDEntryType.RecoveryRateForLong,
332
+ MDEntryType.RecoveryRateForShort,
333
+ MDEntryType.MarketBid,
334
+ MDEntryType.MarketOffer,
335
+ MDEntryType.ShortSaleMinPrice,
336
+ MDEntryType.PreviousClosingPrice,
337
+ MDEntryType.ThresholdLimitPriceBanding,
338
+ MDEntryType.DailyFinancingValue,
339
+ MDEntryType.AccruedFinancingValue,
340
+ MDEntryType.TWAP
341
+ ];
289
342
  const messageFields = [
290
343
  new Field(Fields.MsgType, Messages.MarketDataRequest),
291
344
  new Field(Fields.SenderCompID, parser.sender),
@@ -307,6 +360,10 @@ var createMarketDataRequestHandler = (parser, pendingRequests) => {
307
360
  });
308
361
  const mdr = parser.createMessage(...messageFields);
309
362
  if (!parser.connected) {
363
+ parser.logger.log({
364
+ level: "error",
365
+ message: "Not connected. Cannot send market data request."
366
+ });
310
367
  return {
311
368
  content: [
312
369
  {
@@ -318,8 +375,16 @@ var createMarketDataRequestHandler = (parser, pendingRequests) => {
318
375
  isError: true
319
376
  };
320
377
  }
378
+ parser.logger.log({
379
+ level: "info",
380
+ message: `Sending market data request message: ${JSON.stringify(mdr?.toFIXJSON())}`
381
+ });
321
382
  parser.send(mdr);
322
383
  const fixData = await response;
384
+ parser.logger.log({
385
+ level: "info",
386
+ message: `Received market data response for request ID: ${args.mdReqID}`
387
+ });
323
388
  return {
324
389
  content: [
325
390
  {
@@ -368,6 +433,9 @@ var createGetStockGraphHandler = (marketDataPrices) => {
368
433
  const offerData = priceHistory.map((point) => point.offer);
369
434
  const spreadData = priceHistory.map((point) => point.spread);
370
435
  const volumeData = priceHistory.map((point) => point.volume);
436
+ const tradeData = priceHistory.map((point) => point.trade);
437
+ const vwapData = priceHistory.map((point) => point.vwap);
438
+ const twapData = priceHistory.map((point) => point.twap);
371
439
  const config = {
372
440
  type: "line",
373
441
  data: {
@@ -397,6 +465,30 @@ var createGetStockGraphHandler = (marketDataPrices) => {
397
465
  fill: false,
398
466
  tension: 0.4
399
467
  },
468
+ {
469
+ label: "Trade",
470
+ data: tradeData,
471
+ borderColor: "#ffc107",
472
+ backgroundColor: "rgba(255, 193, 7, 0.1)",
473
+ fill: false,
474
+ tension: 0.4
475
+ },
476
+ {
477
+ label: "VWAP",
478
+ data: vwapData,
479
+ borderColor: "#17a2b8",
480
+ backgroundColor: "rgba(23, 162, 184, 0.1)",
481
+ fill: false,
482
+ tension: 0.4
483
+ },
484
+ {
485
+ label: "TWAP",
486
+ data: twapData,
487
+ borderColor: "#6610f2",
488
+ backgroundColor: "rgba(102, 16, 242, 0.1)",
489
+ fill: false,
490
+ tension: 0.4
491
+ },
400
492
  {
401
493
  label: "Volume",
402
494
  data: volumeData,
@@ -442,7 +534,7 @@ var createGetStockGraphHandler = (marketDataPrices) => {
442
534
  content: [
443
535
  {
444
536
  type: "text",
445
- text: `Error: ${error instanceof Error ? error.message : "Failed to generate chart"}`,
537
+ text: `Error: ${error instanceof Error ? error.message : "Failed to generate graph"}`,
446
538
  uri: "getStockGraph"
447
539
  }
448
540
  ],
@@ -480,7 +572,48 @@ var createGetStockPriceHistoryHandler = (marketDataPrices) => {
480
572
  bid: point.bid,
481
573
  offer: point.offer,
482
574
  spread: point.spread,
483
- volume: point.volume
575
+ volume: point.volume,
576
+ trade: point.trade,
577
+ indexValue: point.indexValue,
578
+ openingPrice: point.openingPrice,
579
+ closingPrice: point.closingPrice,
580
+ settlementPrice: point.settlementPrice,
581
+ tradingSessionHighPrice: point.tradingSessionHighPrice,
582
+ tradingSessionLowPrice: point.tradingSessionLowPrice,
583
+ vwap: point.vwap,
584
+ imbalance: point.imbalance,
585
+ openInterest: point.openInterest,
586
+ compositeUnderlyingPrice: point.compositeUnderlyingPrice,
587
+ simulatedSellPrice: point.simulatedSellPrice,
588
+ simulatedBuyPrice: point.simulatedBuyPrice,
589
+ marginRate: point.marginRate,
590
+ midPrice: point.midPrice,
591
+ emptyBook: point.emptyBook,
592
+ settleHighPrice: point.settleHighPrice,
593
+ settleLowPrice: point.settleLowPrice,
594
+ priorSettlePrice: point.priorSettlePrice,
595
+ sessionHighBid: point.sessionHighBid,
596
+ sessionLowOffer: point.sessionLowOffer,
597
+ earlyPrices: point.earlyPrices,
598
+ auctionClearingPrice: point.auctionClearingPrice,
599
+ swapValueFactor: point.swapValueFactor,
600
+ dailyValueAdjustmentForLongPositions: point.dailyValueAdjustmentForLongPositions,
601
+ cumulativeValueAdjustmentForLongPositions: point.cumulativeValueAdjustmentForLongPositions,
602
+ dailyValueAdjustmentForShortPositions: point.dailyValueAdjustmentForShortPositions,
603
+ cumulativeValueAdjustmentForShortPositions: point.cumulativeValueAdjustmentForShortPositions,
604
+ fixingPrice: point.fixingPrice,
605
+ cashRate: point.cashRate,
606
+ recoveryRate: point.recoveryRate,
607
+ recoveryRateForLong: point.recoveryRateForLong,
608
+ recoveryRateForShort: point.recoveryRateForShort,
609
+ marketBid: point.marketBid,
610
+ marketOffer: point.marketOffer,
611
+ shortSaleMinPrice: point.shortSaleMinPrice,
612
+ previousClosingPrice: point.previousClosingPrice,
613
+ thresholdLimitPriceBanding: point.thresholdLimitPriceBanding,
614
+ dailyFinancingValue: point.dailyFinancingValue,
615
+ accruedFinancingValue: point.accruedFinancingValue,
616
+ twap: point.twap
484
617
  }))
485
618
  },
486
619
  null,
@@ -495,7 +628,7 @@ var createGetStockPriceHistoryHandler = (marketDataPrices) => {
495
628
  content: [
496
629
  {
497
630
  type: "text",
498
- text: `Error: ${error instanceof Error ? error.message : "Failed to get stock price history"}`,
631
+ text: `Error: ${error instanceof Error ? error.message : "Failed to get price history"}`,
499
632
  uri: "getStockPriceHistory"
500
633
  }
501
634
  ],
@@ -603,7 +736,7 @@ Parameters verified:
603
736
  - Symbol: ${args.symbol}
604
737
  - TimeInForce: ${args.timeInForce} (${timeInForceNames[args.timeInForce]})
605
738
 
606
- To execute this order, call the executeOrder tool with these exact same parameters.`,
739
+ To execute this order, call the executeOrder tool with these exact same parameters. Important: The user has to explicitly confirm before executeOrder is called!`,
607
740
  uri: "verifyOrder"
608
741
  }
609
742
  ]
@@ -804,52 +937,256 @@ var createToolHandlers = (parser, verifiedOrders, pendingRequests, marketDataPri
804
937
 
805
938
  // src/utils/messageHandler.ts
806
939
  import { Fields as Fields3, MDEntryType as MDEntryType2, Messages as Messages3 } from "fixparser";
940
+ function getEnumValue(enumObj, name) {
941
+ return enumObj[name] || name;
942
+ }
807
943
  function handleMessage(message, parser, pendingRequests, marketDataPrices, maxPriceHistory, onPriceUpdate) {
808
944
  parser.logger.log({
809
945
  level: "info",
810
946
  message: `MCP Server received message: ${message.messageType}: ${message.description}`
811
947
  });
812
948
  const msgType = message.messageType;
813
- if (msgType === Messages3.MarketDataSnapshotFullRefresh) {
949
+ if (msgType === Messages3.MarketDataSnapshotFullRefresh || msgType === Messages3.MarketDataIncrementalRefresh) {
814
950
  const symbol = message.getField(Fields3.Symbol)?.value;
815
- const entries = message.getField(Fields3.NoMDEntries)?.value;
816
- let bid = 0;
817
- let offer = 0;
818
- let volume = 0;
819
- const entryTypes = message.getFields(Fields3.MDEntryType);
820
- const entryPrices = message.getFields(Fields3.MDEntryPx);
821
- const entrySizes = message.getFields(Fields3.MDEntrySize);
822
- if (entryTypes && entryPrices && entrySizes) {
823
- for (let i = 0; i < entries; i++) {
824
- const entryType = entryTypes[i]?.value;
825
- const entryPrice = Number.parseFloat(entryPrices[i]?.value);
826
- const entrySize = Number.parseFloat(entrySizes[i]?.value);
827
- if (entryType === MDEntryType2.Bid) {
828
- bid = entryPrice;
829
- } else if (entryType === MDEntryType2.Offer) {
830
- offer = entryPrice;
831
- }
832
- volume += entrySize;
833
- }
834
- }
835
- const spread = offer - bid;
836
- const timestamp = Date.now();
951
+ parser.logger.log({
952
+ level: "info",
953
+ message: `Processing market data for symbol: ${symbol}`
954
+ });
955
+ const fixJson = message.toFIXJSON();
956
+ const entries = fixJson.Body?.NoMDEntries || [];
957
+ parser.logger.log({
958
+ level: "info",
959
+ message: `Found ${entries.length} market data entries`
960
+ });
837
961
  const data = {
838
- timestamp,
839
- bid,
840
- offer,
841
- spread,
842
- volume
962
+ timestamp: Date.now(),
963
+ bid: 0,
964
+ offer: 0,
965
+ spread: 0,
966
+ volume: 0,
967
+ trade: 0,
968
+ indexValue: 0,
969
+ openingPrice: 0,
970
+ closingPrice: 0,
971
+ settlementPrice: 0,
972
+ tradingSessionHighPrice: 0,
973
+ tradingSessionLowPrice: 0,
974
+ vwap: 0,
975
+ imbalance: 0,
976
+ openInterest: 0,
977
+ compositeUnderlyingPrice: 0,
978
+ simulatedSellPrice: 0,
979
+ simulatedBuyPrice: 0,
980
+ marginRate: 0,
981
+ midPrice: 0,
982
+ emptyBook: 0,
983
+ settleHighPrice: 0,
984
+ settleLowPrice: 0,
985
+ priorSettlePrice: 0,
986
+ sessionHighBid: 0,
987
+ sessionLowOffer: 0,
988
+ earlyPrices: 0,
989
+ auctionClearingPrice: 0,
990
+ swapValueFactor: 0,
991
+ dailyValueAdjustmentForLongPositions: 0,
992
+ cumulativeValueAdjustmentForLongPositions: 0,
993
+ dailyValueAdjustmentForShortPositions: 0,
994
+ cumulativeValueAdjustmentForShortPositions: 0,
995
+ fixingPrice: 0,
996
+ cashRate: 0,
997
+ recoveryRate: 0,
998
+ recoveryRateForLong: 0,
999
+ recoveryRateForShort: 0,
1000
+ marketBid: 0,
1001
+ marketOffer: 0,
1002
+ shortSaleMinPrice: 0,
1003
+ previousClosingPrice: 0,
1004
+ thresholdLimitPriceBanding: 0,
1005
+ dailyFinancingValue: 0,
1006
+ accruedFinancingValue: 0,
1007
+ twap: 0
843
1008
  };
1009
+ for (const entry of entries) {
1010
+ const entryType = entry.MDEntryType;
1011
+ const price = entry.MDEntryPx ? Number.parseFloat(entry.MDEntryPx) : 0;
1012
+ const size = entry.MDEntrySize ? Number.parseFloat(entry.MDEntrySize) : 0;
1013
+ if (entryType === MDEntryType2.Bid || entryType === MDEntryType2.Offer || entryType === MDEntryType2.TradeVolume) {
1014
+ parser.logger.log({
1015
+ level: "info",
1016
+ message: `Market Data Entry - Type: ${entryType}, Price: ${price}, Size: ${size}`
1017
+ });
1018
+ }
1019
+ const enumValue = getEnumValue(MDEntryType2, entryType);
1020
+ switch (enumValue) {
1021
+ case MDEntryType2.Bid:
1022
+ data.bid = price;
1023
+ break;
1024
+ case MDEntryType2.Offer:
1025
+ data.offer = price;
1026
+ break;
1027
+ case MDEntryType2.Trade:
1028
+ data.trade = price;
1029
+ break;
1030
+ case MDEntryType2.IndexValue:
1031
+ data.indexValue = price;
1032
+ break;
1033
+ case MDEntryType2.OpeningPrice:
1034
+ data.openingPrice = price;
1035
+ break;
1036
+ case MDEntryType2.ClosingPrice:
1037
+ data.closingPrice = price;
1038
+ break;
1039
+ case MDEntryType2.SettlementPrice:
1040
+ data.settlementPrice = price;
1041
+ break;
1042
+ case MDEntryType2.TradingSessionHighPrice:
1043
+ data.tradingSessionHighPrice = price;
1044
+ break;
1045
+ case MDEntryType2.TradingSessionLowPrice:
1046
+ data.tradingSessionLowPrice = price;
1047
+ break;
1048
+ case MDEntryType2.VWAP:
1049
+ data.vwap = price;
1050
+ break;
1051
+ case MDEntryType2.Imbalance:
1052
+ data.imbalance = size;
1053
+ break;
1054
+ case MDEntryType2.TradeVolume:
1055
+ data.volume = size;
1056
+ break;
1057
+ case MDEntryType2.OpenInterest:
1058
+ data.openInterest = size;
1059
+ break;
1060
+ case MDEntryType2.CompositeUnderlyingPrice:
1061
+ data.compositeUnderlyingPrice = price;
1062
+ break;
1063
+ case MDEntryType2.SimulatedSellPrice:
1064
+ data.simulatedSellPrice = price;
1065
+ break;
1066
+ case MDEntryType2.SimulatedBuyPrice:
1067
+ data.simulatedBuyPrice = price;
1068
+ break;
1069
+ case MDEntryType2.MarginRate:
1070
+ data.marginRate = price;
1071
+ break;
1072
+ case MDEntryType2.MidPrice:
1073
+ data.midPrice = price;
1074
+ break;
1075
+ case MDEntryType2.EmptyBook:
1076
+ data.emptyBook = 1;
1077
+ break;
1078
+ case MDEntryType2.SettleHighPrice:
1079
+ data.settleHighPrice = price;
1080
+ break;
1081
+ case MDEntryType2.SettleLowPrice:
1082
+ data.settleLowPrice = price;
1083
+ break;
1084
+ case MDEntryType2.PriorSettlePrice:
1085
+ data.priorSettlePrice = price;
1086
+ break;
1087
+ case MDEntryType2.SessionHighBid:
1088
+ data.sessionHighBid = price;
1089
+ break;
1090
+ case MDEntryType2.SessionLowOffer:
1091
+ data.sessionLowOffer = price;
1092
+ break;
1093
+ case MDEntryType2.EarlyPrices:
1094
+ data.earlyPrices = price;
1095
+ break;
1096
+ case MDEntryType2.AuctionClearingPrice:
1097
+ data.auctionClearingPrice = price;
1098
+ break;
1099
+ case MDEntryType2.SwapValueFactor:
1100
+ data.swapValueFactor = price;
1101
+ break;
1102
+ case MDEntryType2.DailyValueAdjustmentForLongPositions:
1103
+ data.dailyValueAdjustmentForLongPositions = price;
1104
+ break;
1105
+ case MDEntryType2.CumulativeValueAdjustmentForLongPositions:
1106
+ data.cumulativeValueAdjustmentForLongPositions = price;
1107
+ break;
1108
+ case MDEntryType2.DailyValueAdjustmentForShortPositions:
1109
+ data.dailyValueAdjustmentForShortPositions = price;
1110
+ break;
1111
+ case MDEntryType2.CumulativeValueAdjustmentForShortPositions:
1112
+ data.cumulativeValueAdjustmentForShortPositions = price;
1113
+ break;
1114
+ case MDEntryType2.FixingPrice:
1115
+ data.fixingPrice = price;
1116
+ break;
1117
+ case MDEntryType2.CashRate:
1118
+ data.cashRate = price;
1119
+ break;
1120
+ case MDEntryType2.RecoveryRate:
1121
+ data.recoveryRate = price;
1122
+ break;
1123
+ case MDEntryType2.RecoveryRateForLong:
1124
+ data.recoveryRateForLong = price;
1125
+ break;
1126
+ case MDEntryType2.RecoveryRateForShort:
1127
+ data.recoveryRateForShort = price;
1128
+ break;
1129
+ case MDEntryType2.MarketBid:
1130
+ data.marketBid = price;
1131
+ break;
1132
+ case MDEntryType2.MarketOffer:
1133
+ data.marketOffer = price;
1134
+ break;
1135
+ case MDEntryType2.ShortSaleMinPrice:
1136
+ data.shortSaleMinPrice = price;
1137
+ break;
1138
+ case MDEntryType2.PreviousClosingPrice:
1139
+ data.previousClosingPrice = price;
1140
+ break;
1141
+ case MDEntryType2.ThresholdLimitPriceBanding:
1142
+ data.thresholdLimitPriceBanding = price;
1143
+ break;
1144
+ case MDEntryType2.DailyFinancingValue:
1145
+ data.dailyFinancingValue = price;
1146
+ break;
1147
+ case MDEntryType2.AccruedFinancingValue:
1148
+ data.accruedFinancingValue = price;
1149
+ break;
1150
+ case MDEntryType2.TWAP:
1151
+ data.twap = price;
1152
+ break;
1153
+ }
1154
+ }
1155
+ data.spread = data.offer - data.bid;
1156
+ console.log(">>>>>>>>>>>>>>>>>>>>", data);
844
1157
  if (!marketDataPrices.has(symbol)) {
1158
+ parser.logger.log({
1159
+ level: "info",
1160
+ message: `Creating new price history array for symbol: ${symbol}`
1161
+ });
845
1162
  marketDataPrices.set(symbol, []);
846
1163
  }
847
1164
  const prices = marketDataPrices.get(symbol);
848
1165
  prices.push(data);
1166
+ parser.logger.log({
1167
+ level: "info",
1168
+ message: `Updated price history for ${symbol}. Current size: ${prices.length}`
1169
+ });
849
1170
  if (prices.length > maxPriceHistory) {
850
1171
  prices.splice(0, prices.length - maxPriceHistory);
1172
+ parser.logger.log({
1173
+ level: "info",
1174
+ message: `Trimmed price history for ${symbol} to ${maxPriceHistory} entries`
1175
+ });
851
1176
  }
852
1177
  onPriceUpdate?.(symbol, data);
1178
+ const mdReqID = message.getField(Fields3.MDReqID)?.value;
1179
+ if (mdReqID) {
1180
+ const callback = pendingRequests.get(mdReqID);
1181
+ if (callback) {
1182
+ callback(message);
1183
+ pendingRequests.delete(mdReqID);
1184
+ parser.logger.log({
1185
+ level: "info",
1186
+ message: `Resolved market data request for ID: ${mdReqID}`
1187
+ });
1188
+ }
1189
+ }
853
1190
  } else if (msgType === Messages3.ExecutionReport) {
854
1191
  const reqId = message.getField(Fields3.ClOrdID)?.value;
855
1192
  const callback = pendingRequests.get(reqId);
@@ -878,7 +1215,7 @@ var MCPLocal = class extends MCPBase {
878
1215
  */
879
1216
  marketDataPrices = /* @__PURE__ */ new Map();
880
1217
  /**
881
- * Maximum number of price points to store per symbol
1218
+ * Maximum number of price history entries to keep per symbol
882
1219
  * @private
883
1220
  */
884
1221
  MAX_PRICE_HISTORY = 1e5;