@decorators/server
Version:
node decorators - decorators for express library
144 lines (135 loc) • 14.3 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.SwaggerResolver = void 0;
const di_1 = require("@decorators/di");
const fs_1 = require("fs");
const path_1 = require("path");
const swagger_ui_dist_1 = require("swagger-ui-dist");
const core_1 = require("../../../../core");
const http_1 = require("../../../http");
const constants_1 = require("../constants");
const swagger_document_1 = require("./swagger-document");
let SwaggerResolver = exports.SwaggerResolver = class SwaggerResolver {
constructor(adapter, config, document) {
this.adapter = adapter;
this.config = config;
this.document = document;
}
resolve() {
const swaggerPath = (0, core_1.addLeadingSlash)(this.config.path);
const swaggerFilePath = (0, core_1.addLeadingSlash)((0, core_1.buildUrl)(this.config.path, 'swagger.json'));
this.adapter.routes([
{
handler: this.handler(JSON.stringify(this.document.generate()), { 'content-type': 'application/json' }),
type: 'get',
url: swaggerFilePath,
},
{
handler: this.handler(indexStyles(this.config.theme), { 'content-type': 'text/css' }),
type: 'get',
url: `${swaggerPath}/index.css`,
},
{
handler: this.handler(initializerScriptContent(swaggerFilePath), { 'content-type': 'text/javascript' }),
type: 'get',
url: `${swaggerPath}/swagger-initializer.js`,
},
]);
this.adapter.serveStatic(swaggerPath, (0, swagger_ui_dist_1.absolutePath)());
}
handler(response, headers = {}) {
return async (...args) => {
const res = await this.adapter.getParam(http_1.ParameterType.RESPONSE, null, ...args);
Object.entries(headers).forEach(([name, value]) => this.adapter.setHeader(res(), name, value));
this.adapter.reply(res(), response);
};
}
};
exports.SwaggerResolver = SwaggerResolver = __decorate([
(0, di_1.Injectable)(),
__param(0, (0, di_1.Inject)(http_1.HTTP_ADAPTER)),
__param(1, (0, di_1.Inject)(constants_1.SWAGGER_CONFIG)),
__metadata("design:paramtypes", [http_1.HttpApplicationAdapter, Object, swagger_document_1.SwaggerDocument])
], SwaggerResolver);
function initializerScriptContent(jsonPath) {
return `
window.onload = function() {
window.ui = SwaggerUIBundle({
url: "${jsonPath}",
dom_id: '#swagger-ui',
deepLinking: true,
presets: [
SwaggerUIBundle.presets.apis,
SwaggerUIStandalonePreset.slice(1) // slice removes top-bar plugin
],
plugins: [
SwaggerUIBundle.plugins.DownloadUrl
],
layout: "StandaloneLayout"
});
onSwaggerUIReady(disableSocketsTryItOut);
};
function onSwaggerUIReady(fn) {
const interval = setInterval(() => {
if (document.getElementById('swagger-ui')) {
clearInterval(interval);
fn();
}
}, 100);
}
function disableSocketsTryItOut() {
const socketEvents = [...document.querySelectorAll('[data-tag^="sockets" i]')];
for (const el of socketEvents) {
el.parentElement.classList.add('server-swagger-sockets');
}
}
`;
}
function indexStyles(theme) {
const styles = (0, fs_1.readFileSync)((0, path_1.resolve)((0, swagger_ui_dist_1.absolutePath)(), 'index.css'), 'utf-8');
const darkStyles = (0, fs_1.readFileSync)((0, path_1.resolve)(__dirname, 'swagger-dark.css'), 'utf-8');
const styleOverrides = `
${styles}
.server-swagger-sockets .try-out {
display: none;
}
.server-swagger-sockets .opblock-summary-method {
overflow: hidden;
position: relative;
}
.server-swagger-sockets .opblock-summary-method::after {
align-items: center;
background-color: #a748cb;
bottom: 0;
content: 'SOCKETS';
display: flex;
justify-content: center;
left: 0;
position: absolute;
right: 0;
top: 0;
}
`;
if (theme === 'light') {
return styleOverrides;
}
return `
${styleOverrides}
${theme === 'auto' ? '@media only screen and (prefers-color-scheme: dark) {' : ''}
${darkStyles}
${theme === 'auto' ? '}' : ''}
`;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"swagger-resolver.js","sourceRoot":"","sources":["../../../../../src/platforms/swagger/helpers/swagger-ui/swagger-resolver.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from '@decorators/di';\nimport { readFileSync } from 'fs';\nimport { resolve } from 'path';\nimport { absolutePath } from 'swagger-ui-dist';\n\nimport { addLeadingSlash, buildUrl } from '../../../../core';\nimport { HTTP_ADAPTER, HttpApplicationAdapter, ParameterType } from '../../../http';\nimport { SwaggerConfig } from '../../types';\nimport { SWAGGER_CONFIG } from '../constants';\nimport { SwaggerDocument } from './swagger-document';\n\n@Injectable()\nexport class SwaggerResolver {\n  constructor(\n    @Inject(HTTP_ADAPTER) private adapter: HttpApplicationAdapter,\n    @Inject(SWAGGER_CONFIG) private config: SwaggerConfig,\n    private document: SwaggerDocument,\n  ) { }\n\n  resolve() {\n    const swaggerPath = addLeadingSlash(this.config.path);\n    const swaggerFilePath = addLeadingSlash(buildUrl(this.config.path, 'swagger.json'));\n\n    this.adapter.routes([\n      {\n        handler: this.handler(\n          JSON.stringify(this.document.generate()),\n          { 'content-type': 'application/json' },\n        ),\n        type: 'get',\n        url: swaggerFilePath,\n      },\n      {\n        handler: this.handler(\n          indexStyles(this.config.theme),\n          { 'content-type': 'text/css' },\n        ),\n        type: 'get',\n        url: `${swaggerPath}/index.css`,\n      },\n      {\n        handler: this.handler(\n          initializerScriptContent(swaggerFilePath),\n          { 'content-type': 'text/javascript' },\n        ),\n        type: 'get',\n        url: `${swaggerPath}/swagger-initializer.js`,\n      },\n    ]);\n\n    this.adapter.serveStatic(swaggerPath, absolutePath());\n  }\n\n  private handler(response: unknown, headers: Record<string, string> = {}) {\n    return async (...args: any[]) => {\n      const res = await this.adapter.getParam(ParameterType.RESPONSE, null, ...args);\n\n      Object.entries(headers).forEach(([name, value]) =>\n        this.adapter.setHeader(res(), name, value),\n      );\n\n      this.adapter.reply(res(), response);\n    };\n  }\n}\n\nfunction initializerScriptContent(jsonPath: string) {\n  return `\n    window.onload = function() {\n      window.ui = SwaggerUIBundle({\n        url: \"${jsonPath}\",\n        dom_id: '#swagger-ui',\n        deepLinking: true,\n        presets: [\n          SwaggerUIBundle.presets.apis,\n          SwaggerUIStandalonePreset.slice(1) // slice removes top-bar plugin\n        ],\n        plugins: [\n          SwaggerUIBundle.plugins.DownloadUrl\n        ],\n        layout: \"StandaloneLayout\"\n      });\n\n      onSwaggerUIReady(disableSocketsTryItOut);\n    };\n\n    function onSwaggerUIReady(fn) {\n      const interval = setInterval(() => {\n        if (document.getElementById('swagger-ui')) {\n          clearInterval(interval);\n\n          fn();\n        }\n      }, 100);\n    }\n\n    function disableSocketsTryItOut() {\n      const socketEvents = [...document.querySelectorAll('[data-tag^=\"sockets\" i]')];\n\n      for (const el of socketEvents) {\n        el.parentElement.classList.add('server-swagger-sockets');\n      }\n    }\n  `;\n}\n\nfunction indexStyles(theme: SwaggerConfig['theme']) {\n  const styles = readFileSync(resolve(absolutePath(), 'index.css'), 'utf-8');\n  const darkStyles = readFileSync(resolve(__dirname, 'swagger-dark.css'), 'utf-8');\n\n  const styleOverrides = `\n    ${styles}\n\n    .server-swagger-sockets .try-out {\n      display: none;\n    }\n\n    .server-swagger-sockets .opblock-summary-method {\n      overflow: hidden;\n      position: relative;\n    }\n\n    .server-swagger-sockets .opblock-summary-method::after {\n      align-items: center;\n      background-color: #a748cb;\n      bottom: 0;\n      content: 'SOCKETS';\n      display: flex;\n      justify-content: center;\n      left: 0;\n      position: absolute;\n      right: 0;\n      top: 0;\n    }\n  `;\n\n  if (theme === 'light') {\n    return styleOverrides;\n  }\n\n  return `\n    ${styleOverrides}\n\n    ${theme === 'auto' ? '@media only screen and (prefers-color-scheme: dark) {' : ''}\n      ${darkStyles}\n    ${theme === 'auto' ? '}' : ''}\n  `;\n}\n"]}