@nestjs/microservices
Version:
Nest - modern, fast, powerful node.js web framework (@microservices)
105 lines (104 loc) • 4.9 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.MessagePattern = exports.GrpcMethodStreamingType = void 0;
exports.GrpcMethod = GrpcMethod;
exports.GrpcStreamMethod = GrpcStreamMethod;
exports.GrpcStreamCall = GrpcStreamCall;
exports.createGrpcMethodMetadata = createGrpcMethodMetadata;
const shared_utils_1 = require("@nestjs/common/utils/shared.utils");
const constants_1 = require("../constants");
const enums_1 = require("../enums");
const pattern_handler_enum_1 = require("../enums/pattern-handler.enum");
const invalid_grpc_message_decorator_exception_1 = require("../errors/invalid-grpc-message-decorator.exception");
var GrpcMethodStreamingType;
(function (GrpcMethodStreamingType) {
GrpcMethodStreamingType["NO_STREAMING"] = "no_stream";
GrpcMethodStreamingType["RX_STREAMING"] = "rx_stream";
GrpcMethodStreamingType["PT_STREAMING"] = "pt_stream";
})(GrpcMethodStreamingType || (exports.GrpcMethodStreamingType = GrpcMethodStreamingType = {}));
/**
* Subscribes to incoming messages which fulfils chosen pattern.
*
* @publicApi
*/
const MessagePattern = (metadata, transportOrExtras, maybeExtras) => {
let transport;
let extras;
if (((0, shared_utils_1.isNumber)(transportOrExtras) || (0, shared_utils_1.isSymbol)(transportOrExtras)) &&
(0, shared_utils_1.isNil)(maybeExtras)) {
transport = transportOrExtras;
}
else if ((0, shared_utils_1.isObject)(transportOrExtras) && (0, shared_utils_1.isNil)(maybeExtras)) {
extras = transportOrExtras;
}
else {
transport = transportOrExtras;
extras = maybeExtras;
}
return (target, key, descriptor) => {
try {
Reflect.defineMetadata(constants_1.PATTERN_METADATA, [].concat(metadata), descriptor.value);
Reflect.defineMetadata(constants_1.PATTERN_HANDLER_METADATA, pattern_handler_enum_1.PatternHandler.MESSAGE, descriptor.value);
Reflect.defineMetadata(constants_1.TRANSPORT_METADATA, transport, descriptor.value);
Reflect.defineMetadata(constants_1.PATTERN_EXTRAS_METADATA, {
...Reflect.getMetadata(constants_1.PATTERN_EXTRAS_METADATA, descriptor.value),
...extras,
}, descriptor.value);
return descriptor;
}
catch (err) {
throw new invalid_grpc_message_decorator_exception_1.InvalidGrpcDecoratorException(metadata);
}
};
};
exports.MessagePattern = MessagePattern;
function GrpcMethod(service, method) {
return (target, key, descriptor) => {
const metadata = createGrpcMethodMetadata(target, key, service, method);
return (0, exports.MessagePattern)(metadata, enums_1.Transport.GRPC)(target, key, descriptor);
};
}
function GrpcStreamMethod(service, method) {
return (target, key, descriptor) => {
const metadata = createGrpcMethodMetadata(target, key, service, method, GrpcMethodStreamingType.RX_STREAMING);
(0, exports.MessagePattern)(metadata, enums_1.Transport.GRPC)(target, key, descriptor);
const originalMethod = descriptor.value;
// Override original method to call the "drainBuffer" method on the first parameter
// This is required to avoid premature message emission
descriptor.value = async function (observable, ...args) {
const result = await Promise.resolve(originalMethod.apply(this, [observable, ...args]));
// Drain buffer if "drainBuffer" method is available
if (observable && observable.drainBuffer) {
observable.drainBuffer();
}
return result;
};
// Copy all metadata from the original method to the new one
const metadataKeys = Reflect.getMetadataKeys(originalMethod);
metadataKeys.forEach(metadataKey => {
const metadataValue = Reflect.getMetadata(metadataKey, originalMethod);
Reflect.defineMetadata(metadataKey, metadataValue, descriptor.value);
});
};
}
function GrpcStreamCall(service, method) {
return (target, key, descriptor) => {
const metadata = createGrpcMethodMetadata(target, key, service, method, GrpcMethodStreamingType.PT_STREAMING);
return (0, exports.MessagePattern)(metadata, enums_1.Transport.GRPC)(target, key, descriptor);
};
}
function createGrpcMethodMetadata(target, key, service, method, streaming = GrpcMethodStreamingType.NO_STREAMING) {
const capitalizeFirstLetter = (str) => str.charAt(0).toUpperCase() + str.slice(1);
if (!service) {
const { name } = target.constructor;
return {
service: name,
rpc: capitalizeFirstLetter(key),
streaming,
};
}
if (service && !method) {
return { service, rpc: capitalizeFirstLetter(key), streaming };
}
return { service, rpc: method, streaming };
}
;