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)
193 lines (192 loc) • 28.5 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "default", {
enumerable: true,
get: function() {
return LatencyMonitor;
}
});
const _latencyListener = /*#__PURE__*/ _interop_require_default(require("../clients/metaApi/latencyListener"));
const _statisticalReservoir = /*#__PURE__*/ _interop_require_default(require("./reservoir/statisticalReservoir"));
const _reservoir = /*#__PURE__*/ _interop_require_default(require("./reservoir/reservoir"));
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
let LatencyMonitor = class LatencyMonitor extends _latencyListener.default {
/**
* 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.default(1000, 60 * 60 * 1000),
reservoir: new _reservoir.default(60, 60 * 60 * 1000)
},
"1d": {
percentiles: new _statisticalReservoir.default(1000, 24 * 60 * 60 * 1000),
reservoir: new _reservoir.default(60, 24 * 60 * 60 * 1000)
},
"1w": {
percentiles: new _statisticalReservoir.default(1000, 7 * 24 * 60 * 60 * 1000),
reservoir: new _reservoir.default(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
};
}
};
//# 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":["LatencyMonitor","LatencyListener","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","StatisticalReservoir","Reservoir","constructor"],"mappings":"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"}