@decorators/server
Version:
node decorators - decorators for express library
180 lines • 26 kB
JavaScript
;
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __param = (this && this.__param) || function (paramIndex, decorator) {
return function (target, key) { decorator(target, key, paramIndex); }
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.SwaggerDocument = exports.DEFAULT_METHOD = exports.DEFAULT_STATUS = void 0;
const di_1 = require("@decorators/di");
const core_1 = require("../../../../core");
const http_1 = require("../../../http");
const sockets_1 = require("../../../sockets");
const constants_1 = require("../constants");
const utils_1 = require("./utils");
exports.DEFAULT_STATUS = 200;
exports.DEFAULT_METHOD = http_1.HttpMethodType.POST;
let SwaggerDocument = exports.SwaggerDocument = class SwaggerDocument {
constructor(appVersion, config, metadataScanner, reflector) {
this.appVersion = appVersion;
this.config = config;
this.metadataScanner = metadataScanner;
this.reflector = reflector;
this.paths = {};
this.schemas = {};
this.securitySchemas = {};
}
generate() {
this.processMetadata();
return {
components: {
schemas: this.schemas,
securitySchemes: this.securitySchemas,
},
info: {
description: this.config.description,
title: this.config.title,
version: this.appVersion,
},
openapi: '3.0.3',
paths: this.paths,
};
}
addSchema(type) {
if (type && !(0, utils_1.isStandardType)(type) && !this.schemas[type.name]) {
this.schemas[type.name] = this.getSchema(type);
}
}
getPath(route) {
const simpleMeta = this.reflector.getMetadata(constants_1.METHOD_API_RESPONSE_METADATA, route.controller.prototype[route.methodName]);
const detailedMeta = this.reflector.getMetadata(constants_1.METHOD_API_RESPONSES_METADATA, route.controller.prototype[route.methodName]);
const operationMeta = this.reflector.getMetadata(constants_1.METHOD_API_OPERATION_METADATA, route.controller.prototype[route.methodName]);
const ctrlMethod = `${route.controller.name}.${route.methodName}`;
const bodyParam = route.params.find(param => param.paramType === http_1.ParameterType.BODY);
const parameters = route.params
.filter(param => param.paramType)
.map(param => ({
in: param.paramType,
name: param.argName,
required: param.paramType !== http_1.ParameterType.BODY,
schema: this.getRefType(param.argType),
}));
const body = parameters.find(param => param.in === http_1.ParameterType.BODY);
let requestBody;
if (body) {
parameters.splice(parameters.indexOf(body), 1);
requestBody = {
content: {
[(0, utils_1.typeToContentType)(bodyParam.argType)]: (0, utils_1.pick)(body, 'schema'),
},
};
}
const preResponses = {
[route.status || exports.DEFAULT_STATUS]: { type: route.returnType, ...simpleMeta },
...detailedMeta,
};
const responses = Object.keys(preResponses).reduce((acc, status) => ({
...acc, [status]: this.toResponse(preResponses[status], route.methodName),
}), {});
const security = this.securitySchemas[ctrlMethod] ? [{ [ctrlMethod]: [] }] : [];
const tag = route.source === sockets_1.SOURCE_TYPE
? `Sockets :: ${route.controller.name}`
: route.controller.name;
return {
...operationMeta,
parameters,
requestBody,
responses,
security,
tags: [tag],
};
}
getRefType(type) {
const typeName = type === null || type === void 0 ? void 0 : type.name;
return this.schemas[typeName]
? { $ref: `#/components/schemas/${typeName}` }
: { type: typeName === null || typeName === void 0 ? void 0 : typeName.toLowerCase() };
}
getSchema(type) {
const schema = {
properties: {},
required: [],
title: type.name,
};
const meta = (0, utils_1.getValidationMeta)(type);
if (!meta.length) {
return schema;
}
const keys = new Set(meta.map(({ propertyName }) => propertyName));
for (const key of [...keys]) {
const validators = meta.filter(({ propertyName }) => propertyName === key);
const parameterMeta = this.reflector.getMetadata(constants_1.PROPERTY_API_PARAMETER_METADATA, type.prototype, key);
// applied validators
const typeValidator = validators.find(({ name }) => name === null || name === void 0 ? void 0 : name.startsWith('is'));
const minimum = validators.find(({ name }) => name === 'min');
const maximum = validators.find(({ name }) => name === 'max');
if (!validators.some(({ type }) => type.startsWith('conditional'))) {
schema.required.push(key);
}
schema.properties[key] = {
...parameterMeta,
maximum: maximum === null || maximum === void 0 ? void 0 : maximum.constraints[0],
minimum: minimum === null || minimum === void 0 ? void 0 : minimum.constraints[0],
type: typeValidator === null || typeValidator === void 0 ? void 0 : typeValidator.name.toLowerCase().replace('is', ''),
};
}
return schema;
}
processMetadata() {
var _a;
const routeMetadata = this.metadataScanner.scan();
for (const route of routeMetadata) {
for (const param of route.params) {
this.addSchema(param.argType);
}
this.addSchema(route.returnType);
const meta = this.reflector.getMetadata(constants_1.METHOD_API_SECURITY_METADATA, route.controller.prototype[route.methodName]);
if (meta) {
this.securitySchemas[`${route.controller.name}.${route.methodName}`] = meta;
}
const rawUrl = (0, utils_1.replaceUrlParameters)(route.url);
const url = route.source === sockets_1.SOURCE_TYPE
? `${rawUrl} => ${route.type}` + (route['event'] ? ` => ${route['event']}` : '')
: rawUrl;
const methodType = route.source === sockets_1.SOURCE_TYPE
? exports.DEFAULT_METHOD
: route.type;
this.paths[url] = {
...((_a = this.paths[url]) !== null && _a !== void 0 ? _a : {}),
[methodType]: this.getPath(route),
};
}
}
toResponse(response, description) {
var _a;
return {
content: {
[(0, utils_1.typeToContentType)(response.type)]: {
schema: this.getRefType(response.type),
},
},
description: (_a = response.description) !== null && _a !== void 0 ? _a : description,
};
}
};
exports.SwaggerDocument = SwaggerDocument = __decorate([
(0, di_1.Injectable)(),
__param(0, (0, di_1.Inject)(core_1.APP_VERSION)),
__param(0, (0, di_1.Optional)()),
__param(1, (0, di_1.Inject)(constants_1.SWAGGER_CONFIG)),
__metadata("design:paramtypes", [String, Object, core_1.MetadataScanner,
core_1.Reflector])
], SwaggerDocument);
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"swagger-document.js","sourceRoot":"","sources":["../../../../../src/platforms/swagger/helpers/swagger-ui/swagger-document.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, Optional } from '@decorators/di';\nimport { OpenAPIV3_1 } from 'openapi-types';\n\nimport { APP_VERSION, ClassConstructor, Handler, MetadataScanner, Reflector } from '../../../../core';\nimport { HttpMethodType, ParameterType, RouteMetadata } from '../../../http';\nimport { SOURCE_TYPE as SOCKETS_SOURCE_TYPE } from '../../../sockets';\nimport { ApiResponse, SwaggerConfig } from '../../types';\nimport { METHOD_API_OPERATION_METADATA, METHOD_API_RESPONSE_METADATA, METHOD_API_RESPONSES_METADATA, METHOD_API_SECURITY_METADATA, PROPERTY_API_PARAMETER_METADATA, SWAGGER_CONFIG } from '../constants';\nimport { getValidationMeta, isStandardType, pick, replaceUrlParameters, typeToContentType } from './utils';\n\nexport const DEFAULT_STATUS = 200;\nexport const DEFAULT_METHOD = HttpMethodType.POST;\n\n@Injectable()\nexport class SwaggerDocument {\n  private paths = {};\n  private schemas = {};\n  private securitySchemas = {};\n\n  constructor(\n    @Inject(APP_VERSION) @Optional() private appVersion: string,\n    @Inject(SWAGGER_CONFIG) private config: SwaggerConfig,\n    private metadataScanner: MetadataScanner,\n    private reflector: Reflector,\n  ) { }\n\n  generate() {\n    this.processMetadata();\n\n    return {\n      components: {\n        schemas: this.schemas,\n        securitySchemes: this.securitySchemas,\n      },\n      info: {\n        description: this.config.description,\n        title: this.config.title,\n        version: this.appVersion,\n      },\n      openapi: '3.0.3',\n      paths: this.paths,\n    } as OpenAPIV3_1.Document;\n  }\n\n  private addSchema(type?: Handler | ClassConstructor) {\n    if (type && !isStandardType(type) && !this.schemas[type.name]) {\n      this.schemas[type.name] = this.getSchema(type);\n    }\n  }\n\n  private getPath(route: RouteMetadata) {\n    const simpleMeta = this.reflector.getMetadata(\n      METHOD_API_RESPONSE_METADATA,\n      route.controller.prototype[route.methodName],\n    );\n    const detailedMeta = this.reflector.getMetadata(\n      METHOD_API_RESPONSES_METADATA,\n      route.controller.prototype[route.methodName],\n    );\n    const operationMeta = this.reflector.getMetadata(\n      METHOD_API_OPERATION_METADATA,\n      route.controller.prototype[route.methodName],\n    );\n\n    const ctrlMethod = `${route.controller.name}.${route.methodName}`;\n\n    const bodyParam = route.params.find(param => param.paramType === ParameterType.BODY);\n    const parameters = route.params\n      .filter(param => param.paramType)\n      .map(param => ({\n        in: param.paramType,\n        name: param.argName,\n        required: param.paramType !== ParameterType.BODY,\n        schema: this.getRefType(param.argType),\n      } as OpenAPIV3_1.ParameterObject));\n\n    const body = parameters.find(param => param.in === ParameterType.BODY);\n    let requestBody: OpenAPIV3_1.RequestBodyObject;\n\n    if (body) {\n      parameters.splice(parameters.indexOf(body), 1);\n\n      requestBody = {\n        content: {\n          [typeToContentType(bodyParam.argType)]: pick(body, 'schema'),\n        },\n      };\n    }\n\n    const preResponses = {\n      [route.status || DEFAULT_STATUS]: { type: route.returnType, ...simpleMeta },\n      ...detailedMeta,\n    };\n    const responses = Object.keys(preResponses).reduce((acc, status) => ({\n      ...acc, [status]: this.toResponse(preResponses[status], route.methodName),\n    }), {});\n\n    const security = this.securitySchemas[ctrlMethod] ? [{ [ctrlMethod]: [] }] : [];\n\n    const tag = route.source === SOCKETS_SOURCE_TYPE\n      ? `Sockets :: ${route.controller.name}`\n      : route.controller.name;\n\n    return {\n      ...operationMeta,\n      parameters,\n      requestBody,\n      responses,\n      security,\n      tags: [tag],\n    } as OpenAPIV3_1.OperationObject;\n  }\n\n  private getRefType(type?: Handler | ClassConstructor) {\n    const typeName = type?.name;\n\n    return this.schemas[typeName]\n      ? { $ref: `#/components/schemas/${typeName}` }\n      : { type: typeName?.toLowerCase() };\n  }\n\n  private getSchema(type: Handler | ClassConstructor) {\n    const schema = {\n      properties: {},\n      required: [],\n      title: type.name,\n    } as OpenAPIV3_1.SchemaObject;\n\n    const meta = getValidationMeta(type);\n\n    if (!meta.length) {\n      return schema;\n    }\n\n    const keys = new Set(meta.map(({ propertyName }) => propertyName));\n\n    for (const key of [...keys]) {\n      const validators = meta.filter(({ propertyName }) => propertyName === key);\n      const parameterMeta = this.reflector.getMetadata(\n        PROPERTY_API_PARAMETER_METADATA,\n        type.prototype,\n        key,\n      );\n\n      // applied validators\n      const typeValidator = validators.find(({ name }) => name?.startsWith('is'));\n      const minimum = validators.find(({ name }) => name === 'min');\n      const maximum = validators.find(({ name }) => name === 'max');\n\n      if (!validators.some(({ type }) => type.startsWith('conditional'))) {\n        schema.required.push(key);\n      }\n\n      schema.properties[key] = {\n        ...parameterMeta,\n        maximum: maximum?.constraints[0],\n        minimum: minimum?.constraints[0],\n        type: typeValidator?.name.toLowerCase().replace('is', ''),\n      } as OpenAPIV3_1.SchemaObject;\n    }\n\n    return schema;\n  }\n\n  private processMetadata() {\n    const routeMetadata = this.metadataScanner.scan();\n\n    for (const route of routeMetadata) {\n      for (const param of route.params) {\n        this.addSchema(param.argType);\n      }\n\n      this.addSchema(route.returnType);\n\n      const meta = this.reflector.getMetadata(\n        METHOD_API_SECURITY_METADATA,\n        route.controller.prototype[route.methodName],\n      );\n\n      if (meta) {\n        this.securitySchemas[`${route.controller.name}.${route.methodName}`] = meta;\n      }\n\n      const rawUrl = replaceUrlParameters(route.url);\n      const url = route.source === SOCKETS_SOURCE_TYPE\n        ? `${rawUrl} => ${route.type}` + (route['event'] ? ` => ${route['event']}` : '')\n        : rawUrl;\n\n      const methodType = route.source === SOCKETS_SOURCE_TYPE\n        ? DEFAULT_METHOD\n        : route.type;\n\n      this.paths[url] = {\n        ...(this.paths[url] ?? {}),\n        [methodType]: this.getPath(route),\n      };\n    }\n  }\n\n  private toResponse(response: ApiResponse, description: string) {\n    return {\n      content: {\n        [typeToContentType(response.type)]: {\n          schema: this.getRefType(response.type),\n        },\n      },\n      description: response.description ?? description,\n    } as OpenAPIV3_1.ResponseObject;\n  }\n}\n"]}