You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
138 lines
4.7 KiB
138 lines
4.7 KiB
"use strict";
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const events_1 = require("events");
|
|
const utils_1 = require("../utils");
|
|
const util_1 = require("./util");
|
|
const redis_1 = require("../redis");
|
|
const debug = utils_1.Debug("cluster:connectionPool");
|
|
class ConnectionPool extends events_1.EventEmitter {
|
|
constructor(redisOptions) {
|
|
super();
|
|
this.redisOptions = redisOptions;
|
|
// master + slave = all
|
|
this.nodes = {
|
|
all: {},
|
|
master: {},
|
|
slave: {},
|
|
};
|
|
this.specifiedOptions = {};
|
|
}
|
|
getNodes(role = "all") {
|
|
const nodes = this.nodes[role];
|
|
return Object.keys(nodes).map((key) => nodes[key]);
|
|
}
|
|
getInstanceByKey(key) {
|
|
return this.nodes.all[key];
|
|
}
|
|
getSampleInstance(role) {
|
|
const keys = Object.keys(this.nodes[role]);
|
|
const sampleKey = utils_1.sample(keys);
|
|
return this.nodes[role][sampleKey];
|
|
}
|
|
/**
|
|
* Find or create a connection to the node
|
|
*
|
|
* @param {IRedisOptions} node
|
|
* @param {boolean} [readOnly=false]
|
|
* @returns {*}
|
|
* @memberof ConnectionPool
|
|
*/
|
|
findOrCreate(node, readOnly = false) {
|
|
const key = util_1.getNodeKey(node);
|
|
readOnly = Boolean(readOnly);
|
|
if (this.specifiedOptions[key]) {
|
|
Object.assign(node, this.specifiedOptions[key]);
|
|
}
|
|
else {
|
|
this.specifiedOptions[key] = node;
|
|
}
|
|
let redis;
|
|
if (this.nodes.all[key]) {
|
|
redis = this.nodes.all[key];
|
|
if (redis.options.readOnly !== readOnly) {
|
|
redis.options.readOnly = readOnly;
|
|
debug("Change role of %s to %s", key, readOnly ? "slave" : "master");
|
|
redis[readOnly ? "readonly" : "readwrite"]().catch(utils_1.noop);
|
|
if (readOnly) {
|
|
delete this.nodes.master[key];
|
|
this.nodes.slave[key] = redis;
|
|
}
|
|
else {
|
|
delete this.nodes.slave[key];
|
|
this.nodes.master[key] = redis;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
debug("Connecting to %s as %s", key, readOnly ? "slave" : "master");
|
|
redis = new redis_1.default(utils_1.defaults({
|
|
// Never try to reconnect when a node is lose,
|
|
// instead, waiting for a `MOVED` error and
|
|
// fetch the slots again.
|
|
retryStrategy: null,
|
|
// Offline queue should be enabled so that
|
|
// we don't need to wait for the `ready` event
|
|
// before sending commands to the node.
|
|
enableOfflineQueue: true,
|
|
readOnly: readOnly,
|
|
}, node, this.redisOptions, { lazyConnect: true }));
|
|
this.nodes.all[key] = redis;
|
|
this.nodes[readOnly ? "slave" : "master"][key] = redis;
|
|
redis.once("end", () => {
|
|
this.removeNode(key);
|
|
this.emit("-node", redis, key);
|
|
if (!Object.keys(this.nodes.all).length) {
|
|
this.emit("drain");
|
|
}
|
|
});
|
|
this.emit("+node", redis, key);
|
|
redis.on("error", function (error) {
|
|
this.emit("nodeError", error, key);
|
|
});
|
|
}
|
|
return redis;
|
|
}
|
|
/**
|
|
* Remove a node from the pool.
|
|
*/
|
|
removeNode(key) {
|
|
const { nodes } = this;
|
|
if (nodes.all[key]) {
|
|
debug("Remove %s from the pool", key);
|
|
delete nodes.all[key];
|
|
}
|
|
delete nodes.master[key];
|
|
delete nodes.slave[key];
|
|
}
|
|
/**
|
|
* Reset the pool with a set of nodes.
|
|
* The old node will be removed.
|
|
*
|
|
* @param {(Array<string | number | object>)} nodes
|
|
* @memberof ConnectionPool
|
|
*/
|
|
reset(nodes) {
|
|
debug("Reset with %O", nodes);
|
|
const newNodes = {};
|
|
nodes.forEach((node) => {
|
|
const key = util_1.getNodeKey(node);
|
|
// Don't override the existing (master) node
|
|
// when the current one is slave.
|
|
if (!(node.readOnly && newNodes[key])) {
|
|
newNodes[key] = node;
|
|
}
|
|
});
|
|
Object.keys(this.nodes.all).forEach((key) => {
|
|
if (!newNodes[key]) {
|
|
debug("Disconnect %s because the node does not hold any slot", key);
|
|
this.nodes.all[key].disconnect();
|
|
this.removeNode(key);
|
|
}
|
|
});
|
|
Object.keys(newNodes).forEach((key) => {
|
|
const node = newNodes[key];
|
|
this.findOrCreate(node, node.readOnly);
|
|
});
|
|
}
|
|
}
|
|
exports.default = ConnectionPool;
|
|
|