bifocal/node_modules/@elastic/transport/lib/pool/WeightedConnectionPool.js

165 lines
5.9 KiB
JavaScript

"use strict";
/*
* Licensed to Elasticsearch B.V. under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch B.V. licenses this file to you under
* the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
Object.defineProperty(exports, "__esModule", { value: true });
const tslib_1 = require("tslib");
const connection_1 = require("../connection");
const BaseConnectionPool_1 = tslib_1.__importDefault(require("./BaseConnectionPool"));
const noFilter = () => true;
class WeightedConnectionPool extends BaseConnectionPool_1.default {
constructor(opts) {
super(opts);
Object.defineProperty(this, "index", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "maxWeight", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "greatestCommonDivisor", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "currentWeight", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
// index choosen last time
this.index = -1;
// max weight of all nodes
this.maxWeight = 0;
// greatest common divisor of all nodes weights
this.greatestCommonDivisor = 0;
// current weight in scheduling
this.currentWeight = 0;
}
/**
* Returns a connection, even if the connection might be dead.
*
* @param {object} options (filter)
* @returns {object|null} connection
*/
getConnection(opts) {
const filter = opts.filter != null ? opts.filter : noFilter;
// we should be able to find the next node in 1 array scan,
// if we don't, it means that we are in an infinite loop
let counter = 0;
while (counter++ < this.size) {
// 0 <= index < size
this.index = (this.index + 1) % this.size;
if (this.index === 0) {
this.currentWeight = this.currentWeight - this.greatestCommonDivisor;
if (this.currentWeight <= 0) {
this.currentWeight = this.maxWeight;
/* istanbul ignore if */
if (this.currentWeight === 0) {
return null;
}
}
}
const connection = this.connections[this.index];
if (connection.weight >= this.currentWeight && filter(connection)) {
return connection;
}
}
return null;
}
/**
* Set the weight of a connection to the maximum value.
* If sniffing is not enabled and there is only
* one node, this method is a noop.
*
* @param {object} connection
*/
markAlive(connection) {
if (this.size === 1 || connection.status === connection_1.BaseConnection.statuses.ALIVE)
return this;
connection.status = connection_1.BaseConnection.statuses.ALIVE;
connection.deadCount = 0;
connection.weight = Math.round(1000 / this.size);
this.maxWeight = Math.max(...(this.connections.map(c => c.weight)));
this.greatestCommonDivisor = this.connections.map(c => c.weight).reduce(getGreatestCommonDivisor, 0);
return this;
}
/**
* Decreases the connection weight.
* If sniffing is not enabled and there is only
* one node, this method is a noop.
*
* @param {object} connection
*/
markDead(connection) {
if (this.size === 1)
return this;
connection.status = connection_1.BaseConnection.statuses.DEAD;
connection.deadCount++;
connection.weight -= Math.round(Math.pow(Math.log2(connection.weight), connection.deadCount));
/* istanbul ignore if */
if (connection.weight <= 0)
connection.weight = 1;
this.maxWeight = Math.max(...(this.connections.map(c => c.weight)));
this.greatestCommonDivisor = this.connections.map(c => c.weight).reduce(getGreatestCommonDivisor, 0);
return this;
}
/**
* Empties the connection pool.
*
* @returns {ConnectionPool}
*/
async empty() {
await super.empty();
this.maxWeight = 0;
this.greatestCommonDivisor = 0;
this.index = -1;
this.currentWeight = 0;
}
/**
* Update the ConnectionPool with new connections.
*
* @param {array} array of connections
* @returns {ConnectionPool}
*/
update(connections) {
super.update(connections);
this.connections.forEach(connection => {
connection.weight = Math.round(1000 / this.size);
});
this.maxWeight = Math.max(...(this.connections.map(c => c.weight)));
this.greatestCommonDivisor = this.connections.map(c => c.weight).reduce(getGreatestCommonDivisor, 0);
this.index = -1;
this.currentWeight = 0;
return this;
}
}
exports.default = WeightedConnectionPool;
function getGreatestCommonDivisor(a, b) {
if (b === 0)
return a;
return getGreatestCommonDivisor(b, a % b);
}
//# sourceMappingURL=WeightedConnectionPool.js.map