@nestjs/microservices
Version:
Nest - modern, fast, powerful node.js web framework (@microservices)
134 lines (133 loc) • 5.13 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.ServerNats = void 0;
const shared_utils_1 = require("@nestjs/common/utils/shared.utils");
const constants_1 = require("../constants");
const nats_context_1 = require("../ctx-host/nats.context");
const nats_request_json_deserializer_1 = require("../deserializers/nats-request-json.deserializer");
const enums_1 = require("../enums");
const nats_record_serializer_1 = require("../serializers/nats-record.serializer");
const server_1 = require("./server");
let natsPackage = {};
class ServerNats extends server_1.Server {
constructor(options) {
super();
this.options = options;
this.transportId = enums_1.Transport.NATS;
natsPackage = this.loadPackage('nats', ServerNats.name, () => require('nats'));
this.initializeSerializer(options);
this.initializeDeserializer(options);
}
async listen(callback) {
try {
this.natsClient = await this.createNatsClient();
this.handleStatusUpdates(this.natsClient);
this.start(callback);
}
catch (err) {
callback(err);
}
}
start(callback) {
this.bindEvents(this.natsClient);
callback();
}
bindEvents(client) {
const queue = this.getOptionsProp(this.options, 'queue');
const subscribe = (channel) => client.subscribe(channel, {
queue,
callback: this.getMessageHandler(channel).bind(this),
});
const registeredPatterns = [...this.messageHandlers.keys()];
registeredPatterns.forEach(channel => subscribe(channel));
}
async close() {
await this.natsClient?.close();
this.natsClient = null;
}
createNatsClient() {
const options = this.options || {};
return natsPackage.connect({
servers: constants_1.NATS_DEFAULT_URL,
...options,
});
}
getMessageHandler(channel) {
return async (error, message) => {
if (error) {
return this.logger.error(error);
}
return this.handleMessage(channel, message);
};
}
async handleMessage(channel, natsMsg) {
const callerSubject = natsMsg.subject;
const rawMessage = natsMsg.data;
const replyTo = natsMsg.reply;
const natsCtx = new nats_context_1.NatsContext([callerSubject, natsMsg.headers]);
const message = await this.deserializer.deserialize(rawMessage, {
channel,
replyTo,
});
if ((0, shared_utils_1.isUndefined)(message.id)) {
return this.handleEvent(channel, message, natsCtx);
}
const publish = this.getPublisher(natsMsg, message.id);
const handler = this.getHandlerByPattern(channel);
if (!handler) {
const status = 'error';
const noHandlerPacket = {
id: message.id,
status,
err: constants_1.NO_MESSAGE_HANDLER,
};
return publish(noHandlerPacket);
}
const response$ = this.transformToObservable(await handler(message.data, natsCtx));
response$ && this.send(response$, publish);
}
getPublisher(natsMsg, id) {
if (natsMsg.reply) {
return (response) => {
Object.assign(response, { id });
const outgoingResponse = this.serializer.serialize(response);
return natsMsg.respond(outgoingResponse.data, {
headers: outgoingResponse.headers,
});
};
}
// In case the "reply" topic is not provided, there's no need for a reply.
// Method returns a noop function instead
// eslint-disable-next-line @typescript-eslint/no-empty-function
return () => { };
}
async handleStatusUpdates(client) {
for await (const status of client.status()) {
const data = status.data && (0, shared_utils_1.isObject)(status.data)
? JSON.stringify(status.data)
: status.data;
switch (status.type) {
case 'error':
case 'disconnect':
this.logger.error(`NatsError: type: "${status.type}", data: "${data}".`);
break;
case 'pingTimer':
if (this.options.debug) {
this.logger.debug(`NatsStatus: type: "${status.type}", data: "${data}".`);
}
break;
default:
this.logger.log(`NatsStatus: type: "${status.type}", data: "${data}".`);
break;
}
}
}
initializeSerializer(options) {
this.serializer = options?.serializer ?? new nats_record_serializer_1.NatsRecordSerializer();
}
initializeDeserializer(options) {
this.deserializer =
options?.deserializer ?? new nats_request_json_deserializer_1.NatsRequestJSONDeserializer();
}
}
exports.ServerNats = ServerNats;