"use strict";
|
Object.defineProperty(exports, "__esModule", { value: true });
|
const events_1 = require("events");
|
const ClusterAllFailedError_1 = require("../errors/ClusterAllFailedError");
|
const utils_1 = require("../utils");
|
const ConnectionPool_1 = require("./ConnectionPool");
|
const util_1 = require("./util");
|
const ClusterSubscriber_1 = require("./ClusterSubscriber");
|
const DelayQueue_1 = require("./DelayQueue");
|
const ScanStream_1 = require("../ScanStream");
|
const redis_errors_1 = require("redis-errors");
|
const standard_as_callback_1 = require("standard-as-callback");
|
const PromiseContainer = require("../promiseContainer");
|
const ClusterOptions_1 = require("./ClusterOptions");
|
const utils_2 = require("../utils");
|
const commands = require("redis-commands");
|
const command_1 = require("../command");
|
const redis_1 = require("../redis");
|
const commander_1 = require("../commander");
|
const Deque = require("denque");
|
const debug = utils_1.Debug("cluster");
|
/**
|
* Client for the official Redis Cluster
|
*
|
* @class Cluster
|
* @extends {EventEmitter}
|
*/
|
class Cluster extends events_1.EventEmitter {
|
/**
|
* Creates an instance of Cluster.
|
*
|
* @param {(Array<string | number | object>)} startupNodes
|
* @param {IClusterOptions} [options={}]
|
* @memberof Cluster
|
*/
|
constructor(startupNodes, options = {}) {
|
super();
|
this.slots = [];
|
this.retryAttempts = 0;
|
this.delayQueue = new DelayQueue_1.default();
|
this.offlineQueue = new Deque();
|
this.isRefreshing = false;
|
/**
|
* Every time Cluster#connect() is called, this value will be
|
* auto-incrementing. The purpose of this value is used for
|
* discarding previous connect attampts when creating a new
|
* connection.
|
*
|
* @private
|
* @type {number}
|
* @memberof Cluster
|
*/
|
this.connectionEpoch = 0;
|
commander_1.default.call(this);
|
this.startupNodes = startupNodes;
|
this.options = utils_1.defaults({}, options, ClusterOptions_1.DEFAULT_CLUSTER_OPTIONS, this.options);
|
// validate options
|
if (typeof this.options.scaleReads !== "function" &&
|
["all", "master", "slave"].indexOf(this.options.scaleReads) === -1) {
|
throw new Error('Invalid option scaleReads "' +
|
this.options.scaleReads +
|
'". Expected "all", "master", "slave" or a custom function');
|
}
|
this.connectionPool = new ConnectionPool_1.default(this.options.redisOptions);
|
this.connectionPool.on("-node", (redis, key) => {
|
this.emit("-node", redis);
|
});
|
this.connectionPool.on("+node", redis => {
|
this.emit("+node", redis);
|
});
|
this.connectionPool.on("drain", () => {
|
this.setStatus("close");
|
});
|
this.connectionPool.on("nodeError", (error, key) => {
|
this.emit("node error", error, key);
|
});
|
this.subscriber = new ClusterSubscriber_1.default(this.connectionPool, this);
|
if (this.options.lazyConnect) {
|
this.setStatus("wait");
|
}
|
else {
|
this.connect().catch(err => {
|
debug("connecting failed: %s", err);
|
});
|
}
|
}
|
resetOfflineQueue() {
|
this.offlineQueue = new Deque();
|
}
|
clearNodesRefreshInterval() {
|
if (this.slotsTimer) {
|
clearTimeout(this.slotsTimer);
|
this.slotsTimer = null;
|
}
|
}
|
resetNodesRefreshInterval() {
|
if (this.slotsTimer) {
|
return;
|
}
|
const nextRound = () => {
|
this.slotsTimer = setTimeout(() => {
|
debug('refreshing slot caches... (triggered by "slotsRefreshInterval" option)');
|
this.refreshSlotsCache(() => {
|
nextRound();
|
});
|
}, this.options.slotsRefreshInterval);
|
};
|
nextRound();
|
}
|
/**
|
* Connect to a cluster
|
*
|
* @returns {Promise<void>}
|
* @memberof Cluster
|
*/
|
connect() {
|
const Promise = PromiseContainer.get();
|
return new Promise((resolve, reject) => {
|
if (this.status === "connecting" ||
|
this.status === "connect" ||
|
this.status === "ready") {
|
reject(new Error("Redis is already connecting/connected"));
|
return;
|
}
|
const epoch = ++this.connectionEpoch;
|
this.setStatus("connecting");
|
this.resolveStartupNodeHostnames()
|
.then(nodes => {
|
if (this.connectionEpoch !== epoch) {
|
debug("discard connecting after resolving startup nodes because epoch not match: %d != %d", epoch, this.connectionEpoch);
|
reject(new redis_errors_1.RedisError("Connection is discarded because a new connection is made"));
|
return;
|
}
|
if (this.status !== "connecting") {
|
debug("discard connecting after resolving startup nodes because the status changed to %s", this.status);
|
reject(new redis_errors_1.RedisError("Connection is aborted"));
|
return;
|
}
|
this.connectionPool.reset(nodes);
|
function readyHandler() {
|
this.setStatus("ready");
|
this.retryAttempts = 0;
|
this.executeOfflineCommands();
|
this.resetNodesRefreshInterval();
|
resolve();
|
}
|
let closeListener;
|
const refreshListener = () => {
|
this.removeListener("close", closeListener);
|
this.manuallyClosing = false;
|
this.setStatus("connect");
|
if (this.options.enableReadyCheck) {
|
this.readyCheck((err, fail) => {
|
if (err || fail) {
|
debug("Ready check failed (%s). Reconnecting...", err || fail);
|
if (this.status === "connect") {
|
this.disconnect(true);
|
}
|
}
|
else {
|
readyHandler.call(this);
|
}
|
});
|
}
|
else {
|
readyHandler.call(this);
|
}
|
};
|
closeListener = function () {
|
this.removeListener("refresh", refreshListener);
|
reject(new Error("None of startup nodes is available"));
|
};
|
this.once("refresh", refreshListener);
|
this.once("close", closeListener);
|
this.once("close", this.handleCloseEvent.bind(this));
|
this.refreshSlotsCache(function (err) {
|
if (err && err.message === "Failed to refresh slots cache.") {
|
redis_1.default.prototype.silentEmit.call(this, "error", err);
|
this.connectionPool.reset([]);
|
}
|
}.bind(this));
|
this.subscriber.start();
|
})
|
.catch(err => {
|
this.setStatus("close");
|
this.handleCloseEvent(err);
|
reject(err);
|
});
|
});
|
}
|
/**
|
* Called when closed to check whether a reconnection should be made
|
*
|
* @private
|
* @memberof Cluster
|
*/
|
handleCloseEvent(reason) {
|
if (reason) {
|
debug("closed because %s", reason);
|
}
|
let retryDelay;
|
if (!this.manuallyClosing &&
|
typeof this.options.clusterRetryStrategy === "function") {
|
retryDelay = this.options.clusterRetryStrategy.call(this, ++this.retryAttempts, reason);
|
}
|
if (typeof retryDelay === "number") {
|
this.setStatus("reconnecting");
|
this.reconnectTimeout = setTimeout(function () {
|
this.reconnectTimeout = null;
|
debug("Cluster is disconnected. Retrying after %dms", retryDelay);
|
this.connect().catch(function (err) {
|
debug("Got error %s when reconnecting. Ignoring...", err);
|
});
|
}.bind(this), retryDelay);
|
}
|
else {
|
this.setStatus("end");
|
this.flushQueue(new Error("None of startup nodes is available"));
|
}
|
}
|
/**
|
* Disconnect from every node in the cluster.
|
*
|
* @param {boolean} [reconnect=false]
|
* @memberof Cluster
|
*/
|
disconnect(reconnect = false) {
|
const status = this.status;
|
this.setStatus("disconnecting");
|
if (!reconnect) {
|
this.manuallyClosing = true;
|
}
|
if (this.reconnectTimeout) {
|
clearTimeout(this.reconnectTimeout);
|
this.reconnectTimeout = null;
|
debug("Canceled reconnecting attempts");
|
}
|
this.clearNodesRefreshInterval();
|
this.subscriber.stop();
|
if (status === "wait") {
|
this.setStatus("close");
|
this.handleCloseEvent();
|
}
|
else {
|
this.connectionPool.reset([]);
|
}
|
}
|
/**
|
* Quit the cluster gracefully.
|
*
|
* @param {CallbackFunction<'OK'>} [callback]
|
* @returns {Promise<'OK'>}
|
* @memberof Cluster
|
*/
|
quit(callback) {
|
const status = this.status;
|
this.setStatus("disconnecting");
|
this.manuallyClosing = true;
|
if (this.reconnectTimeout) {
|
clearTimeout(this.reconnectTimeout);
|
this.reconnectTimeout = null;
|
}
|
this.clearNodesRefreshInterval();
|
this.subscriber.stop();
|
const Promise = PromiseContainer.get();
|
if (status === "wait") {
|
const ret = standard_as_callback_1.default(Promise.resolve("OK"), callback);
|
// use setImmediate to make sure "close" event
|
// being emitted after quit() is returned
|
setImmediate(function () {
|
this.setStatus("close");
|
this.handleCloseEvent();
|
}.bind(this));
|
return ret;
|
}
|
return standard_as_callback_1.default(Promise.all(this.nodes().map(node => node.quit().catch(err => {
|
// Ignore the error caused by disconnecting since
|
// we're disconnecting...
|
if (err.message === utils_2.CONNECTION_CLOSED_ERROR_MSG) {
|
return "OK";
|
}
|
throw err;
|
}))).then(() => "OK"), callback);
|
}
|
/**
|
* Create a new instance with the same startup nodes and options as the current one.
|
*
|
* @example
|
* ```js
|
* var cluster = new Redis.Cluster([{ host: "127.0.0.1", port: "30001" }]);
|
* var anotherCluster = cluster.duplicate();
|
* ```
|
*
|
* @public
|
* @param {(Array<string | number | object>)} [overrideStartupNodes=[]]
|
* @param {IClusterOptions} [overrideOptions={}]
|
* @memberof Cluster
|
*/
|
duplicate(overrideStartupNodes = [], overrideOptions = {}) {
|
const startupNodes = overrideStartupNodes.length > 0
|
? overrideStartupNodes
|
: this.startupNodes.slice(0);
|
const options = Object.assign({}, this.options, overrideOptions);
|
return new Cluster(startupNodes, options);
|
}
|
/**
|
* Get nodes with the specified role
|
*
|
* @param {NodeRole} [role='all']
|
* @returns {any[]}
|
* @memberof Cluster
|
*/
|
nodes(role = "all") {
|
if (role !== "all" && role !== "master" && role !== "slave") {
|
throw new Error('Invalid role "' + role + '". Expected "all", "master" or "slave"');
|
}
|
return this.connectionPool.getNodes(role);
|
}
|
/**
|
* Change cluster instance's status
|
*
|
* @private
|
* @param {ClusterStatus} status
|
* @memberof Cluster
|
*/
|
setStatus(status) {
|
debug("status: %s -> %s", this.status || "[empty]", status);
|
this.status = status;
|
process.nextTick(() => {
|
this.emit(status);
|
});
|
}
|
/**
|
* Refresh the slot cache
|
*
|
* @private
|
* @param {CallbackFunction} [callback]
|
* @memberof Cluster
|
*/
|
refreshSlotsCache(callback) {
|
if (this.isRefreshing) {
|
if (typeof callback === "function") {
|
process.nextTick(callback);
|
}
|
return;
|
}
|
this.isRefreshing = true;
|
const _this = this;
|
const wrapper = function (error) {
|
_this.isRefreshing = false;
|
if (typeof callback === "function") {
|
callback(error);
|
}
|
};
|
const nodes = utils_2.shuffle(this.connectionPool.getNodes());
|
let lastNodeError = null;
|
function tryNode(index) {
|
if (index === nodes.length) {
|
const error = new ClusterAllFailedError_1.default("Failed to refresh slots cache.", lastNodeError);
|
return wrapper(error);
|
}
|
const node = nodes[index];
|
const key = `${node.options.host}:${node.options.port}`;
|
debug("getting slot cache from %s", key);
|
_this.getInfoFromNode(node, function (err) {
|
switch (_this.status) {
|
case "close":
|
case "end":
|
return wrapper(new Error("Cluster is disconnected."));
|
case "disconnecting":
|
return wrapper(new Error("Cluster is disconnecting."));
|
}
|
if (err) {
|
_this.emit("node error", err, key);
|
lastNodeError = err;
|
tryNode(index + 1);
|
}
|
else {
|
_this.emit("refresh");
|
wrapper();
|
}
|
});
|
}
|
tryNode(0);
|
}
|
/**
|
* Flush offline queue with error.
|
*
|
* @param {Error} error
|
* @memberof Cluster
|
*/
|
flushQueue(error) {
|
let item;
|
while (this.offlineQueue.length > 0) {
|
item = this.offlineQueue.shift();
|
item.command.reject(error);
|
}
|
}
|
executeOfflineCommands() {
|
if (this.offlineQueue.length) {
|
debug("send %d commands in offline queue", this.offlineQueue.length);
|
const offlineQueue = this.offlineQueue;
|
this.resetOfflineQueue();
|
while (offlineQueue.length > 0) {
|
const item = offlineQueue.shift();
|
this.sendCommand(item.command, item.stream, item.node);
|
}
|
}
|
}
|
natMapper(nodeKey) {
|
if (this.options.natMap && typeof this.options.natMap === "object") {
|
const key = typeof nodeKey === "string"
|
? nodeKey
|
: `${nodeKey.host}:${nodeKey.port}`;
|
const mapped = this.options.natMap[key];
|
if (mapped) {
|
debug("NAT mapping %s -> %O", key, mapped);
|
return Object.assign({}, mapped);
|
}
|
}
|
return typeof nodeKey === "string"
|
? util_1.nodeKeyToRedisOptions(nodeKey)
|
: nodeKey;
|
}
|
sendCommand(command, stream, node) {
|
if (this.status === "wait") {
|
this.connect().catch(utils_1.noop);
|
}
|
if (this.status === "end") {
|
command.reject(new Error(utils_2.CONNECTION_CLOSED_ERROR_MSG));
|
return command.promise;
|
}
|
let to = this.options.scaleReads;
|
if (to !== "master") {
|
const isCommandReadOnly = commands.exists(command.name) &&
|
commands.hasFlag(command.name, "readonly");
|
if (!isCommandReadOnly) {
|
to = "master";
|
}
|
}
|
let targetSlot = node ? node.slot : command.getSlot();
|
const ttl = {};
|
const _this = this;
|
if (!node && !command.__is_reject_overwritten) {
|
// eslint-disable-next-line @typescript-eslint/camelcase
|
command.__is_reject_overwritten = true;
|
const reject = command.reject;
|
command.reject = function (err) {
|
const partialTry = tryConnection.bind(null, true);
|
_this.handleError(err, ttl, {
|
moved: function (slot, key) {
|
debug("command %s is moved to %s", command.name, key);
|
targetSlot = Number(slot);
|
if (_this.slots[slot]) {
|
_this.slots[slot][0] = key;
|
}
|
else {
|
_this.slots[slot] = [key];
|
}
|
_this.connectionPool.findOrCreate(_this.natMapper(key));
|
tryConnection();
|
debug("refreshing slot caches... (triggered by MOVED error)");
|
_this.refreshSlotsCache();
|
},
|
ask: function (slot, key) {
|
debug("command %s is required to ask %s:%s", command.name, key);
|
const mapped = _this.natMapper(key);
|
_this.connectionPool.findOrCreate(mapped);
|
tryConnection(false, `${mapped.host}:${mapped.port}`);
|
},
|
tryagain: partialTry,
|
clusterDown: partialTry,
|
connectionClosed: partialTry,
|
maxRedirections: function (redirectionError) {
|
reject.call(command, redirectionError);
|
},
|
defaults: function () {
|
reject.call(command, err);
|
}
|
});
|
};
|
}
|
tryConnection();
|
function tryConnection(random, asking) {
|
if (_this.status === "end") {
|
command.reject(new redis_errors_1.AbortError("Cluster is ended."));
|
return;
|
}
|
let redis;
|
if (_this.status === "ready" || command.name === "cluster") {
|
if (node && node.redis) {
|
redis = node.redis;
|
}
|
else if (command_1.default.checkFlag("ENTER_SUBSCRIBER_MODE", command.name) ||
|
command_1.default.checkFlag("EXIT_SUBSCRIBER_MODE", command.name)) {
|
redis = _this.subscriber.getInstance();
|
if (!redis) {
|
command.reject(new redis_errors_1.AbortError("No subscriber for the cluster"));
|
return;
|
}
|
}
|
else {
|
if (!random) {
|
if (typeof targetSlot === "number" && _this.slots[targetSlot]) {
|
const nodeKeys = _this.slots[targetSlot];
|
if (typeof to === "function") {
|
const nodes = nodeKeys.map(function (key) {
|
return _this.connectionPool.getInstanceByKey(key);
|
});
|
redis = to(nodes, command);
|
if (Array.isArray(redis)) {
|
redis = utils_2.sample(redis);
|
}
|
if (!redis) {
|
redis = nodes[0];
|
}
|
}
|
else {
|
let key;
|
if (to === "all") {
|
key = utils_2.sample(nodeKeys);
|
}
|
else if (to === "slave" && nodeKeys.length > 1) {
|
key = utils_2.sample(nodeKeys, 1);
|
}
|
else {
|
key = nodeKeys[0];
|
}
|
redis = _this.connectionPool.getInstanceByKey(key);
|
}
|
}
|
if (asking) {
|
redis = _this.connectionPool.getInstanceByKey(asking);
|
redis.asking();
|
}
|
}
|
if (!redis) {
|
redis =
|
(typeof to === "function"
|
? null
|
: _this.connectionPool.getSampleInstance(to)) ||
|
_this.connectionPool.getSampleInstance("all");
|
}
|
}
|
if (node && !node.redis) {
|
node.redis = redis;
|
}
|
}
|
if (redis) {
|
redis.sendCommand(command, stream);
|
}
|
else if (_this.options.enableOfflineQueue) {
|
_this.offlineQueue.push({
|
command: command,
|
stream: stream,
|
node: node
|
});
|
}
|
else {
|
command.reject(new Error("Cluster isn't ready and enableOfflineQueue options is false"));
|
}
|
}
|
return command.promise;
|
}
|
handleError(error, ttl, handlers) {
|
if (typeof ttl.value === "undefined") {
|
ttl.value = this.options.maxRedirections;
|
}
|
else {
|
ttl.value -= 1;
|
}
|
if (ttl.value <= 0) {
|
handlers.maxRedirections(new Error("Too many Cluster redirections. Last error: " + error));
|
return;
|
}
|
const errv = error.message.split(" ");
|
if (errv[0] === "MOVED" || errv[0] === "ASK") {
|
handlers[errv[0] === "MOVED" ? "moved" : "ask"](errv[1], errv[2]);
|
}
|
else if (errv[0] === "TRYAGAIN") {
|
this.delayQueue.push("tryagain", handlers.tryagain, {
|
timeout: this.options.retryDelayOnTryAgain
|
});
|
}
|
else if (errv[0] === "CLUSTERDOWN" &&
|
this.options.retryDelayOnClusterDown > 0) {
|
this.delayQueue.push("clusterdown", handlers.connectionClosed, {
|
timeout: this.options.retryDelayOnClusterDown,
|
callback: this.refreshSlotsCache.bind(this)
|
});
|
}
|
else if (error.message === utils_2.CONNECTION_CLOSED_ERROR_MSG &&
|
this.options.retryDelayOnFailover > 0 &&
|
this.status === "ready") {
|
this.delayQueue.push("failover", handlers.connectionClosed, {
|
timeout: this.options.retryDelayOnFailover,
|
callback: this.refreshSlotsCache.bind(this)
|
});
|
}
|
else {
|
handlers.defaults();
|
}
|
}
|
getInfoFromNode(redis, callback) {
|
if (!redis) {
|
return callback(new Error("Node is disconnected"));
|
}
|
// Use a duplication of the connection to avoid
|
// timeouts when the connection is in the blocking
|
// mode (e.g. waiting for BLPOP).
|
const duplicatedConnection = redis.duplicate({
|
enableOfflineQueue: true,
|
enableReadyCheck: false,
|
retryStrategy: null,
|
connectionName: "ioredisClusterRefresher"
|
});
|
// Ignore error events since we will handle
|
// exceptions for the CLUSTER SLOTS command.
|
duplicatedConnection.on("error", utils_1.noop);
|
duplicatedConnection.cluster("slots", utils_2.timeout((err, result) => {
|
duplicatedConnection.disconnect();
|
if (err) {
|
return callback(err);
|
}
|
if (this.status === "disconnecting" ||
|
this.status === "close" ||
|
this.status === "end") {
|
debug("ignore CLUSTER.SLOTS results (count: %d) since cluster status is %s", result.length, this.status);
|
callback();
|
return;
|
}
|
const nodes = [];
|
debug("cluster slots result count: %d", result.length);
|
for (let i = 0; i < result.length; ++i) {
|
const items = result[i];
|
const slotRangeStart = items[0];
|
const slotRangeEnd = items[1];
|
const keys = [];
|
for (let j = 2; j < items.length; j++) {
|
items[j] = this.natMapper({ host: items[j][0], port: items[j][1] });
|
items[j].readOnly = j !== 2;
|
nodes.push(items[j]);
|
keys.push(items[j].host + ":" + items[j].port);
|
}
|
debug("cluster slots result [%d]: slots %d~%d served by %s", i, slotRangeStart, slotRangeEnd, keys);
|
for (let slot = slotRangeStart; slot <= slotRangeEnd; slot++) {
|
this.slots[slot] = keys;
|
}
|
}
|
this.connectionPool.reset(nodes);
|
callback();
|
}, this.options.slotsRefreshTimeout));
|
}
|
/**
|
* Check whether Cluster is able to process commands
|
*
|
* @param {Function} callback
|
* @private
|
*/
|
readyCheck(callback) {
|
this.cluster("info", function (err, res) {
|
if (err) {
|
return callback(err);
|
}
|
if (typeof res !== "string") {
|
return callback();
|
}
|
let state;
|
const lines = res.split("\r\n");
|
for (let i = 0; i < lines.length; ++i) {
|
const parts = lines[i].split(":");
|
if (parts[0] === "cluster_state") {
|
state = parts[1];
|
break;
|
}
|
}
|
if (state === "fail") {
|
debug("cluster state not ok (%s)", state);
|
callback(null, state);
|
}
|
else {
|
callback();
|
}
|
});
|
}
|
dnsLookup(hostname) {
|
return new Promise((resolve, reject) => {
|
this.options.dnsLookup(hostname, (err, address) => {
|
if (err) {
|
debug("failed to resolve hostname %s to IP: %s", hostname, err.message);
|
reject(err);
|
}
|
else {
|
debug("resolved hostname %s to IP %s", hostname, address);
|
resolve(address);
|
}
|
});
|
});
|
}
|
/**
|
* Normalize startup nodes, and resolving hostnames to IPs.
|
*
|
* This process happens every time when #connect() is called since
|
* #startupNodes and DNS records may chanage.
|
*
|
* @private
|
* @returns {Promise<IRedisOptions[]>}
|
*/
|
resolveStartupNodeHostnames() {
|
if (!Array.isArray(this.startupNodes) || this.startupNodes.length === 0) {
|
return Promise.reject(new Error("`startupNodes` should contain at least one node."));
|
}
|
const startupNodes = util_1.normalizeNodeOptions(this.startupNodes);
|
const hostnames = util_1.getUniqueHostnamesFromOptions(startupNodes);
|
if (hostnames.length === 0) {
|
return Promise.resolve(startupNodes);
|
}
|
return Promise.all(hostnames.map(hostname => this.dnsLookup(hostname))).then(ips => {
|
const hostnameToIP = utils_2.zipMap(hostnames, ips);
|
return startupNodes.map(node => hostnameToIP.has(node.host)
|
? Object.assign({}, node, { host: hostnameToIP.get(node.host) })
|
: node);
|
});
|
}
|
}
|
Object.getOwnPropertyNames(commander_1.default.prototype).forEach(name => {
|
if (!Cluster.prototype.hasOwnProperty(name)) {
|
Cluster.prototype[name] = commander_1.default.prototype[name];
|
}
|
});
|
const scanCommands = [
|
"sscan",
|
"hscan",
|
"zscan",
|
"sscanBuffer",
|
"hscanBuffer",
|
"zscanBuffer"
|
];
|
scanCommands.forEach(command => {
|
Cluster.prototype[command + "Stream"] = function (key, options) {
|
return new ScanStream_1.default(utils_1.defaults({
|
objectMode: true,
|
key: key,
|
redis: this,
|
command: command
|
}, options));
|
};
|
});
|
require("../transaction").addTransactionSupport(Cluster.prototype);
|
exports.default = Cluster;
|