UNPKG

metaapi.cloud-sdk

Version:

SDK for MetaApi, a professional cloud forex API which includes MetaTrader REST API and MetaTrader websocket API. Supports both MetaTrader 5 (MT5) and MetaTrader 4 (MT4). CopyFactory copy trading API included. (https://metaapi.cloud)

182 lines (181 loc) 28.1 kB
'use strict'; import LatencyListener from '../clients/metaApi/latencyListener'; import StatisticalReservoir from './reservoir/statisticalReservoir'; import Reservoir from './reservoir/reservoir'; let LatencyMonitor = class LatencyMonitor extends LatencyListener { /** * Invoked with latency information when application receives a response to RPC request * @param {string} accountId account id * @param {string} type request type * @param {ResponseTimestamps} timestamps request timestamps object containing latency information */ onResponse(accountId, type, timestamps) { if (!this._requestReservoirs[type]) { this._requestReservoirs[type] = { branch: true, clientLatency: this._initializeReservoirs(), serverLatency: this._initializeReservoirs() }; } if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) { let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime(); this._saveMeasurement(this._requestReservoirs[type].serverLatency, serverLatency); } if (timestamps.clientProcessingStarted && timestamps.clientProcessingFinished && timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) { let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime(); let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.clientProcessingStarted.getTime() - serverLatency; this._saveMeasurement(this._requestReservoirs[type].clientLatency, clientLatency); } } /** * Returns request processing latencies * @returns {Object} request processing latencies */ get requestLatencies() { return this._constructLatenciesRecursively(this._requestReservoirs); } /** * Invoked with latency information when application receives symbol price update event * @param {string} accountId account id * @param {string} symbol price symbol * @param {SymbolPriceTimestamps} timestamps timestamps object containing latency information about price streaming */ onSymbolPrice(accountId, symbol, timestamps) { if (timestamps.eventGenerated && timestamps.serverProcessingStarted) { let brokerLatency = timestamps.serverProcessingStarted.getTime() - timestamps.eventGenerated.getTime(); this._saveMeasurement(this._priceReservoirs.brokerLatency, brokerLatency); } if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) { let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime(); this._saveMeasurement(this._priceReservoirs.serverLatency, serverLatency); } if (timestamps.serverProcessingFinished && timestamps.clientProcessingFinished) { let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.serverProcessingFinished.getTime(); this._saveMeasurement(this._priceReservoirs.clientLatency, clientLatency); } } /** * Returns price streaming latencies * @returns {Object} price streaming latencies */ get priceLatencies() { return this._constructLatenciesRecursively(this._priceReservoirs); } /** * Invoked with latency information when application receives update event * @param {string} accountId account id * @param {UpdateTimestamps} timestamps timestamps object containing latency information about update streaming */ onUpdate(accountId, timestamps) { if (timestamps.eventGenerated && timestamps.serverProcessingStarted) { let brokerLatency = timestamps.serverProcessingStarted.getTime() - timestamps.eventGenerated.getTime(); this._saveMeasurement(this._updateReservoirs.brokerLatency, brokerLatency); } if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) { let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime(); this._saveMeasurement(this._updateReservoirs.serverLatency, serverLatency); } if (timestamps.serverProcessingFinished && timestamps.clientProcessingFinished) { let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.serverProcessingFinished.getTime(); this._saveMeasurement(this._updateReservoirs.clientLatency, clientLatency); } } /** * Returns update streaming latencies * @returns {Object} update streaming latencies */ get updateLatencies() { return this._constructLatenciesRecursively(this._updateReservoirs); } /** * Invoked with latency information when application receives trade response * @param {string} accountId account id * @param {TradeTimestamps} timestamps timestamps object containing latency information about a trade */ onTrade(accountId, timestamps) { if (timestamps.clientProcessingStarted && timestamps.serverProcessingStarted) { let clientLatency = timestamps.serverProcessingStarted.getTime() - timestamps.clientProcessingStarted.getTime(); this._saveMeasurement(this._tradeReservoirs.clientLatency, clientLatency); } if (timestamps.serverProcessingStarted && timestamps.tradeStarted) { let serverLatency = timestamps.tradeStarted.getTime() - timestamps.serverProcessingStarted.getTime(); this._saveMeasurement(this._tradeReservoirs.serverLatency, serverLatency); } if (timestamps.tradeStarted && timestamps.tradeExecuted) { let brokerLatency = timestamps.tradeExecuted.getTime() - timestamps.tradeStarted.getTime(); this._saveMeasurement(this._tradeReservoirs.brokerLatency, brokerLatency); } } /** * Returns trade latencies * @returns {Object} trade latencies */ get tradeLatencies() { return this._constructLatenciesRecursively(this._tradeReservoirs); } _saveMeasurement(reservoirs, clientLatency) { for (let e of Object.entries(reservoirs)){ if (e[0] === 'branch') { continue; } e[1].percentiles.pushMeasurement(clientLatency); e[1].reservoir.pushMeasurement(clientLatency); } } _constructLatenciesRecursively(reservoirs) { let result = {}; for (let e of Object.entries(reservoirs)){ if (e[0] === 'branch') { continue; } result[e[0]] = e[1].branch ? this._constructLatenciesRecursively(e[1]) : { p50: e[1].percentiles.getPercentile(50), p75: e[1].percentiles.getPercentile(75), p90: e[1].percentiles.getPercentile(90), p95: e[1].percentiles.getPercentile(95), p98: e[1].percentiles.getPercentile(98), avg: e[1].reservoir.getStatistics().average, count: e[1].reservoir.getStatistics().count, min: e[1].reservoir.getStatistics().min, max: e[1].reservoir.getStatistics().max }; } return result; } _initializeReservoirs() { return { branch: true, '1h': { percentiles: new StatisticalReservoir(1000, 60 * 60 * 1000), reservoir: new Reservoir(60, 60 * 60 * 1000) }, '1d': { percentiles: new StatisticalReservoir(1000, 24 * 60 * 60 * 1000), reservoir: new Reservoir(60, 24 * 60 * 60 * 1000) }, '1w': { percentiles: new StatisticalReservoir(1000, 7 * 24 * 60 * 60 * 1000), reservoir: new Reservoir(60, 7 * 24 * 60 * 60 * 1000) } }; } /** * Constructs latency monitor instance */ constructor(){ super(); this._tradeReservoirs = { clientLatency: this._initializeReservoirs(), serverLatency: this._initializeReservoirs(), brokerLatency: this._initializeReservoirs() }; this._updateReservoirs = { clientLatency: this._initializeReservoirs(), serverLatency: this._initializeReservoirs(), brokerLatency: this._initializeReservoirs() }; this._priceReservoirs = { clientLatency: this._initializeReservoirs(), serverLatency: this._initializeReservoirs(), brokerLatency: this._initializeReservoirs() }; this._requestReservoirs = { branch: true }; } }; /** * Responsible for monitoring MetaApi application latencies */ export { LatencyMonitor as default }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["<anon>"],"sourcesContent":["'use strict';\n\nimport LatencyListener from '../clients/metaApi/latencyListener';\nimport StatisticalReservoir from './reservoir/statisticalReservoir';\nimport Reservoir from './reservoir/reservoir';\n\n/**\n * Responsible for monitoring MetaApi application latencies\n */\nexport default class LatencyMonitor extends LatencyListener {\n\n  /**\n   * Constructs latency monitor instance\n   */\n  constructor() {\n    super();\n    this._tradeReservoirs = {\n      clientLatency: this._initializeReservoirs(),\n      serverLatency: this._initializeReservoirs(),\n      brokerLatency: this._initializeReservoirs()\n    };\n    this._updateReservoirs = {\n      clientLatency: this._initializeReservoirs(),\n      serverLatency: this._initializeReservoirs(),\n      brokerLatency: this._initializeReservoirs()\n    };\n    this._priceReservoirs = {\n      clientLatency: this._initializeReservoirs(),\n      serverLatency: this._initializeReservoirs(),\n      brokerLatency: this._initializeReservoirs()\n    };\n    this._requestReservoirs = {\n      branch: true\n    };\n  }\n\n  /**\n   * Invoked with latency information when application receives a response to RPC request\n   * @param {string} accountId account id\n   * @param {string} type request type\n   * @param {ResponseTimestamps} timestamps request timestamps object containing latency information\n   */\n  onResponse(accountId, type, timestamps) {\n    if (!this._requestReservoirs[type]) {\n      this._requestReservoirs[type] = {\n        branch: true,\n        clientLatency: this._initializeReservoirs(),\n        serverLatency: this._initializeReservoirs()\n      };\n    }\n    if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) {\n      let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime();\n      this._saveMeasurement(this._requestReservoirs[type].serverLatency, serverLatency);\n    }\n    if (timestamps.clientProcessingStarted && timestamps.clientProcessingFinished &&\n      timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) {\n      let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime();\n      let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.clientProcessingStarted.getTime() -\n        serverLatency;\n      this._saveMeasurement(this._requestReservoirs[type].clientLatency, clientLatency);\n    }\n  }\n\n  /**\n   * Returns request processing latencies\n   * @returns {Object} request processing latencies\n   */\n  get requestLatencies() {\n    return this._constructLatenciesRecursively(this._requestReservoirs);\n  }\n\n  /**\n   * Invoked with latency information when application receives symbol price update event\n   * @param {string} accountId account id\n   * @param {string} symbol price symbol\n   * @param {SymbolPriceTimestamps} timestamps timestamps object containing latency information about price streaming\n   */\n  onSymbolPrice(accountId, symbol, timestamps) {\n    if (timestamps.eventGenerated && timestamps.serverProcessingStarted) {\n      let brokerLatency = timestamps.serverProcessingStarted.getTime() - timestamps.eventGenerated.getTime();\n      this._saveMeasurement(this._priceReservoirs.brokerLatency, brokerLatency);\n    }\n    if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) {\n      let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime();\n      this._saveMeasurement(this._priceReservoirs.serverLatency, serverLatency);\n    }\n    if (timestamps.serverProcessingFinished && timestamps.clientProcessingFinished) {\n      let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.serverProcessingFinished.getTime();\n      this._saveMeasurement(this._priceReservoirs.clientLatency, clientLatency);\n    }\n  }\n\n  /**\n   * Returns price streaming latencies\n   * @returns {Object} price streaming latencies\n   */\n  get priceLatencies() {\n    return this._constructLatenciesRecursively(this._priceReservoirs);\n  }\n\n  /**\n   * Invoked with latency information when application receives update event\n   * @param {string} accountId account id\n   * @param {UpdateTimestamps} timestamps timestamps object containing latency information about update streaming\n   */\n  onUpdate(accountId, timestamps) {\n    if (timestamps.eventGenerated && timestamps.serverProcessingStarted) {\n      let brokerLatency = timestamps.serverProcessingStarted.getTime() - timestamps.eventGenerated.getTime();\n      this._saveMeasurement(this._updateReservoirs.brokerLatency, brokerLatency);\n    }\n    if (timestamps.serverProcessingStarted && timestamps.serverProcessingFinished) {\n      let serverLatency = timestamps.serverProcessingFinished.getTime() - timestamps.serverProcessingStarted.getTime();\n      this._saveMeasurement(this._updateReservoirs.serverLatency, serverLatency);\n    }\n    if (timestamps.serverProcessingFinished && timestamps.clientProcessingFinished) {\n      let clientLatency = timestamps.clientProcessingFinished.getTime() - timestamps.serverProcessingFinished.getTime();\n      this._saveMeasurement(this._updateReservoirs.clientLatency, clientLatency);\n    }\n  }\n\n  /**\n   * Returns update streaming latencies\n   * @returns {Object} update streaming latencies\n   */\n  get updateLatencies() {\n    return this._constructLatenciesRecursively(this._updateReservoirs);\n  }\n\n  /**\n   * Invoked with latency information when application receives trade response\n   * @param {string} accountId account id\n   * @param {TradeTimestamps} timestamps timestamps object containing latency information about a trade\n   */\n  onTrade(accountId, timestamps) {\n    if (timestamps.clientProcessingStarted && timestamps.serverProcessingStarted) {\n      let clientLatency = timestamps.serverProcessingStarted.getTime() - timestamps.clientProcessingStarted.getTime();\n      this._saveMeasurement(this._tradeReservoirs.clientLatency, clientLatency);\n    }\n    if (timestamps.serverProcessingStarted && timestamps.tradeStarted) {\n      let serverLatency = timestamps.tradeStarted.getTime() - timestamps.serverProcessingStarted.getTime();\n      this._saveMeasurement(this._tradeReservoirs.serverLatency, serverLatency);\n    }\n    if (timestamps.tradeStarted && timestamps.tradeExecuted) {\n      let brokerLatency = timestamps.tradeExecuted.getTime() - timestamps.tradeStarted.getTime();\n      this._saveMeasurement(this._tradeReservoirs.brokerLatency, brokerLatency);\n    }\n  }\n\n  /**\n   * Returns trade latencies\n   * @returns {Object} trade latencies\n   */\n  get tradeLatencies() {\n    return this._constructLatenciesRecursively(this._tradeReservoirs);\n  }\n\n  _saveMeasurement(reservoirs, clientLatency) {\n    for (let e of Object.entries(reservoirs)) {\n      if (e[0] === 'branch') {\n        continue;\n      }\n      e[1].percentiles.pushMeasurement(clientLatency);\n      e[1].reservoir.pushMeasurement(clientLatency);\n    }\n  }\n\n  _constructLatenciesRecursively(reservoirs) {\n    let result = {};\n    for (let e of Object.entries(reservoirs)) {\n      if (e[0] === 'branch') {\n        continue;\n      }\n      result[e[0]] = e[1].branch ? this._constructLatenciesRecursively(e[1]) : {\n        p50: e[1].percentiles.getPercentile(50),\n        p75: e[1].percentiles.getPercentile(75),\n        p90: e[1].percentiles.getPercentile(90),\n        p95: e[1].percentiles.getPercentile(95),\n        p98: e[1].percentiles.getPercentile(98),\n        avg: e[1].reservoir.getStatistics().average,\n        count: e[1].reservoir.getStatistics().count,\n        min: e[1].reservoir.getStatistics().min,\n        max: e[1].reservoir.getStatistics().max\n      };\n    }\n    return result;\n  }\n\n  _initializeReservoirs() {\n    return {\n      branch: true,\n      '1h': {\n        percentiles: new StatisticalReservoir(1000, 60 * 60 * 1000),\n        reservoir: new Reservoir(60, 60 * 60 * 1000)\n      },\n      '1d': {\n        percentiles: new StatisticalReservoir(1000, 24 * 60 * 60 * 1000),\n        reservoir: new Reservoir(60, 24 * 60 * 60 * 1000)\n      },\n      '1w': {\n        percentiles: new StatisticalReservoir(1000, 7 * 24 * 60 * 60 * 1000),\n        reservoir: new Reservoir(60, 7 * 24 * 60 * 60 * 1000)\n      }\n    };\n  }\n\n}\n"],"names":["LatencyListener","StatisticalReservoir","Reservoir","LatencyMonitor","onResponse","accountId","type","timestamps","_requestReservoirs","branch","clientLatency","_initializeReservoirs","serverLatency","serverProcessingStarted","serverProcessingFinished","getTime","_saveMeasurement","clientProcessingStarted","clientProcessingFinished","requestLatencies","_constructLatenciesRecursively","onSymbolPrice","symbol","eventGenerated","brokerLatency","_priceReservoirs","priceLatencies","onUpdate","_updateReservoirs","updateLatencies","onTrade","_tradeReservoirs","tradeStarted","tradeExecuted","tradeLatencies","reservoirs","e","Object","entries","percentiles","pushMeasurement","reservoir","result","p50","getPercentile","p75","p90","p95","p98","avg","getStatistics","average","count","min","max","constructor"],"mappings":"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"}