@nestjs/microservices
Version:
Nest - modern, fast, powerful node.js web framework (@microservices)
131 lines (130 loc) • 4.4 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.ClientTCP = void 0;
const common_1 = require("@nestjs/common");
const net = require("net");
const rxjs_1 = require("rxjs");
const operators_1 = require("rxjs/operators");
const constants_1 = require("../constants");
const helpers_1 = require("../helpers");
const tls_1 = require("tls");
const client_proxy_1 = require("./client-proxy");
/**
* @publicApi
*/
class ClientTCP extends client_proxy_1.ClientProxy {
constructor(options) {
super();
this.logger = new common_1.Logger(ClientTCP.name);
this.isConnected = false;
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.initializeSerializer(options);
this.initializeDeserializer(options);
}
connect() {
if (this.connection) {
return this.connection;
}
this.socket = this.createSocket();
this.bindEvents(this.socket);
const source$ = this.connect$(this.socket.netSocket).pipe((0, operators_1.tap)(() => {
this.isConnected = true;
this.socket.on(constants_1.MESSAGE_EVENT, (buffer) => this.handleResponse(buffer));
}), (0, operators_1.share)());
// For TLS connections, the connection is initiated when the socket is created
if (!this.tlsOptions) {
this.socket.connect(this.port, this.host);
}
this.connection = (0, rxjs_1.lastValueFrom)(source$).catch(err => {
if (err instanceof rxjs_1.EmptyError) {
return;
}
throw err;
});
return this.connection;
}
async handleResponse(buffer) {
const { err, response, isDisposed, id } = await this.deserializer.deserialize(buffer);
const callback = this.routingMap.get(id);
if (!callback) {
return undefined;
}
if (isDisposed || err) {
return callback({
err,
response,
isDisposed: true,
});
}
callback({
err,
response,
});
}
createSocket() {
let socket;
/**
* TLS enabled, "upgrade" the TCP Socket to TLS
*/
if (this.tlsOptions) {
socket = (0, tls_1.connect)({
...this.tlsOptions,
port: this.port,
host: this.host,
socket,
});
}
else {
socket = new net.Socket();
}
return new this.socketClass(socket);
}
close() {
this.socket && this.socket.end();
this.handleClose();
}
bindEvents(socket) {
socket.on(constants_1.ERROR_EVENT, (err) => err.code !== constants_1.ECONNREFUSED && this.handleError(err));
socket.on(constants_1.CLOSE_EVENT, () => this.handleClose());
}
handleError(err) {
this.logger.error(err);
}
handleClose() {
this.isConnected = false;
this.socket = null;
this.connection = undefined;
if (this.routingMap.size > 0) {
const err = new Error('Connection closed');
for (const callback of this.routingMap.values()) {
callback({ err });
}
this.routingMap.clear();
}
}
publish(partialPacket, callback) {
try {
const packet = this.assignPacketId(partialPacket);
const serializedPacket = this.serializer.serialize(packet);
this.routingMap.set(packet.id, callback);
this.socket.sendMessage(serializedPacket);
return () => this.routingMap.delete(packet.id);
}
catch (err) {
callback({ err });
}
}
async dispatchEvent(packet) {
const pattern = this.normalizePattern(packet.pattern);
const serializedPacket = this.serializer.serialize({
...packet,
pattern,
});
return this.socket.sendMessage(serializedPacket);
}
}
exports.ClientTCP = ClientTCP;