hazelcast-client
Version:
Hazelcast - open source In-Memory Data Grid - client for NodeJS
303 lines • 12.9 kB
JavaScript
"use strict";
/*
* Copyright (c) 2008-2018, Hazelcast, Inc. All Rights Reserved.
*
* 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.
*/
Object.defineProperty(exports, "__esModule", { value: true });
var ClientMessage = require("../ClientMessage");
var Long = require("long");
var Promise = require("bluebird");
var BitsUtil_1 = require("../BitsUtil");
var LoggingService_1 = require("../logging/LoggingService");
var HazelcastError_1 = require("../HazelcastError");
var assert = require("assert");
var EXCEPTION_MESSAGE_TYPE = 109;
var MAX_FAST_INVOCATION_COUNT = 5;
var PROPERTY_INVOCATION_RETRY_PAUSE_MILLIS = 'hazelcast.client.invocation.retry.pause.millis';
var PROPERTY_INVOCATION_TIMEOUT_MILLIS = 'hazelcast.client.invocation.timeout.millis';
/**
* A request to be sent to a hazelcast node.
*/
var Invocation = /** @class */ (function () {
function Invocation(client, request) {
this.invokeCount = 0;
this.client = client;
this.invocationService = client.getInvocationService();
this.deadline = new Date(new Date().getTime() + this.invocationService.getInvocationTimeoutMillis());
this.request = request;
}
/**
* @returns {boolean}
*/
Invocation.prototype.hasPartitionId = function () {
return this.hasOwnProperty('partitionId') && this.partitionId >= 0;
};
Invocation.prototype.isAllowedToRetryOnSelection = function (err) {
return (this.connection == null && this.address == null) || !(err instanceof HazelcastError_1.IOError);
};
Invocation.isRetrySafeError = function (err) {
return err instanceof HazelcastError_1.IOError || err instanceof HazelcastError_1.HazelcastInstanceNotActiveError || err instanceof HazelcastError_1.RetryableHazelcastError;
};
return Invocation;
}());
exports.Invocation = Invocation;
/**
* Sends requests to appropriate nodes. Resolves waiting promises with responses.
*/
var InvocationService = /** @class */ (function () {
function InvocationService(hazelcastClient) {
this.correlationCounter = 1;
this.eventHandlers = {};
this.pending = {};
this.logger = LoggingService_1.LoggingService.getLoggingService();
this.client = hazelcastClient;
this.smartRoutingEnabled = hazelcastClient.getConfig().networkConfig.smartRouting;
if (hazelcastClient.getConfig().networkConfig.smartRouting) {
this.doInvoke = this.invokeSmart;
}
else {
this.doInvoke = this.invokeNonSmart;
}
this.invocationRetryPauseMillis = this.client.getConfig().properties[PROPERTY_INVOCATION_RETRY_PAUSE_MILLIS];
this.invocationTimeoutMillis = this.client.getConfig().properties[PROPERTY_INVOCATION_TIMEOUT_MILLIS];
this.isShutdown = false;
}
InvocationService.prototype.shutdown = function () {
this.isShutdown = true;
};
InvocationService.prototype.invoke = function (invocation) {
var newCorrelationId = Long.fromNumber(this.correlationCounter++);
invocation.deferred = Promise.defer();
invocation.request.setCorrelationId(newCorrelationId);
this.doInvoke(invocation);
return invocation.deferred.promise;
};
/**
* Invokes given invocation on specified connection.
* @param connection
* @param request
* @param handler called with values returned from server for this invocation.
* @returns
*/
InvocationService.prototype.invokeOnConnection = function (connection, request, handler) {
var invocation = new Invocation(this.client, request);
invocation.connection = connection;
if (handler) {
invocation.handler = handler;
}
return this.invoke(invocation);
};
/**
* Invokes given invocation on the node that owns given partition.
* @param request
* @param partitionId
* @returns
*/
InvocationService.prototype.invokeOnPartition = function (request, partitionId) {
var invocation = new Invocation(this.client, request);
invocation.partitionId = partitionId;
return this.invoke(invocation);
};
/**
* Invokes given invocation on the host with given address.
* @param request
* @param target
* @returns
*/
InvocationService.prototype.invokeOnTarget = function (request, target) {
var invocation = new Invocation(this.client, request);
invocation.address = target;
return this.invoke(invocation);
};
/**
* Invokes given invocation on any host.
* Useful when an operation is not bound to any host but a generic operation.
* @param request
* @returns
*/
InvocationService.prototype.invokeOnRandomTarget = function (request) {
return this.invoke(new Invocation(this.client, request));
};
InvocationService.prototype.getInvocationTimeoutMillis = function () {
return this.invocationTimeoutMillis;
};
InvocationService.prototype.getInvocationRetryPauseMillis = function () {
return this.invocationRetryPauseMillis;
};
InvocationService.prototype.invokeSmart = function (invocation) {
var _this = this;
var invocationPromise;
invocation.invokeCount++;
if (invocation.hasOwnProperty('connection')) {
invocationPromise = this.send(invocation, invocation.connection);
}
else if (invocation.hasPartitionId()) {
invocationPromise = this.invokeOnPartitionOwner(invocation, invocation.partitionId);
}
else if (invocation.hasOwnProperty('address')) {
invocationPromise = this.invokeOnAddress(invocation, invocation.address);
}
else {
invocationPromise = this.invokeOnOwner(invocation);
}
invocationPromise.catch(function (err) {
_this.notifyError(invocation, err);
});
};
InvocationService.prototype.invokeNonSmart = function (invocation) {
var _this = this;
var invocationPromise;
invocation.invokeCount++;
if (invocation.hasOwnProperty('connection')) {
invocationPromise = this.send(invocation, invocation.connection);
}
else {
invocationPromise = this.invokeOnOwner(invocation);
}
invocationPromise.catch(function (err) {
_this.notifyError(invocation, err);
});
};
InvocationService.prototype.invokeOnOwner = function (invocation) {
var owner = this.client.getClusterService().getOwnerConnection();
if (owner == null) {
return Promise.reject(new HazelcastError_1.IOError('Unisocket client\'s owner connection is not available.'));
}
return this.send(invocation, owner);
};
InvocationService.prototype.invokeOnAddress = function (invocation, address) {
var _this = this;
return this.client.getConnectionManager().getOrConnect(address).then(function (connection) {
if (connection == null) {
throw new Error(address.toString() + ' is not available.');
}
return _this.send(invocation, connection);
});
};
InvocationService.prototype.invokeOnPartitionOwner = function (invocation, partitionId) {
var _this = this;
var ownerAddress = this.client.getPartitionService().getAddressForPartition(partitionId);
return this.client.getConnectionManager().getOrConnect(ownerAddress).then(function (connection) {
if (connection == null) {
throw new HazelcastError_1.IOError(ownerAddress.toString() + '(partition owner) is not available.');
}
return _this.send(invocation, connection);
});
};
InvocationService.prototype.send = function (invocation, connection) {
assert(connection != null);
if (this.isShutdown) {
return Promise.reject(new HazelcastError_1.ClientNotActiveError('Client is shutdown.'));
}
this.registerInvocation(invocation);
return this.write(invocation, connection);
};
InvocationService.prototype.write = function (invocation, connection) {
return connection.write(invocation.request.getBuffer());
};
InvocationService.prototype.notifyError = function (invocation, error) {
var correlationId = invocation.request.getCorrelationId().toNumber();
if (this.rejectIfNotRetryable(invocation, error)) {
delete this.pending[invocation.request.getCorrelationId().toNumber()];
return;
}
this.logger.debug('InvocationService', 'Retrying(' + invocation.invokeCount + ') on correlation-id=' + correlationId, error);
if (invocation.invokeCount < MAX_FAST_INVOCATION_COUNT) {
this.doInvoke(invocation);
}
else {
setTimeout(this.doInvoke.bind(this, invocation), this.getInvocationRetryPauseMillis());
}
};
/**
* Determines if an error is retryable. The given invocation is rejected with approprate error if the error is not retryable.
* @param invocation
* @param error
* @returns `true` if invocation is rejected, `false` otherwise
*/
InvocationService.prototype.rejectIfNotRetryable = function (invocation, error) {
if (!this.client.getLifecycleService().isRunning()) {
invocation.deferred.reject(new HazelcastError_1.ClientNotActiveError('Client is not active.', error));
return true;
}
if (!invocation.isAllowedToRetryOnSelection(error)) {
invocation.deferred.reject(error);
return true;
}
if (!Invocation.isRetrySafeError(error)) {
invocation.deferred.reject(error);
return true;
}
if (invocation.deadline.getTime() < Date.now()) {
this.logger.trace('InvocationService', 'Error will not be retried because invocation timed out');
invocation.deferred.reject(new HazelcastError_1.InvocationTimeoutError('Invocation ' + invocation.request.getCorrelationId() + ')'
+ ' reached its deadline.', error));
return true;
}
};
InvocationService.prototype.registerInvocation = function (invocation) {
var message = invocation.request;
var correlationId = message.getCorrelationId().toNumber();
if (invocation.hasPartitionId()) {
message.setPartitionId(invocation.partitionId);
}
else {
message.setPartitionId(-1);
}
if (invocation.hasOwnProperty('handler')) {
this.eventHandlers[correlationId] = invocation;
}
this.pending[correlationId] = invocation;
};
/**
* Removes the handler for all event handlers with a specific correlation id.
* @param id correlation id
*/
InvocationService.prototype.removeEventHandler = function (id) {
if (this.eventHandlers.hasOwnProperty('' + id)) {
delete this.eventHandlers[id];
}
};
/**
* Extract codec specific properties in a protocol message and resolves waiting promise.
* @param buffer
*/
InvocationService.prototype.processResponse = function (buffer) {
var _this = this;
var clientMessage = new ClientMessage(buffer);
var correlationId = clientMessage.getCorrelationId().toNumber();
var messageType = clientMessage.getMessageType();
if (clientMessage.hasFlags(BitsUtil_1.BitsUtil.LISTENER_FLAG)) {
setImmediate(function () {
if (_this.eventHandlers[correlationId] !== undefined) {
_this.eventHandlers[correlationId].handler(clientMessage);
}
});
return;
}
var pendingInvocation = this.pending[correlationId];
var deferred = pendingInvocation.deferred;
if (messageType === EXCEPTION_MESSAGE_TYPE) {
var remoteError = this.client.getErrorFactory().createErrorFromClientMessage(clientMessage);
this.notifyError(pendingInvocation, remoteError);
}
else {
delete this.pending[correlationId];
deferred.resolve(clientMessage);
}
};
return InvocationService;
}());
exports.InvocationService = InvocationService;
//# sourceMappingURL=InvocationService.js.map