/*
* Copyright DataStax, Inc.
*
* Licensed 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.
*/
'use strict';
const util = require('util');
const errors = require('../errors');
/** @module policies/speculativeExecution */
/**
* @classdesc
* The policy that decides if the driver will send speculative queries to the next hosts when the current host takes too
* long to respond.
*
Note that only idempotent statements will be speculatively retried.
* @constructor
* @abstract
*/
function SpeculativeExecutionPolicy() {
}
/**
* Initialization method that gets invoked on Client startup.
* @param {Client} client
* @abstract
*/
SpeculativeExecutionPolicy.prototype.init = function (client) {
};
/**
* Gets invoked at client shutdown, giving the opportunity to the implementor to perform cleanup.
* @abstract
*/
SpeculativeExecutionPolicy.prototype.shutdown = function () {
};
/**
* Gets the plan to use for a new query.
* Returns an object with a nextExecution()
method, which returns a positive number representing the
* amount of milliseconds to delay the next execution or a non-negative number to avoid further executions.
* @param {String} keyspace The currently logged keyspace.
* @param {String|Array} queryInfo The query, or queries in the case of batches, for which to build a plan.
* @return {{nextExecution: function}}
* @abstract
*/
SpeculativeExecutionPolicy.prototype.newPlan = function (keyspace, queryInfo) {
throw new Error('You must implement newPlan() method in the SpeculativeExecutionPolicy');
};
/**
* Gets an associative array containing the policy options.
*/
SpeculativeExecutionPolicy.prototype.getOptions = function () {
return new Map();
};
/**
* Creates a new instance of NoSpeculativeExecutionPolicy.
* @classdesc
* A {@link SpeculativeExecutionPolicy} that never schedules speculative executions.
* @constructor
* @extends {SpeculativeExecutionPolicy}
*/
function NoSpeculativeExecutionPolicy() {
this._plan = {
nextExecution: function () {
return -1;
}
};
}
util.inherits(NoSpeculativeExecutionPolicy, SpeculativeExecutionPolicy);
NoSpeculativeExecutionPolicy.prototype.newPlan = function () {
return this._plan;
};
/**
* Creates a new instance of ConstantSpeculativeExecutionPolicy.
* @classdesc
* A {@link SpeculativeExecutionPolicy} that schedules a given number of speculative executions,
* separated by a fixed delay.
* @constructor
* @param {Number} delay The delay between each speculative execution.
* @param {Number} maxSpeculativeExecutions The amount of speculative executions that should be scheduled after the
* initial execution. Must be strictly positive.
* @extends {SpeculativeExecutionPolicy}
*/
function ConstantSpeculativeExecutionPolicy(delay, maxSpeculativeExecutions) {
if (!(delay >= 0)) {
throw new errors.ArgumentError('delay must be a positive number or zero');
}
if (!(maxSpeculativeExecutions > 0)) {
throw new errors.ArgumentError('maxSpeculativeExecutions must be a positive number');
}
this._delay = delay;
this._maxSpeculativeExecutions = maxSpeculativeExecutions;
}
util.inherits(ConstantSpeculativeExecutionPolicy, SpeculativeExecutionPolicy);
ConstantSpeculativeExecutionPolicy.prototype.newPlan = function () {
let executions = 0;
const self = this;
return {
nextExecution: function () {
if (executions++ < self._maxSpeculativeExecutions) {
return self._delay;
}
return -1;
}
};
};
/**
* Gets an associative array containing the policy options.
*/
ConstantSpeculativeExecutionPolicy.prototype.getOptions = function () {
return new Map([
['delay', this._delay ],
['maxSpeculativeExecutions', this._maxSpeculativeExecutions ]
]);
};
exports.NoSpeculativeExecutionPolicy = NoSpeculativeExecutionPolicy;
exports.SpeculativeExecutionPolicy = SpeculativeExecutionPolicy;
exports.ConstantSpeculativeExecutionPolicy = ConstantSpeculativeExecutionPolicy;