@nestjs/microservices
Version:
Nest - modern, fast, powerful node.js web framework (@microservices)
140 lines (139 loc) • 5.59 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.ServerTCP = void 0;
const shared_utils_1 = require("@nestjs/common/utils/shared.utils");
const net = require("net");
const tls_1 = require("tls");
const constants_1 = require("../constants");
const tcp_context_1 = require("../ctx-host/tcp.context");
const enums_1 = require("../enums");
const helpers_1 = require("../helpers");
const server_1 = require("./server");
/**
* @publicApi
*/
class ServerTCP extends server_1.Server {
constructor(options) {
super();
this.options = options;
this.transportId = enums_1.Transport.TCP;
this.isManuallyTerminated = false;
this.retryAttemptsCount = 0;
this.pendingEventListeners = [];
this.port = this.getOptionsProp(options, 'port', constants_1.TCP_DEFAULT_PORT);
this.host = this.getOptionsProp(options, 'host', constants_1.TCP_DEFAULT_HOST);
this.socketClass = this.getOptionsProp(options, 'socketClass', helpers_1.JsonSocket);
this.tlsOptions = this.getOptionsProp(options, 'tlsOptions');
this.init();
this.initializeSerializer(options);
this.initializeDeserializer(options);
}
listen(callback) {
this.server.once("error" /* TcpEventsMap.ERROR */, (err) => {
if (err?.code === constants_1.EADDRINUSE || err?.code === constants_1.ECONNREFUSED) {
this._status$.next("disconnected" /* TcpStatus.DISCONNECTED */);
return callback(err);
}
});
this.server.listen(this.port, this.host, callback);
}
close() {
this.isManuallyTerminated = true;
this.server.close();
this.pendingEventListeners = [];
}
bindHandler(socket) {
const readSocket = this.getSocketInstance(socket);
readSocket.on('message', async (msg) => this.handleMessage(readSocket, msg));
readSocket.on("error" /* TcpEventsMap.ERROR */, this.handleError.bind(this));
}
async handleMessage(socket, rawMessage) {
const packet = await this.deserializer.deserialize(rawMessage);
const pattern = !(0, shared_utils_1.isString)(packet.pattern)
? JSON.stringify(packet.pattern)
: packet.pattern;
const tcpContext = new tcp_context_1.TcpContext([socket, pattern]);
if ((0, shared_utils_1.isUndefined)(packet.id)) {
return this.handleEvent(pattern, packet, tcpContext);
}
const handler = this.getHandlerByPattern(pattern);
if (!handler) {
const status = 'error';
const noHandlerPacket = this.serializer.serialize({
id: packet.id,
status,
err: constants_1.NO_MESSAGE_HANDLER,
});
return socket.sendMessage(noHandlerPacket);
}
const response$ = this.transformToObservable(await handler(packet.data, tcpContext));
response$ &&
this.send(response$, data => {
Object.assign(data, { id: packet.id });
const outgoingResponse = this.serializer.serialize(data);
socket.sendMessage(outgoingResponse);
});
}
handleClose() {
if (this.isManuallyTerminated ||
!this.getOptionsProp(this.options, 'retryAttempts') ||
this.retryAttemptsCount >=
this.getOptionsProp(this.options, 'retryAttempts', 0)) {
return undefined;
}
++this.retryAttemptsCount;
return setTimeout(() => this.server.listen(this.port, this.host), this.getOptionsProp(this.options, 'retryDelay', 0));
}
unwrap() {
if (!this.server) {
throw new Error('Not initialized. Please call the "listen"/"startAllMicroservices" method before accessing the server.');
}
return this.server;
}
on(event, callback) {
if (this.server) {
this.server.on(event, callback);
}
else {
this.pendingEventListeners.push({ event, callback });
}
}
init() {
if (this.tlsOptions) {
// TLS enabled, use tls server
this.server = (0, tls_1.createServer)(this.tlsOptions, this.bindHandler.bind(this));
}
else {
// TLS disabled, use net server
this.server = net.createServer(this.bindHandler.bind(this));
}
this.registerListeningListener(this.server);
this.registerErrorListener(this.server);
this.registerCloseListener(this.server);
this.pendingEventListeners.forEach(({ event, callback }) => this.server.on(event, callback));
this.pendingEventListeners = [];
}
registerListeningListener(socket) {
socket.on("listening" /* TcpEventsMap.LISTENING */, () => {
this._status$.next("connected" /* TcpStatus.CONNECTED */);
});
}
registerErrorListener(socket) {
socket.on("error" /* TcpEventsMap.ERROR */, err => {
if ('code' in err && err.code === constants_1.ECONNREFUSED) {
this._status$.next("disconnected" /* TcpStatus.DISCONNECTED */);
}
this.handleError(err);
});
}
registerCloseListener(socket) {
socket.on("close" /* TcpEventsMap.CLOSE */, () => {
this._status$.next("disconnected" /* TcpStatus.DISCONNECTED */);
this.handleClose();
});
}
getSocketInstance(socket) {
return new this.socketClass(socket);
}
}
exports.ServerTCP = ServerTCP;