mirror of
https://github.com/nestjs/nest.git
synced 2026-02-21 15:08:37 +00:00
277 lines
7.9 KiB
TypeScript
277 lines
7.9 KiB
TypeScript
import { isObject, isUndefined } from '@nestjs/common/utils/shared.utils';
|
|
import { EventEmitter } from 'events';
|
|
import {
|
|
NATS_DEFAULT_GRACE_PERIOD,
|
|
NATS_DEFAULT_URL,
|
|
NO_MESSAGE_HANDLER,
|
|
} from '../constants';
|
|
import { NatsContext } from '../ctx-host/nats.context';
|
|
import { NatsRequestJSONDeserializer } from '../deserializers/nats-request-json.deserializer';
|
|
import { Transport } from '../enums';
|
|
import { NatsEvents, NatsEventsMap, NatsStatus } from '../events/nats.events';
|
|
import { NatsOptions } from '../interfaces/microservice-configuration.interface';
|
|
import { IncomingRequest } from '../interfaces/packet.interface';
|
|
import { NatsRecord } from '../record-builders';
|
|
import { NatsRecordSerializer } from '../serializers/nats-record.serializer';
|
|
import { Server } from './server';
|
|
|
|
let natsPackage = {} as any;
|
|
|
|
// To enable type safety for Nats. This cant be uncommented by default
|
|
// because it would require the user to install the nats package even if they dont use Nats
|
|
// Otherwise, TypeScript would fail to compile the code.
|
|
//
|
|
// type Client = import('nats').NatsConnection;
|
|
// type NatsMsg = import('nats').Msg;
|
|
// type Subscription = import('nats').Subscription;
|
|
|
|
type Client = any;
|
|
type NatsMsg = any;
|
|
type Subscription = any;
|
|
|
|
/**
|
|
* @publicApi
|
|
*/
|
|
export class ServerNats<
|
|
E extends NatsEvents = NatsEvents,
|
|
S extends NatsStatus = NatsStatus,
|
|
> extends Server<E, S> {
|
|
public readonly transportId = Transport.NATS;
|
|
|
|
private natsClient: Client;
|
|
protected statusEventEmitter = new EventEmitter<{
|
|
[key in keyof NatsEvents]: Parameters<NatsEvents[key]>;
|
|
}>();
|
|
private readonly subscriptions: Subscription[] = [];
|
|
|
|
constructor(private readonly options: Required<NatsOptions>['options']) {
|
|
super();
|
|
|
|
natsPackage = this.loadPackage('nats', ServerNats.name, () =>
|
|
require('nats'),
|
|
);
|
|
|
|
this.initializeSerializer(options);
|
|
this.initializeDeserializer(options);
|
|
}
|
|
|
|
public async listen(
|
|
callback: (err?: unknown, ...optionalParams: unknown[]) => void,
|
|
) {
|
|
try {
|
|
this.natsClient = await this.createNatsClient();
|
|
|
|
this._status$.next(NatsStatus.CONNECTED as S);
|
|
void this.handleStatusUpdates(this.natsClient);
|
|
this.start(callback);
|
|
} catch (err) {
|
|
callback(err);
|
|
}
|
|
}
|
|
|
|
public start(
|
|
callback: (err?: unknown, ...optionalParams: unknown[]) => void,
|
|
) {
|
|
this.bindEvents(this.natsClient);
|
|
callback();
|
|
}
|
|
|
|
public bindEvents(client: Client) {
|
|
const queue = this.getOptionsProp(this.options, 'queue');
|
|
const subscribe = (channel: string) =>
|
|
client.subscribe(channel, {
|
|
queue,
|
|
callback: this.getMessageHandler(channel).bind(this),
|
|
});
|
|
|
|
const registeredPatterns = [...this.messageHandlers.keys()];
|
|
for (const channel of registeredPatterns) {
|
|
const sub = subscribe(channel);
|
|
this.subscriptions.push(sub);
|
|
}
|
|
}
|
|
|
|
private async waitForGracePeriod() {
|
|
const gracePeriod = this.getOptionsProp(
|
|
this.options,
|
|
'gracePeriod',
|
|
NATS_DEFAULT_GRACE_PERIOD,
|
|
);
|
|
await new Promise<void>(res => {
|
|
setTimeout(() => {
|
|
res();
|
|
}, gracePeriod);
|
|
});
|
|
}
|
|
|
|
public async close() {
|
|
if (!this.natsClient) {
|
|
return;
|
|
}
|
|
const graceful = this.getOptionsProp(this.options, 'gracefulShutdown');
|
|
if (graceful) {
|
|
this.subscriptions.forEach(sub => sub.unsubscribe());
|
|
await this.waitForGracePeriod();
|
|
}
|
|
await this.natsClient?.close();
|
|
this.statusEventEmitter.removeAllListeners();
|
|
this.natsClient = null;
|
|
}
|
|
|
|
public createNatsClient(): Promise<Client> {
|
|
const options = this.options || ({} as NatsOptions);
|
|
return natsPackage.connect({
|
|
servers: NATS_DEFAULT_URL,
|
|
...options,
|
|
});
|
|
}
|
|
|
|
public getMessageHandler(channel: string): Function {
|
|
return async (error: object | undefined, message: NatsMsg) => {
|
|
if (error) {
|
|
return this.logger.error(error);
|
|
}
|
|
return this.handleMessage(channel, message);
|
|
};
|
|
}
|
|
|
|
public async handleMessage(channel: string, natsMsg: NatsMsg) {
|
|
const callerSubject = natsMsg.subject;
|
|
const rawMessage = natsMsg.data;
|
|
const replyTo = natsMsg.reply;
|
|
|
|
const natsCtx = new NatsContext([callerSubject, natsMsg.headers]);
|
|
const message = await this.deserializer.deserialize(rawMessage, {
|
|
channel,
|
|
replyTo,
|
|
});
|
|
if (isUndefined((message as IncomingRequest).id)) {
|
|
return this.handleEvent(channel, message, natsCtx);
|
|
}
|
|
const publish = this.getPublisher(natsMsg, (message as IncomingRequest).id);
|
|
const handler = this.getHandlerByPattern(channel);
|
|
if (!handler) {
|
|
const status = 'error';
|
|
const noHandlerPacket = {
|
|
id: (message as IncomingRequest).id,
|
|
status,
|
|
err: NO_MESSAGE_HANDLER,
|
|
};
|
|
return publish(noHandlerPacket);
|
|
}
|
|
const response$ = this.transformToObservable(
|
|
await handler(message.data, natsCtx),
|
|
);
|
|
response$ && this.send(response$, publish);
|
|
}
|
|
|
|
public getPublisher(natsMsg: NatsMsg, id: string) {
|
|
if (natsMsg.reply) {
|
|
return (response: any) => {
|
|
Object.assign(response, { id });
|
|
const outgoingResponse: NatsRecord =
|
|
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
|
|
|
|
return () => {};
|
|
}
|
|
|
|
public async handleStatusUpdates(client: Client) {
|
|
for await (const status of client.status()) {
|
|
const data =
|
|
status.data && isObject(status.data)
|
|
? JSON.stringify(status.data)
|
|
: status.data;
|
|
|
|
switch (status.type) {
|
|
case 'error':
|
|
this.logger.error(
|
|
`NatsError: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
break;
|
|
|
|
case 'disconnect':
|
|
this.logger.error(
|
|
`NatsError: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
|
|
this._status$.next(NatsStatus.DISCONNECTED as S);
|
|
this.statusEventEmitter.emit(
|
|
NatsEventsMap.DISCONNECT,
|
|
status.data as string,
|
|
);
|
|
break;
|
|
|
|
case 'pingTimer':
|
|
if (this.options.debug) {
|
|
this.logger.debug!(
|
|
`NatsStatus: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
}
|
|
break;
|
|
|
|
case 'reconnecting':
|
|
this._status$.next(NatsStatus.RECONNECTING as S);
|
|
break;
|
|
|
|
case 'reconnect':
|
|
this.logger.log(
|
|
`NatsStatus: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
|
|
this._status$.next(NatsStatus.CONNECTED as S);
|
|
this.statusEventEmitter.emit(
|
|
NatsEventsMap.RECONNECT,
|
|
status.data as string,
|
|
);
|
|
break;
|
|
|
|
case 'update':
|
|
this.logger.log(
|
|
`NatsStatus: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
this.statusEventEmitter.emit(NatsEventsMap.UPDATE, status.data);
|
|
break;
|
|
|
|
default:
|
|
this.logger.log(
|
|
`NatsStatus: type: "${status.type}", data: "${data}".`,
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
public unwrap<T>(): T {
|
|
if (!this.natsClient) {
|
|
throw new Error(
|
|
'Not initialized. Please call the "listen"/"startAllMicroservices" method before accessing the server.',
|
|
);
|
|
}
|
|
return this.natsClient as T;
|
|
}
|
|
|
|
public on<
|
|
EventKey extends keyof E = keyof E,
|
|
EventCallback extends E[EventKey] = E[EventKey],
|
|
>(event: EventKey, callback: EventCallback) {
|
|
this.statusEventEmitter.on(event, callback as any);
|
|
}
|
|
|
|
protected initializeSerializer(options: NatsOptions['options']) {
|
|
this.serializer = options?.serializer ?? new NatsRecordSerializer();
|
|
}
|
|
|
|
protected initializeDeserializer(options: NatsOptions['options']) {
|
|
this.deserializer =
|
|
options?.deserializer ?? new NatsRequestJSONDeserializer();
|
|
}
|
|
}
|