@stemy/backend 3.5.16 → 3.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,151 @@
1
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ 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;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ };
7
+ var __metadata = (this && this.__metadata) || function (k, v) {
8
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
+ };
10
+ import { injectable } from "tsyringe";
11
+ import { Controller, Get, Header } from "routing-controllers";
12
+ import { Configuration } from "../services/configuration";
13
+ let TerminalController = class TerminalController {
14
+ constructor(config) {
15
+ this.config = config;
16
+ this.serviceName = config.resolve("serviceName");
17
+ this.serviceUrl = config.resolve("serviceUrl");
18
+ }
19
+ terminal() {
20
+ return `
21
+ <!DOCTYPE html>
22
+ <html>
23
+ <head>
24
+ <meta charset="utf-8">
25
+ <title>${this.serviceName} console</title>
26
+ <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/xterm@5.1.0/css/xterm.min.css"/>
27
+ <script type="text/javascript" src="${this.serviceUrl}/socket/socket.io.js"></script>
28
+ <style>
29
+ body {
30
+ margin: 0;
31
+ padding: 0;
32
+ background: #1e1e1e;
33
+ }
34
+ body * {
35
+ box-sizing: border-box;
36
+ }
37
+ #terminal {
38
+ margin: 40px;
39
+ background: black;
40
+ border: 1px solid #dedede;
41
+ border-radius: 5px;
42
+ overflow: hidden;
43
+ }
44
+ #terminal .xterm {
45
+ margin: 10px;
46
+ height: calc(100vh - 120px);
47
+ border-radius: 5px;
48
+ }
49
+ #terminal .xterm-viewport::-webkit-scrollbar {
50
+ width: 0.4em;
51
+ background-color: #222;
52
+ }
53
+ #terminal .xterm-viewport::-webkit-scrollbar-thumb {
54
+ background-color: #555;
55
+ }
56
+ </style>
57
+ </head>
58
+ <body>
59
+ <div id="terminal"></div>
60
+ <script type="module">
61
+ // Import terminal modules
62
+ import xterm from 'https://cdn.jsdelivr.net/npm/xterm@5.1.0/+esm';
63
+ import xtermFit from 'https://cdn.jsdelivr.net/npm/xterm-addon-fit@0.7.0/+esm';
64
+ // Initialize variables
65
+ var terminal = new xterm.Terminal();
66
+ var fitAddon = new xtermFit.FitAddon();
67
+ var socket = io("${this.serviceUrl}", {path: "/socket"});
68
+ var clear = function () {
69
+ terminal.clear();
70
+ terminal.reset();
71
+ // Prev line
72
+ terminal.write("\u001B[F");
73
+ }
74
+ // Initialize terminal
75
+ terminal.loadAddon(fitAddon);
76
+ terminal.open(document.getElementById('terminal'));
77
+ terminal.onData(function (data) {
78
+ socket.emit("terminal-data", data.charCodeAt(0) === 127 ? "\b" : data);
79
+ });
80
+ // Socket listeners
81
+ socket.on("terminal-data", function (data) {
82
+ if (data === "\x1bc") {
83
+ clear();
84
+ return;
85
+ }
86
+ terminal.write(data);
87
+ });
88
+ socket.on("terminal-upload", function (data) {
89
+ var input = document.createElement("input");
90
+ input.type = "file";
91
+ input.accept = data.accept;
92
+ input.onchange = function () {
93
+ var file = input.files[0];
94
+ var reader = new FileReader();
95
+ reader.onload = function () {
96
+ console.log(reader.result);
97
+ socket.emit("terminal-upload", {
98
+ id: data.id,
99
+ label: file.name,
100
+ content: reader.result,
101
+ accept: data.accept
102
+ });
103
+ };
104
+ reader.onerror = function () {
105
+ console.error(reader.error);
106
+ socket.emit("terminal-upload", {
107
+ id: data.id,
108
+ label: file.name,
109
+ error: reader.error,
110
+ accept: data.accept
111
+ });
112
+ };
113
+ reader.readAsDataURL(file);
114
+ };
115
+ input.click();
116
+ });
117
+ socket.on("terminal-download", function (data) {
118
+ var link = document.createElement("a");
119
+ link.href = data.content;
120
+ link.download = data.filename;
121
+ link.click();
122
+ });
123
+ socket.on("connect", function () {
124
+ clear();
125
+ terminal.writeln("Welcome to ${this.serviceName} service's console!");
126
+ socket.emit("terminal-init");
127
+ });
128
+ socket.on("disconnect", function () {
129
+ clear();
130
+ terminal.writeln("Disconnected from ${this.serviceName} service's console.");
131
+ });
132
+ </script>
133
+ </body>
134
+ </html>
135
+ `;
136
+ }
137
+ };
138
+ __decorate([
139
+ Get(),
140
+ Header("Content-Type", "text/html"),
141
+ __metadata("design:type", Function),
142
+ __metadata("design:paramtypes", []),
143
+ __metadata("design:returntype", void 0)
144
+ ], TerminalController.prototype, "terminal", null);
145
+ TerminalController = __decorate([
146
+ injectable(),
147
+ Controller("/console"),
148
+ __metadata("design:paramtypes", [Configuration])
149
+ ], TerminalController);
150
+ export { TerminalController };
151
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"terminal.controller.js","sourceRoot":"","sources":["../../../src/rest-controllers/terminal.controller.ts"],"names":[],"mappings":";;;;;;;;;AAAA,OAAO,EAAC,UAAU,EAAC,MAAM,UAAU,CAAC;AACpC,OAAO,EAAC,UAAU,EAAE,GAAG,EAAE,MAAM,EAAC,MAAM,qBAAqB,CAAC;AAC5D,OAAO,EAAC,aAAa,EAAC,MAAM,2BAA2B,CAAC;IAI3C,kBAAkB,SAAlB,kBAAkB;IAM3B,YAAqB,MAAqB;QAArB,WAAM,GAAN,MAAM,CAAe;QACtC,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QACjD,IAAI,CAAC,UAAU,GAAG,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IACnD,CAAC;IAID,QAAQ;QACJ,OAAO;;;;;qBAKM,IAAI,CAAC,WAAW;;kDAEa,IAAI,CAAC,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;mCAwC9B,IAAI,CAAC,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;mDA0DC,IAAI,CAAC,WAAW;;;;;0DAKT,IAAI,CAAC,WAAW;;;;;SAKjE,CAAC;IACN,CAAC;CACJ,CAAA;AAtHG;IAFC,GAAG,EAAE;IACL,MAAM,CAAC,cAAc,EAAE,WAAW,CAAC;;;;kDAsHnC;AAlIQ,kBAAkB;IAF9B,UAAU,EAAE;IACZ,UAAU,CAAC,UAAU,CAAC;qCAOU,aAAa;GANjC,kBAAkB,CAmI9B;SAnIY,kBAAkB","sourcesContent":["import {injectable} from \"tsyringe\";\r\nimport {Controller, Get, Header} from \"routing-controllers\";\r\nimport {Configuration} from \"../services/configuration\";\r\n\r\n@injectable()\r\n@Controller(\"/console\")\r\nexport class TerminalController {\r\n\r\n    protected serviceName: string;\r\n\r\n    protected serviceUrl: string;\r\n\r\n    constructor(readonly config: Configuration) {\r\n        this.serviceName = config.resolve(\"serviceName\");\r\n        this.serviceUrl = config.resolve(\"serviceUrl\");\r\n    }\r\n\r\n    @Get()\r\n    @Header(\"Content-Type\", \"text/html\")\r\n    terminal() {\r\n        return `\r\n        <!DOCTYPE html>\r\n        <html>\r\n        <head>\r\n            <meta charset=\"utf-8\">\r\n            <title>${this.serviceName} console</title>\r\n            <link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/xterm@5.1.0/css/xterm.min.css\"/>\r\n            <script type=\"text/javascript\" src=\"${this.serviceUrl}/socket/socket.io.js\"></script>\r\n            <style>\r\n                body {\r\n                    margin: 0;\r\n                    padding: 0;\r\n                    background: #1e1e1e;\r\n                }\r\n                body * {\r\n                    box-sizing: border-box;\r\n                }\r\n                #terminal {\r\n                    margin: 40px;\r\n                    background: black;\r\n                    border: 1px solid #dedede;\r\n                    border-radius: 5px;\r\n                    overflow: hidden;\r\n                }\r\n                #terminal .xterm {\r\n                    margin: 10px;\r\n                    height: calc(100vh - 120px);\r\n                    border-radius: 5px;\r\n                }\r\n                #terminal .xterm-viewport::-webkit-scrollbar {\r\n                    width: 0.4em;\r\n                    background-color: #222;\r\n                }\r\n                #terminal .xterm-viewport::-webkit-scrollbar-thumb {\r\n                    background-color: #555;\r\n                }\r\n            </style>\r\n        </head>\r\n        <body>\r\n            <div id=\"terminal\"></div>\r\n            <script type=\"module\">\r\n                // Import terminal modules\r\n                import xterm from 'https://cdn.jsdelivr.net/npm/xterm@5.1.0/+esm';\r\n                import xtermFit from 'https://cdn.jsdelivr.net/npm/xterm-addon-fit@0.7.0/+esm';\r\n                // Initialize variables\r\n                var terminal = new xterm.Terminal();\r\n                var fitAddon = new xtermFit.FitAddon();\r\n                var socket = io(\"${this.serviceUrl}\", {path: \"/socket\"});\r\n                var clear = function () {\r\n                    terminal.clear();\r\n                    terminal.reset();\r\n                    // Prev line\r\n                    terminal.write(\"\\u001B[F\");\r\n                }\r\n                // Initialize terminal\r\n                terminal.loadAddon(fitAddon);\r\n                terminal.open(document.getElementById('terminal'));\r\n                terminal.onData(function (data) {\r\n                    socket.emit(\"terminal-data\", data.charCodeAt(0) === 127 ? \"\\b\" : data);\r\n                });\r\n                // Socket listeners\r\n                socket.on(\"terminal-data\", function (data) {\r\n                    if (data === \"\\x1bc\") {\r\n                        clear();\r\n                        return;\r\n                    }\r\n                    terminal.write(data);\r\n                });\r\n                socket.on(\"terminal-upload\", function (data) {\r\n                    var input = document.createElement(\"input\");\r\n                    input.type = \"file\";\r\n                    input.accept = data.accept;\r\n                    input.onchange = function () {\r\n                        var file = input.files[0];\r\n                        var reader = new FileReader();\r\n                        reader.onload = function () {\r\n                            console.log(reader.result);\r\n                            socket.emit(\"terminal-upload\", {\r\n                                id: data.id,\r\n                                label: file.name,\r\n                                content: reader.result,\r\n                                accept: data.accept\r\n                            });\r\n                        };\r\n                        reader.onerror = function () {\r\n                            console.error(reader.error);\r\n                            socket.emit(\"terminal-upload\", {\r\n                                id: data.id,\r\n                                label: file.name,\r\n                                error: reader.error,\r\n                                accept: data.accept\r\n                            });\r\n                        };\r\n                        reader.readAsDataURL(file);\r\n                    };\r\n                    input.click();\r\n                });\r\n                socket.on(\"terminal-download\", function (data) {\r\n                    var link = document.createElement(\"a\");\r\n                    link.href = data.content;\r\n                    link.download = data.filename;\r\n                    link.click();\r\n                });\r\n                socket.on(\"connect\", function () {\r\n                    clear();\r\n                    terminal.writeln(\"Welcome to ${this.serviceName} service's console!\");\r\n                    socket.emit(\"terminal-init\");\r\n                });\r\n                socket.on(\"disconnect\", function () {\r\n                    clear();\r\n                    terminal.writeln(\"Disconnected from ${this.serviceName} service's console.\");\r\n                });\r\n            </script>\r\n        </body>\r\n        </html>\r\n        `;\r\n    }\r\n}\r\n"]}
@@ -16,7 +16,6 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
16
16
  var AssetProcessor_1;
17
17
  import { injectable, Lifecycle, scoped } from "tsyringe";
18
18
  import fontKit_ from "fontkit";
19
- import { fromBuffer } from "file-type";
20
19
  import sharp_ from "sharp";
21
20
  const sharp = sharp_;
22
21
  const fontKit = fontKit_;
@@ -32,26 +31,6 @@ const fontProps = [
32
31
  "xHeight", "numGlyphs", "characterSet", "availableFeatures"
33
32
  ];
34
33
  let AssetProcessor = AssetProcessor_1 = class AssetProcessor {
35
- static checkTextFileType(type) {
36
- return type.mime.indexOf("text") >= 0 || type.mime.indexOf("xml") >= 0;
37
- }
38
- static fixTextFileType(type, buffer) {
39
- const text = buffer.toString("utf8");
40
- if (text.indexOf("<svg") >= 0) {
41
- return { ext: "svg", mime: "image/svg+xml" };
42
- }
43
- return type;
44
- }
45
- static fileTypeFromBuffer(buffer) {
46
- var _a;
47
- return __awaiter(this, void 0, void 0, function* () {
48
- const type = ((_a = yield fromBuffer(buffer)) !== null && _a !== void 0 ? _a : { ext: "txt", mime: "text/plain" });
49
- if (AssetProcessor_1.checkTextFileType(type)) {
50
- return AssetProcessor_1.fixTextFileType(type, buffer);
51
- }
52
- return type;
53
- });
54
- }
55
34
  static extractFontFormat(font) {
56
35
  const name = font.constructor.name;
57
36
  const tag = font["directory"].tag;
@@ -126,4 +105,4 @@ AssetProcessor = AssetProcessor_1 = __decorate([
126
105
  scoped(Lifecycle.ContainerScoped)
127
106
  ], AssetProcessor);
128
107
  export { AssetProcessor };
129
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset-processor.js","sourceRoot":"","sources":["../../../src/services/asset-processor.ts"],"names":[],"mappings":"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","sourcesContent":["import {injectable, Lifecycle, scoped} from \"tsyringe\";\r\nimport fontKit_, {Font} from \"fontkit\";\r\nimport {fromBuffer} from \"file-type\";\r\nimport sharp_ from \"sharp\";\r\nimport {FontFormat, IAssetMeta, IFileType} from \"../common-types\";\r\n\r\nconst sharp = sharp_;\r\nconst fontKit = fontKit_;\r\n\r\nconst fontTypes = [\r\n    \"application/font-woff\", \"application/font-woff2\", \"application/x-font-opentype\", \"application/x-font-truetype\", \"application/x-font-datafork\",\r\n    \"font/woff\", \"font/woff2\", \"font/otf\", \"font/ttf\", \"font/datafork\"\r\n];\r\n\r\nconst imageTypes = [\"image/jpeg\", \"image/jpg\", \"image/png\", \"image/svg+xml\"];\r\n\r\nconst fontProps = [\r\n    \"postscriptName\", \"fullName\", \"familyName\", \"subfamilyName\",\r\n    \"copyright\", \"version\", \"unitsPerEm\", \"ascent\", \"descent\", \"lineGap\",\r\n    \"underlinePosition\", \"underlineThickness\", \"italicAngle\", \"capHeight\",\r\n    \"xHeight\", \"numGlyphs\", \"characterSet\", \"availableFeatures\"\r\n];\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class AssetProcessor {\r\n\r\n    private static checkTextFileType(type: IFileType): boolean {\r\n        return type.mime.indexOf(\"text\") >= 0 || type.mime.indexOf(\"xml\") >= 0;\r\n    }\r\n\r\n    private static fixTextFileType(type: IFileType, buffer: Buffer): IFileType {\r\n        const text = buffer.toString(\"utf8\");\r\n        if (text.indexOf(\"<svg\") >= 0) {\r\n            return {ext: \"svg\", mime: \"image/svg+xml\"};\r\n        }\r\n        return type;\r\n    }\r\n\r\n    static async fileTypeFromBuffer(buffer: Buffer): Promise<IFileType> {\r\n        const type = (await fromBuffer(buffer) ?? {ext: \"txt\", mime: \"text/plain\"}) as IFileType;\r\n        if (AssetProcessor.checkTextFileType(type)) {\r\n            return AssetProcessor.fixTextFileType(type, buffer);\r\n        }\r\n        return type;\r\n    }\r\n\r\n    static extractFontFormat(font: Font): FontFormat {\r\n        const name: string = font.constructor.name;\r\n        const tag: string  = font[\"directory\"].tag;\r\n        switch (name) {\r\n            case \"TTFFont\":\r\n                return tag === \"OTTO\" ? \"opentype\" : \"truetype\";\r\n            case \"WOFF2Font\":\r\n                return \"woff2\";\r\n            case \"WOFFFont\":\r\n                return \"woff\";\r\n            case \"DFont\":\r\n                return \"datafork\";\r\n        }\r\n        return null;\r\n    }\r\n\r\n    static isImage(contentType: string): boolean {\r\n        return imageTypes.indexOf(contentType) >= 0;\r\n    }\r\n\r\n    static async copyImageMeta(buffer: Buffer, metadata: IAssetMeta, fileType: IFileType): Promise<Buffer> {\r\n        if (fileType.mime === \"image/svg+xml\") {\r\n            const match = /<svg([^<>]+)>/gi.exec(buffer.toString(\"utf8\"));\r\n            if (match && match.length > 1) {\r\n                const attrs = match[1].match(/([a-z]+)=\"([^\"]+)\"/gi);\r\n                attrs.forEach(attr => {\r\n                    if (attr.length < 5) return;\r\n                    const [name, value] = attr.split(\"=\");\r\n                    const val = value.replace(/\"/gi, \"\") as any;\r\n                    metadata[name] = isNaN(val) ? val : Number(val);\r\n                });\r\n                if (metadata.viewBox && (isNaN(metadata.width) || isNaN(metadata.height))) {\r\n                    const parts = (metadata.viewBox as string).split(\" \");\r\n                    metadata.width = Number(parts[0]) + Number(parts[2]);\r\n                    metadata.height = Number(parts[1]) + Number(parts[3]);\r\n                }\r\n                if (!isNaN(metadata.width) && !isNaN(metadata.height)) {\r\n                    metadata.svgSize = {x: metadata.width, y: metadata.height};\r\n                }\r\n            }\r\n            return buffer;\r\n        }\r\n        const output = await sharp(buffer).rotate().toBuffer({resolveWithObject: true});\r\n        Object.assign(metadata, output.info);\r\n        return output.data;\r\n    }\r\n\r\n    static isFont(contentType: string): boolean {\r\n        return fontTypes.indexOf(contentType) >= 0;\r\n    }\r\n\r\n    static copyFontMeta(buffer: Buffer, metadata: IAssetMeta): void {\r\n        const font: Font = fontKit.create(buffer);\r\n        metadata.format = AssetProcessor.extractFontFormat(font);\r\n        fontProps.forEach(prop => {\r\n            metadata[prop] = font[prop];\r\n        });\r\n    }\r\n\r\n    async process(buffer: Buffer, metadata: IAssetMeta, fileType: IFileType): Promise<Buffer> {\r\n        if (AssetProcessor.isImage(fileType.mime)) {\r\n            buffer = await AssetProcessor.copyImageMeta(buffer, metadata, fileType);\r\n        }\r\n        if (AssetProcessor.isFont(fileType.mime)) {\r\n            AssetProcessor.copyFontMeta(buffer, metadata);\r\n        }\r\n        return buffer;\r\n    }\r\n}\r\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset-processor.js","sourceRoot":"","sources":["../../../src/services/asset-processor.ts"],"names":[],"mappings":"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","sourcesContent":["import {injectable, Lifecycle, scoped} from \"tsyringe\";\r\nimport fontKit_, {Font} from \"fontkit\";\r\nimport sharp_ from \"sharp\";\r\nimport {FontFormat, IAssetMeta, IFileType} from \"../common-types\";\r\n\r\nconst sharp = sharp_;\r\nconst fontKit = fontKit_;\r\n\r\nconst fontTypes = [\r\n    \"application/font-woff\", \"application/font-woff2\", \"application/x-font-opentype\", \"application/x-font-truetype\", \"application/x-font-datafork\",\r\n    \"font/woff\", \"font/woff2\", \"font/otf\", \"font/ttf\", \"font/datafork\"\r\n];\r\n\r\nconst imageTypes = [\"image/jpeg\", \"image/jpg\", \"image/png\", \"image/svg+xml\"];\r\n\r\nconst fontProps = [\r\n    \"postscriptName\", \"fullName\", \"familyName\", \"subfamilyName\",\r\n    \"copyright\", \"version\", \"unitsPerEm\", \"ascent\", \"descent\", \"lineGap\",\r\n    \"underlinePosition\", \"underlineThickness\", \"italicAngle\", \"capHeight\",\r\n    \"xHeight\", \"numGlyphs\", \"characterSet\", \"availableFeatures\"\r\n];\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class AssetProcessor {\r\n\r\n    static extractFontFormat(font: Font): FontFormat {\r\n        const name: string = font.constructor.name;\r\n        const tag: string  = font[\"directory\"].tag;\r\n        switch (name) {\r\n            case \"TTFFont\":\r\n                return tag === \"OTTO\" ? \"opentype\" : \"truetype\";\r\n            case \"WOFF2Font\":\r\n                return \"woff2\";\r\n            case \"WOFFFont\":\r\n                return \"woff\";\r\n            case \"DFont\":\r\n                return \"datafork\";\r\n        }\r\n        return null;\r\n    }\r\n\r\n    static isImage(contentType: string): boolean {\r\n        return imageTypes.indexOf(contentType) >= 0;\r\n    }\r\n\r\n    static async copyImageMeta(buffer: Buffer, metadata: IAssetMeta, fileType: IFileType): Promise<Buffer> {\r\n        if (fileType.mime === \"image/svg+xml\") {\r\n            const match = /<svg([^<>]+)>/gi.exec(buffer.toString(\"utf8\"));\r\n            if (match && match.length > 1) {\r\n                const attrs = match[1].match(/([a-z]+)=\"([^\"]+)\"/gi);\r\n                attrs.forEach(attr => {\r\n                    if (attr.length < 5) return;\r\n                    const [name, value] = attr.split(\"=\");\r\n                    const val = value.replace(/\"/gi, \"\") as any;\r\n                    metadata[name] = isNaN(val) ? val : Number(val);\r\n                });\r\n                if (metadata.viewBox && (isNaN(metadata.width) || isNaN(metadata.height))) {\r\n                    const parts = (metadata.viewBox as string).split(\" \");\r\n                    metadata.width = Number(parts[0]) + Number(parts[2]);\r\n                    metadata.height = Number(parts[1]) + Number(parts[3]);\r\n                }\r\n                if (!isNaN(metadata.width) && !isNaN(metadata.height)) {\r\n                    metadata.svgSize = {x: metadata.width, y: metadata.height};\r\n                }\r\n            }\r\n            return buffer;\r\n        }\r\n        const output = await sharp(buffer).rotate().toBuffer({resolveWithObject: true});\r\n        Object.assign(metadata, output.info);\r\n        return output.data;\r\n    }\r\n\r\n    static isFont(contentType: string): boolean {\r\n        return fontTypes.indexOf(contentType) >= 0;\r\n    }\r\n\r\n    static copyFontMeta(buffer: Buffer, metadata: IAssetMeta): void {\r\n        const font: Font = fontKit.create(buffer);\r\n        metadata.format = AssetProcessor.extractFontFormat(font);\r\n        fontProps.forEach(prop => {\r\n            metadata[prop] = font[prop];\r\n        });\r\n    }\r\n\r\n    async process(buffer: Buffer, metadata: IAssetMeta, fileType: IFileType): Promise<Buffer> {\r\n        if (AssetProcessor.isImage(fileType.mime)) {\r\n            buffer = await AssetProcessor.copyImageMeta(buffer, metadata, fileType);\r\n        }\r\n        if (AssetProcessor.isFont(fileType.mime)) {\r\n            AssetProcessor.copyFontMeta(buffer, metadata);\r\n        }\r\n        return buffer;\r\n    }\r\n}\r\n"]}
@@ -19,7 +19,7 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
19
19
  import { injectable, Lifecycle, scoped } from "tsyringe";
20
20
  import { ObjectId } from "bson";
21
21
  import axios from "axios";
22
- import { bufferToStream, copyStream, streamToBuffer } from "../utils";
22
+ import { bufferToStream, copyStream, streamToBuffer, fileTypeFromBuffer } from "../utils";
23
23
  import { MongoConnector } from "./mongo-connector";
24
24
  import { AssetProcessor } from "./asset-processor";
25
25
  import { Asset } from "./entities/asset";
@@ -38,11 +38,11 @@ let Assets = class Assets {
38
38
  const buffer = yield streamToBuffer(stream);
39
39
  let fileType = { ext: "", mime: contentType };
40
40
  try {
41
- fileType = yield AssetProcessor.fileTypeFromBuffer(buffer);
41
+ fileType = yield fileTypeFromBuffer(buffer);
42
42
  }
43
43
  catch (e) {
44
44
  if (!fileType.mime) {
45
- throw `Can't determine mime type`;
45
+ throw new Error(`Can't determine mime type: ${e}`);
46
46
  }
47
47
  console.log(`Can't determine mime type`, e);
48
48
  }
@@ -54,7 +54,7 @@ let Assets = class Assets {
54
54
  return __awaiter(this, void 0, void 0, function* () {
55
55
  let fileType = { ext: "", mime: contentType };
56
56
  try {
57
- fileType = yield AssetProcessor.fileTypeFromBuffer(buffer);
57
+ fileType = yield fileTypeFromBuffer(buffer);
58
58
  }
59
59
  catch (e) {
60
60
  if (!fileType.mime) {
@@ -86,7 +86,7 @@ let Assets = class Assets {
86
86
  let buffer = (yield axios({ url, responseType: "arraybuffer" })).data;
87
87
  let fileType = { ext: "", mime: contentType };
88
88
  try {
89
- fileType = yield AssetProcessor.fileTypeFromBuffer(buffer);
89
+ fileType = yield fileTypeFromBuffer(buffer);
90
90
  }
91
91
  catch (e) {
92
92
  if (!fileType.mime) {
@@ -178,4 +178,4 @@ Assets = __decorate([
178
178
  __metadata("design:paramtypes", [MongoConnector, AssetProcessor])
179
179
  ], Assets);
180
180
  export { Assets };
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assets.js","sourceRoot":"","sources":["../../../src/services/assets.ts"],"names":[],"mappings":"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","sourcesContent":["import {injectable, Lifecycle, scoped} from \"tsyringe\";\r\nimport {Readable} from \"stream\";\r\nimport {ObjectId} from \"bson\";\r\nimport {Collection, GridFSBucket} from \"mongodb\";\r\nimport {FilterQuery} from \"mongoose\";\r\nimport axios from \"axios\";\r\n\r\nimport {bufferToStream, copyStream, streamToBuffer} from \"../utils\";\r\nimport {IAsset, IAssetMeta, IFileType} from \"../common-types\";\r\nimport {MongoConnector} from \"./mongo-connector\";\r\nimport {AssetProcessor} from \"./asset-processor\";\r\nimport {Asset} from \"./entities/asset\";\r\nimport {TempAsset} from \"./entities/temp-asset\";\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class Assets {\r\n\r\n    readonly bucket: GridFSBucket;\r\n    readonly collection: Collection<Partial<IAsset>>;\r\n\r\n    constructor(readonly connector: MongoConnector, readonly assetProcessor: AssetProcessor) {\r\n        this.bucket = connector.bucket;\r\n        this.collection = connector.database?.collection(\"assets.files\");\r\n    }\r\n\r\n    async write(stream: Readable, contentType: string = null, metadata: IAssetMeta = null): Promise<IAsset> {\r\n        const uploadStream = copyStream(stream);\r\n        const buffer = await streamToBuffer(stream);\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await AssetProcessor.fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw `Can't determine mime type`;\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        metadata = metadata || {};\r\n        return this.upload(uploadStream, fileType, metadata);\r\n    }\r\n\r\n    async writeBuffer(buffer: Buffer, metadata: IAssetMeta = null, contentType: string = null): Promise<IAsset> {\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await AssetProcessor.fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw `Can't determine mime type`;\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        metadata = metadata || {};\r\n        buffer = await this.assetProcessor.process(buffer, metadata, fileType);\r\n        return this.upload(bufferToStream(buffer), fileType, metadata);\r\n    }\r\n\r\n    async writeUrl(url: string, metadata: IAssetMeta = null): Promise<IAsset> {\r\n        metadata = metadata || {};\r\n        metadata.filename = metadata.filename || url;\r\n        metadata.url = url;\r\n        metadata.uploadTime = new Date().getTime();\r\n        const oneWeek = 1000 * 3600 * 24 * 7;\r\n        const asset = await this.find({\"metadata.url\": url, \"metadata.uploadTime\": {$gt: metadata.uploadTime - oneWeek}});\r\n        if (asset) return asset;\r\n        const buffer = (await axios({ url, responseType: \"arraybuffer\" })).data as Buffer;\r\n        return this.writeBuffer(buffer, metadata);\r\n    }\r\n\r\n    async download(url: string, contentType: string = null): Promise<IAsset> {\r\n        let buffer = (await axios({ url, responseType: \"arraybuffer\" })).data as Buffer;\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await AssetProcessor.fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw `Can't determine mime type`;\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        const metadata: IAssetMeta = {\r\n            filename: url,\r\n            extension: (fileType.ext || \"\").trim()\r\n        };\r\n        buffer = await this.assetProcessor.process(buffer, metadata, fileType);\r\n        return new TempAsset(buffer, url, fileType.mime, metadata);\r\n    }\r\n\r\n    async read(id: string): Promise<IAsset> {\r\n        return !id ? null : this.find({_id: new ObjectId(id)});\r\n    }\r\n\r\n    async find(where: FilterQuery<IAsset>): Promise<IAsset> {\r\n        const data = await this.collection.findOne(where);\r\n        return !data ? null : new Asset(data._id, data, this.collection, this.bucket);\r\n    }\r\n\r\n    async findMany(where: FilterQuery<IAsset>): Promise<ReadonlyArray<IAsset>> {\r\n        const cursor = this.collection.find(where);\r\n        const items = await cursor.toArray() || [];\r\n        const result: IAsset[] = [];\r\n        for (let item of items) {\r\n            if (!item) continue;\r\n            result.push(new Asset(item._id, item, this.collection, this.bucket));\r\n        }\r\n        return result;\r\n    }\r\n\r\n    async deleteMany(where: FilterQuery<IAsset>): Promise<ReadonlyArray<string>> {\r\n        const assets = await this.findMany(where);\r\n        return Promise.all(assets.map(a => a.unlink()));\r\n    }\r\n\r\n    async unlink(id: string): Promise<any> {\r\n        const asset = await this.read(id);\r\n        if (!asset) return null;\r\n        return asset.unlink();\r\n    }\r\n\r\n    protected async upload(stream: Readable, fileType: IFileType, metadata: IAssetMeta): Promise<IAsset> {\r\n        const contentType = fileType.mime.trim();\r\n        metadata = Object.assign({\r\n            downloadCount: 0,\r\n            firstDownload: null,\r\n            lastDownload: null\r\n        }, metadata || {});\r\n        metadata.filename = metadata.filename || new ObjectId().toHexString();\r\n        metadata.extension = (fileType.ext || \"\").trim();\r\n        return new Promise<IAsset>(((resolve, reject) => {\r\n            const uploaderStream = this.bucket.openUploadStream(metadata.filename);\r\n            stream.pipe(uploaderStream)\r\n                .on(\"error\", error => {\r\n                    reject(error.message || error);\r\n                })\r\n                .on(\"finish\", () => {\r\n                    const asset = new Asset(uploaderStream.id as ObjectId, {\r\n                        filename: metadata.filename,\r\n                        contentType,\r\n                        metadata\r\n                    }, this.collection, this.bucket);\r\n                    asset.save().then(() => {\r\n                        resolve(asset);\r\n                    }, error => {\r\n                        reject(error.message || error);\r\n                    });\r\n                });\r\n        }));\r\n    }\r\n}\r\n"]}
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assets.js","sourceRoot":"","sources":["../../../src/services/assets.ts"],"names":[],"mappings":"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","sourcesContent":["import {injectable, Lifecycle, scoped} from \"tsyringe\";\r\nimport {Readable} from \"stream\";\r\nimport {ObjectId} from \"bson\";\r\nimport {Collection, GridFSBucket} from \"mongodb\";\r\nimport {FilterQuery} from \"mongoose\";\r\nimport axios from \"axios\";\r\n\r\nimport {bufferToStream, copyStream, streamToBuffer, fileTypeFromBuffer} from \"../utils\";\r\nimport {IAsset, IAssetMeta, IFileType} from \"../common-types\";\r\nimport {MongoConnector} from \"./mongo-connector\";\r\nimport {AssetProcessor} from \"./asset-processor\";\r\nimport {Asset} from \"./entities/asset\";\r\nimport {TempAsset} from \"./entities/temp-asset\";\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class Assets {\r\n\r\n    readonly bucket: GridFSBucket;\r\n    readonly collection: Collection<Partial<IAsset>>;\r\n\r\n    constructor(readonly connector: MongoConnector, readonly assetProcessor: AssetProcessor) {\r\n        this.bucket = connector.bucket;\r\n        this.collection = connector.database?.collection(\"assets.files\");\r\n    }\r\n\r\n    async write(stream: Readable, contentType: string = null, metadata: IAssetMeta = null): Promise<IAsset> {\r\n        const uploadStream = copyStream(stream);\r\n        const buffer = await streamToBuffer(stream);\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw new Error(`Can't determine mime type: ${e}`);\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        metadata = metadata || {};\r\n        return this.upload(uploadStream, fileType, metadata);\r\n    }\r\n\r\n    async writeBuffer(buffer: Buffer, metadata: IAssetMeta = null, contentType: string = null): Promise<IAsset> {\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw `Can't determine mime type`;\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        metadata = metadata || {};\r\n        buffer = await this.assetProcessor.process(buffer, metadata, fileType);\r\n        return this.upload(bufferToStream(buffer), fileType, metadata);\r\n    }\r\n\r\n    async writeUrl(url: string, metadata: IAssetMeta = null): Promise<IAsset> {\r\n        metadata = metadata || {};\r\n        metadata.filename = metadata.filename || url;\r\n        metadata.url = url;\r\n        metadata.uploadTime = new Date().getTime();\r\n        const oneWeek = 1000 * 3600 * 24 * 7;\r\n        const asset = await this.find({\"metadata.url\": url, \"metadata.uploadTime\": {$gt: metadata.uploadTime - oneWeek}});\r\n        if (asset) return asset;\r\n        const buffer = (await axios({ url, responseType: \"arraybuffer\" })).data as Buffer;\r\n        return this.writeBuffer(buffer, metadata);\r\n    }\r\n\r\n    async download(url: string, contentType: string = null): Promise<IAsset> {\r\n        let buffer = (await axios({ url, responseType: \"arraybuffer\" })).data as Buffer;\r\n        let fileType = {ext: \"\", mime: contentType} as IFileType;\r\n        try {\r\n            fileType = await fileTypeFromBuffer(buffer);\r\n        } catch (e) {\r\n            if (!fileType.mime) {\r\n                throw `Can't determine mime type`;\r\n            }\r\n            console.log(`Can't determine mime type`, e);\r\n        }\r\n        const metadata: IAssetMeta = {\r\n            filename: url,\r\n            extension: (fileType.ext || \"\").trim()\r\n        };\r\n        buffer = await this.assetProcessor.process(buffer, metadata, fileType);\r\n        return new TempAsset(buffer, url, fileType.mime, metadata);\r\n    }\r\n\r\n    async read(id: string): Promise<IAsset> {\r\n        return !id ? null : this.find({_id: new ObjectId(id)});\r\n    }\r\n\r\n    async find(where: FilterQuery<IAsset>): Promise<IAsset> {\r\n        const data = await this.collection.findOne(where);\r\n        return !data ? null : new Asset(data._id, data, this.collection, this.bucket);\r\n    }\r\n\r\n    async findMany(where: FilterQuery<IAsset>): Promise<ReadonlyArray<IAsset>> {\r\n        const cursor = this.collection.find(where);\r\n        const items = await cursor.toArray() || [];\r\n        const result: IAsset[] = [];\r\n        for (let item of items) {\r\n            if (!item) continue;\r\n            result.push(new Asset(item._id, item, this.collection, this.bucket));\r\n        }\r\n        return result;\r\n    }\r\n\r\n    async deleteMany(where: FilterQuery<IAsset>): Promise<ReadonlyArray<string>> {\r\n        const assets = await this.findMany(where);\r\n        return Promise.all(assets.map(a => a.unlink()));\r\n    }\r\n\r\n    async unlink(id: string): Promise<any> {\r\n        const asset = await this.read(id);\r\n        if (!asset) return null;\r\n        return asset.unlink();\r\n    }\r\n\r\n    protected async upload(stream: Readable, fileType: IFileType, metadata: IAssetMeta): Promise<IAsset> {\r\n        const contentType = fileType.mime.trim();\r\n        metadata = Object.assign({\r\n            downloadCount: 0,\r\n            firstDownload: null,\r\n            lastDownload: null\r\n        }, metadata || {});\r\n        metadata.filename = metadata.filename || new ObjectId().toHexString();\r\n        metadata.extension = (fileType.ext || \"\").trim();\r\n        return new Promise<IAsset>(((resolve, reject) => {\r\n            const uploaderStream = this.bucket.openUploadStream(metadata.filename);\r\n            stream.pipe(uploaderStream)\r\n                .on(\"error\", error => {\r\n                    reject(error.message || error);\r\n                })\r\n                .on(\"finish\", () => {\r\n                    const asset = new Asset(uploaderStream.id as ObjectId, {\r\n                        filename: metadata.filename,\r\n                        contentType,\r\n                        metadata\r\n                    }, this.collection, this.bucket);\r\n                    asset.save().then(() => {\r\n                        resolve(asset);\r\n                    }, error => {\r\n                        reject(error.message || error);\r\n                    });\r\n                });\r\n        }));\r\n    }\r\n}\r\n"]}
@@ -27,20 +27,21 @@ let Configuration = class Configuration {
27
27
  existingParam.resolver = param.resolver || existingParam.resolver;
28
28
  this.paramMap[param.name] = existingParam;
29
29
  }
30
- resolveValue(param) {
30
+ resolveValue(param, alreadyResolved) {
31
31
  const envName = param.name.replace(/\.?([A-Z|0-9]+)/g, function (x, y) {
32
32
  return "_" + y.toLowerCase();
33
33
  }).replace(/\./gi, "_").replace(/^_/, "").toUpperCase();
34
34
  const envValue = process.env[envName];
35
+ const helper = (p) => this.resolveInternal(p, alreadyResolved);
35
36
  if (typeof envValue !== "undefined") {
36
37
  const value = isFunction(param.resolver)
37
- ? param.resolver(envValue)
38
+ ? param.resolver(envValue, helper)
38
39
  : convertValue(envValue, getType(param.defaultValue));
39
40
  console.log(colorize(`Processing param value`, ConsoleColor.FgYellow), colorize(param.name, ConsoleColor.FgGreen), colorize(envName, ConsoleColor.FgBlue), `"${envValue}"`, value);
40
41
  return value;
41
42
  }
42
43
  else if (isFunction(param.resolver)) {
43
- const value = param.resolver(param.defaultValue);
44
+ const value = param.resolver(param.defaultValue, helper);
44
45
  console.log(colorize(`Processing default param value`, ConsoleColor.FgYellow), colorize(param.name, ConsoleColor.FgGreen), param.defaultValue, value);
45
46
  return value;
46
47
  }
@@ -51,11 +52,18 @@ let Configuration = class Configuration {
51
52
  return !!this.paramMap[name];
52
53
  }
53
54
  resolve(name) {
55
+ return this.resolveInternal(name, []);
56
+ }
57
+ resolveInternal(name, alreadyResolved) {
58
+ if (alreadyResolved.includes(name)) {
59
+ throw new Error(`Circular dependency detected: ${alreadyResolved.join(" -> ")} -> ${name}`);
60
+ }
61
+ alreadyResolved.push(name);
54
62
  const param = this.paramMap[name];
55
63
  if (!param)
56
64
  throw new Error(`Parameter with name: '${name}' does not exists in configuration`);
57
65
  if (!(name in this.paramValues)) {
58
- this.paramValues[name] = this.resolveValue(param);
66
+ this.paramValues[name] = this.resolveValue(param, alreadyResolved);
59
67
  }
60
68
  return this.paramValues[name];
61
69
  }
@@ -66,4 +74,4 @@ Configuration = __decorate([
66
74
  __metadata("design:paramtypes", [Array])
67
75
  ], Configuration);
68
76
  export { Configuration };
69
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlndXJhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZXJ2aWNlcy9jb25maWd1cmF0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7OztBQUFBLE9BQU8sRUFBQyxTQUFTLEVBQUUsU0FBUyxFQUFDLE1BQU0sVUFBVSxDQUFDO0FBQzlDLE9BQU8sTUFBTSxNQUFNLFFBQVEsQ0FBQztBQUU1QixPQUFPLEVBQUMsU0FBUyxFQUFZLE1BQU0saUJBQWlCLENBQUM7QUFDckQsT0FBTyxFQUFDLFFBQVEsRUFBRSxZQUFZLEVBQUUsWUFBWSxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUMsTUFBTSxVQUFVLENBQUM7SUFHdEUsYUFBYSxTQUFiLGFBQWE7SUFLdEIsWUFBa0MsTUFBbUI7UUFDakQsTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ2hCLElBQUksQ0FBQyxRQUFRLEdBQUcsRUFBRSxDQUFDO1FBQ25CLElBQUksQ0FBQyxXQUFXLEdBQUcsRUFBRSxDQUFDO1FBQ3RCLENBQUMsTUFBTSxJQUFJLEVBQUUsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBRVMsR0FBRyxDQUFDLEtBQWdCO1FBQzFCLE1BQU0sYUFBYSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLEtBQUssQ0FBQztRQUN6RCxhQUFhLENBQUMsWUFBWSxHQUFHLEtBQUssQ0FBQyxZQUFZLENBQUM7UUFDaEQsYUFBYSxDQUFDLFFBQVEsR0FBRyxLQUFLLENBQUMsUUFBUSxJQUFJLGFBQWEsQ0FBQyxRQUFRLENBQUM7UUFDbEUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsYUFBYSxDQUFDO0lBQzlDLENBQUM7SUFFUyxZQUFZLENB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ImltcG9ydCB7aW5qZWN0QWxsLCBzaW5nbGV0b259IGZyb20gXCJ0c3lyaW5nZVwiO1xyXG5pbXBvcnQgZG90ZW52IGZyb20gXCJkb3RlbnZcIjtcclxuXHJcbmltcG9ydCB7UEFSQU1FVEVSLCBQYXJhbWV0ZXJ9IGZyb20gXCIuLi9jb21tb24tdHlwZXNcIjtcclxuaW1wb3J0IHtjb2xvcml6ZSwgQ29uc29sZUNvbG9yLCBjb252ZXJ0VmFsdWUsIGdldFR5cGUsIGlzRnVuY3Rpb259IGZyb20gXCIuLi91dGlsc1wiO1xyXG5cclxuQHNpbmdsZXRvbigpXHJcbmV4cG9ydCBjbGFzcyBDb25maWd1cmF0aW9uIHtcclxuXHJcbiAgICBwcm90ZWN0ZWQgcGFyYW1NYXA6IHtbbmFtZTogc3RyaW5nXTogUGFyYW1ldGVyfTtcclxuICAgIHByb3RlY3RlZCBwYXJhbVZhbHVlczoge1tuYW1lOiBzdHJpbmddOiBhbnl9O1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKEBpbmplY3RBbGwoUEFSQU1FVEVSKSBwYXJhbXM6IFBhcmFtZXRlcltdKSB7XHJcbiAgICAgICAgZG90ZW52LmNvbmZpZygpO1xyXG4gICAgICAgIHRoaXMucGFyYW1NYXAgPSB7fTtcclxuICAgICAgICB0aGlzLnBhcmFtVmFsdWVzID0ge307XHJcbiAgICAgICAgKHBhcmFtcyB8fCBbXSkuZm9yRWFjaChwYXJhbSA9PiB0aGlzLmFkZChwYXJhbSkpO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBhZGQocGFyYW06IFBhcmFtZXRlcik6IHZvaWQge1xyXG4gICAgICAgIGNvbnN0IGV4aXN0aW5nUGFyYW0gPSB0aGlzLnBhcmFtTWFwW3BhcmFtLm5hbWVdIHx8IHBhcmFtO1xyXG4gICAgICAgIGV4aXN0aW5nUGFyYW0uZGVmYXVsdFZhbHVlID0gcGFyYW0uZGVmYXVsdFZhbHVlO1xyXG4gICAgICAgIGV4aXN0aW5nUGFyYW0ucmVzb2x2ZXIgPSBwYXJhbS5yZXNvbHZlciB8fCBleGlzdGluZ1BhcmFtLnJlc29sdmVyO1xyXG4gICAgICAgIHRoaXMucGFyYW1NYXBbcGFyYW0ubmFtZV0gPSBleGlzdGluZ1BhcmFtO1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCByZXNvbHZlVmFsdWUocGFyYW06IFBhcmFtZXRlcik6IGFueSB7XHJcbiAgICAgICAgY29uc3QgZW52TmFtZSA9IHBhcmFtLm5hbWUucmVwbGFjZSgvXFwuPyhbQS1afDAtOV0rKS9nLCBmdW5jdGlvbiAoeCwgeSkge1xyXG4gICAgICAgICAgICByZXR1cm4gXCJfXCIgKyB5LnRvTG93ZXJDYXNlKClcclxuICAgICAgICB9KS5yZXBsYWNlKC9cXC4vZ2ksIFwiX1wiKS5yZXBsYWNlKC9eXy8sIFwiXCIpLnRvVXBwZXJDYXNlKCk7XHJcbiAgICAgICAgY29uc3QgZW52VmFsdWUgPSBwcm9jZXNzLmVudltlbnZOYW1lXTtcclxuICAgICAgICBpZiAodHlwZW9mIGVudlZhbHVlICE9PSBcInVuZGVmaW5lZFwiKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHZhbHVlID0gaXNGdW5jdGlvbihwYXJhbS5yZXNvbHZlcilcclxuICAgICAgICAgICAgICAgID8gcGFyYW0ucmVzb2x2ZXIoZW52VmFsdWUpXHJcbiAgICAgICAgICAgICAgICA6IGNvbnZlcnRWYWx1ZShlbnZWYWx1ZSwgZ2V0VHlwZShwYXJhbS5kZWZhdWx0VmFsdWUpKTtcclxuICAgICAgICAgICAgY29uc29sZS5sb2coXHJcbiAgICAgICAgICAgICAgICBjb2xvcml6ZShgUHJvY2Vzc2luZyBwYXJhbSB2YWx1ZWAsIENvbnNvbGVDb2xvci5GZ1llbGxvdyksXHJcbiAgICAgICAgICAgICAgICBjb2xvcml6ZShwYXJhbS5uYW1lLCBDb25zb2xlQ29sb3IuRmdHcmVlbiksXHJcbiAgICAgICAgICAgICAgICBjb2xvcml6ZShlbnZOYW1lLCBDb25zb2xlQ29sb3IuRmdCbHVlKSxcclxuICAgICAgICAgICAgICAgIGBcIiR7ZW52VmFsdWV9XCJgLFxyXG4gICAgICAgICAgICAgICAgdmFsdWVcclxuICAgICAgICAgICAgKTtcclxuICAgICAgICAgICAgcmV0dXJuIHZhbHVlO1xyXG4gICAgICAgIH0gZWxzZSBpZiAoaXNGdW5jdGlvbihwYXJhbS5yZXNvbHZlcikpIHtcclxuICAgICAgICAgICAgY29uc3QgdmFsdWUgPSBwYXJhbS5yZXNvbHZlcihwYXJhbS5kZWZhdWx0VmFsdWUpO1xyXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcclxuICAgICAgICAgICAgICAgIGNvbG9yaXplKGBQcm9jZXNzaW5nIGRlZmF1bHQgcGFyYW0gdmFsdWVgLCBDb25zb2xlQ29sb3IuRmdZZWxsb3cpLFxyXG4gICAgICAgICAgICAgICAgY29sb3JpemUocGFyYW0ubmFtZSwgQ29uc29sZUNvbG9yLkZnR3JlZW4pLFxyXG4gICAgICAgICAgICAgICAgcGFyYW0uZGVmYXVsdFZhbHVlLFxyXG4gICAgICAgICAgICAgICAgdmFsdWVcclxuICAgICAgICAgICAgKTtcclxuICAgICAgICAgICAgcmV0dXJuIHZhbHVlO1xyXG4gICAgICAgIH1cclxuICAgICAgICBjb25zb2xlLmxvZyhcclxuICAgICAgICAgICAgY29sb3JpemUoYFVzaW5nIGRlZmF1bHQgcGFyYW0gdmFsdWVgLCBDb25zb2xlQ29sb3IuRmdZZWxsb3cpLFxyXG4gICAgICAgICAgICBjb2xvcml6ZShwYXJhbS5uYW1lLCBDb25zb2xlQ29sb3IuRmdHcmVlbiksXHJcbiAgICAgICAgICAgIHBhcmFtLmRlZmF1bHRWYWx1ZSxcclxuICAgICAgICApO1xyXG4gICAgICAgIHJldHVybiBwYXJhbS5kZWZhdWx0VmFsdWU7XHJcbiAgICB9XHJcblxyXG4gICAgaGFzUGFyYW0obmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XHJcbiAgICAgICAgcmV0dXJuICEhdGhpcy5wYXJhbU1hcFtuYW1lXTtcclxuICAgIH1cclxuXHJcbiAgICByZXNvbHZlKG5hbWU6IHN0cmluZyk6IGFueSB7XHJcbiAgICAgICAgY29uc3QgcGFyYW0gPSB0aGlzLnBhcmFtTWFwW25hbWVdO1xyXG4gICAgICAgIGlmICghcGFyYW0pIHRocm93IG5ldyBFcnJvcihgUGFyYW1ldGVyIHdpdGggbmFtZTogJyR7bmFtZX0nIGRvZXMgbm90IGV4aXN0cyBpbiBjb25maWd1cmF0aW9uYCk7XHJcbiAgICAgICAgaWYgKCEobmFtZSBpbiB0aGlzLnBhcmFtVmFsdWVzKSkge1xyXG4gICAgICAgICAgICB0aGlzLnBhcmFtVmFsdWVzW25hbWVdID0gdGhpcy5yZXNvbHZlVmFsdWUocGFyYW0pO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gdGhpcy5wYXJhbVZhbHVlc1tuYW1lXTtcclxuICAgIH1cclxufVxyXG4iXX0=
77
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration.js","sourceRoot":"","sources":["../../../src/services/configuration.ts"],"names":[],"mappings":"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","sourcesContent":["import {injectAll, singleton} from \"tsyringe\";\r\nimport dotenv from \"dotenv\";\r\n\r\nimport {PARAMETER, Parameter} from \"../common-types\";\r\nimport {colorize, ConsoleColor, convertValue, getType, isFunction} from \"../utils\";\r\nimport {help} from \"commander\";\r\n\r\n@singleton()\r\nexport class Configuration {\r\n\r\n    protected paramMap: {[name: string]: Parameter};\r\n    protected paramValues: {[name: string]: any};\r\n\r\n    constructor(@injectAll(PARAMETER) params: Parameter[]) {\r\n        dotenv.config();\r\n        this.paramMap = {};\r\n        this.paramValues = {};\r\n        (params || []).forEach(param => this.add(param));\r\n    }\r\n\r\n    protected add(param: Parameter): void {\r\n        const existingParam = this.paramMap[param.name] || param;\r\n        existingParam.defaultValue = param.defaultValue;\r\n        existingParam.resolver = param.resolver || existingParam.resolver;\r\n        this.paramMap[param.name] = existingParam;\r\n    }\r\n\r\n    protected resolveValue(param: Parameter, alreadyResolved: string[]): any {\r\n        const envName = param.name.replace(/\\.?([A-Z|0-9]+)/g, function (x, y) {\r\n            return \"_\" + y.toLowerCase()\r\n        }).replace(/\\./gi, \"_\").replace(/^_/, \"\").toUpperCase();\r\n        const envValue = process.env[envName];\r\n        const helper = (p: string) => this.resolveInternal(p, alreadyResolved);\r\n        if (typeof envValue !== \"undefined\") {\r\n            const value = isFunction(param.resolver)\r\n                ? param.resolver(envValue, helper)\r\n                : convertValue(envValue, getType(param.defaultValue));\r\n            console.log(\r\n                colorize(`Processing param value`, ConsoleColor.FgYellow),\r\n                colorize(param.name, ConsoleColor.FgGreen),\r\n                colorize(envName, ConsoleColor.FgBlue),\r\n                `\"${envValue}\"`,\r\n                value\r\n            );\r\n            return value;\r\n        } else if (isFunction(param.resolver)) {\r\n            const value = param.resolver(param.defaultValue, helper);\r\n            console.log(\r\n                colorize(`Processing default param value`, ConsoleColor.FgYellow),\r\n                colorize(param.name, ConsoleColor.FgGreen),\r\n                param.defaultValue,\r\n                value\r\n            );\r\n            return value;\r\n        }\r\n        console.log(\r\n            colorize(`Using default param value`, ConsoleColor.FgYellow),\r\n            colorize(param.name, ConsoleColor.FgGreen),\r\n            param.defaultValue,\r\n        );\r\n        return param.defaultValue;\r\n    }\r\n\r\n    hasParam(name: string): boolean {\r\n        return !!this.paramMap[name];\r\n    }\r\n\r\n    resolve(name: string): any {\r\n        return this.resolveInternal(name, []);\r\n    }\r\n\r\n    protected resolveInternal(name: string, alreadyResolved: string[]): any {\r\n        if (alreadyResolved.includes(name)) {\r\n            throw new Error(`Circular dependency detected: ${alreadyResolved.join(\" -> \")} -> ${name}`);\r\n        }\r\n        alreadyResolved.push(name);\r\n        const param = this.paramMap[name];\r\n        if (!param) throw new Error(`Parameter with name: '${name}' does not exists in configuration`);\r\n        if (!(name in this.paramValues)) {\r\n            this.paramValues[name] = this.resolveValue(param, alreadyResolved);\r\n        }\r\n        return this.paramValues[name];\r\n    }\r\n}\r\n"]}
@@ -16,7 +16,7 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
16
16
  step((generator = generator.apply(thisArg, _arguments || [])).next());
17
17
  });
18
18
  };
19
- import { injectable, singleton } from "tsyringe";
19
+ import { singleton } from "tsyringe";
20
20
  import { lstatSync, readdir, readFileSync } from "fs";
21
21
  import { join } from "path";
22
22
  import * as Handlebars from "handlebars";
@@ -81,9 +81,8 @@ let TemplateRenderer = class TemplateRenderer {
81
81
  }
82
82
  };
83
83
  TemplateRenderer = __decorate([
84
- injectable(),
85
84
  singleton(),
86
85
  __metadata("design:paramtypes", [Translator, Configuration])
87
86
  ], TemplateRenderer);
88
87
  export { TemplateRenderer };
89
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVtcGxhdGUtcmVuZGVyZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmljZXMvdGVtcGxhdGUtcmVuZGVyZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsT0FBTyxFQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUMsTUFBTSxVQUFVLENBQUM7QUFDL0MsT0FBTyxFQUFDLFNBQVMsRUFBRSxPQUFPLEVBQUUsWUFBWSxFQUFDLE1BQU0sSUFBSSxDQUFDO0FBQ3BELE9BQU8sRUFBQyxJQUFJLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDMUIsT0FBTyxLQUFLLFVBQVUsTUFBTSxZQUFZLENBQUM7QUFFekMsT0FBTyxFQUFDLFVBQVUsRUFBQyxNQUFNLGNBQWMsQ0FBQztBQUN4QyxPQUFPLEVBQUMsYUFBYSxFQUFDLE1BQU0saUJBQWlCLENBQUM7SUFJakMsZ0JBQWdCLFNBQWhCLGdCQUFnQjtJQU16QixZQUFxQixVQUFzQixFQUFXLE1BQXFCO1FBQXRELGVBQVUsR0FBVixVQUFVLENBQVk7UUFBVyxXQUFNLEdBQU4sTUFBTSxDQUFlO1FBQ3ZFLElBQUksQ0FBQyxTQUFTLEdBQUcsRUFBRSxDQUFDO1FBQ3BCLFVBQVUsQ0FBQyxjQUFjLENBQUMsUUFBUSxFQUFFLFVBQVMsRUFBQyxJQUFJLEVBQUM7WUFDL0MsT0FBTyxJQUFJLENBQUM7UUFDaEIsQ0FBQyxDQUFDLENBQUM7UUFDSCxVQUFVLENBQUMsY0FBYyxDQUFDLEtBQUssRUFBRTtZQUM3QixPQUFPLElBQUksSUFBSSxFQUFFLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDaEMsQ0FBQyxDQUFDLENBQUM7UUFDSCxVQUFVLENBQUMsY0FBYyxDQUFDLE1BQU0sRUFBRSxVQUFTLEdBQVE7WUFDL0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FB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QUFHLENBQUM7UUFDbEUsQ0FBQztLQUFBO0NBQ0osQ0FBQTtBQTNEWSxnQkFBZ0I7SUFGNUIsVUFBVSxFQUFFO0lBQ1osU0FBUyxFQUFFO3FDQU95QixVQUFVLEVBQW1CLGFBQWE7R0FObEUsZ0JBQWdCLENBMkQ1QjtTQTNEWSxnQkFBZ0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge2luamVjdGFibGUsIHNpbmdsZXRvbn0gZnJvbSBcInRzeXJpbmdlXCI7XHJcbmltcG9ydCB7bHN0YXRTeW5jLCByZWFkZGlyLCByZWFkRmlsZVN5bmN9IGZyb20gXCJmc1wiO1xyXG5pbXBvcnQge2pvaW59IGZyb20gXCJwYXRoXCI7XHJcbmltcG9ydCAqIGFzIEhhbmRsZWJhcnMgZnJvbSBcImhhbmRsZWJhcnNcIjtcclxuXHJcbmltcG9ydCB7VHJhbnNsYXRvcn0gZnJvbSBcIi4vdHJhbnNsYXRvclwiO1xyXG5pbXBvcnQge0NvbmZpZ3VyYXRpb259IGZyb20gXCIuL2NvbmZpZ3VyYXRpb25cIjtcclxuXHJcbkBpbmplY3RhYmxlKClcclxuQHNpbmdsZXRvbigpXHJcbmV4cG9ydCBjbGFzcyBUZW1wbGF0ZVJlbmRlcmVyIHtcclxuXHJcbiAgICB0ZW1wbGF0ZXM6IHtbbmFtZTogc3RyaW5nXTogRnVuY3Rpb259O1xyXG5cclxuICAgIHByb3RlY3RlZCBpbml0UHJvbWlzZTogUHJvbWlzZTxhbnk+O1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKHJlYWRvbmx5IHRyYW5zbGF0b3I6IFRyYW5zbGF0b3IsIHJlYWRvbmx5IGNvbmZpZzogQ29uZmlndXJhdGlvbikge1xyXG4gICAgICAgIHRoaXMudGVtcGxhdGVzID0ge307XHJcbiAgICAgICAgSGFuZGxlYmFycy5yZWdpc3RlckhlbHBlcihgb2JqZWN0YCwgZnVuY3Rpb24oe2hhc2h9KSB7XHJcbiAgICAgICAgICAgIHJldHVybiBoYXNoO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIEhhbmRsZWJhcnMucmVnaXN0ZXJIZWxwZXIoYG5vd2AsIGZ1bmN0aW9uKCkge1xyXG4gICAgICAgICAgICByZXR1cm4gbmV3IERhdGUoKS5nZXRUaW1lKCk7XHJcbiAgICAgICAgfSk7XHJcbiAgICAgICAgSGFuZGxlYmFycy5yZWdpc3RlckhlbHBlcihga2V5c2AsIGZ1bmN0aW9uKG9iajogYW55KSB7XHJcbiAgICAgICAgICAgIHJldHVybiAhb2JqID8gW10gOiBPYmplY3Qua2V5cyhvYmopO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIEhhbmRsZWJhcnMucmVnaXN0ZXJIZWxwZXIoYHRyYW5zbGF0ZWAsIGZ1bmN0aW9uIChrZXk6IHN0cmluZywgcGFyYW1zOiBhbnkpIHtcclxuICAgICAgICAgICAgcmV0dXJuIHRyYW5zbGF0b3IuZ2V0VHJhbnNsYXRpb25TeW5jKHRoaXMubGFuZ3VhZ2UsIGtleSwgcGFyYW1zKTtcclxuICAgICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgaW5pdCgpOiBQcm9taXNlPGFueT4ge1xyXG4gICAgICAgIHRoaXMuaW5pdFByb21pc2UgPSB0aGlzLmluaXRQcm9taXNlIHx8IHRoaXMucGFyc2VUZW1wbGF0ZXModGhpcy5jb25maWcucmVzb2x2ZShcInRlbXBsYXRlc0RpclwiKSwgW10pO1xyXG4gICAgICAgIHJldHVybiB0aGlzLmluaXRQcm9taXNlO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHBhcnNlVGVtcGxhdGVzKGRpcjogc3RyaW5nLCBkaXJQYXRoOiBzdHJpbmdbXSk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgICAgIHJldHVybiBuZXcgUHJvbWlzZTx2b2lkPihyZXNvbHZlID0+IHtcclxuICAgICAgICAgICAgcmVhZGRpcihkaXIsIGFzeW5jIChlcnIsIGZpbGVzKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBmb3IgKGxldCBmaWxlIG9mIGZpbGVzKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgcGF0aCA9IGpvaW4oZGlyLCBmaWxlKTtcclxuICAgICAgICAgICAgICAgICAgICBpZiAobHN0YXRTeW5jKHBhdGgpLmlzRGlyZWN0b3J5KCkpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgYXdhaXQgdGhpcy5wYXJzZVRlbXBsYXRlcyhqb2luKGRpciwgZmlsZSksIGRpclBhdGguY29uY2F0KFtmaWxlXSkpO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTtcclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgcGFydHMgPSBmaWxlLnNwbGl0KFwiLlwiKTtcclxuICAgICAgICAgICAgICAgICAgICBwYXJ0cy5wb3AoKTtcclxuICAgICAgICAgICAgICAgICAgICBjb25zdCBuYW1lID0gcGFydHMuam9pbihcIi5cIik7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZnVsbE5hbWUgPSBkaXJQYXRoLmNvbmNhdChbbmFtZV0pLmpvaW4oXCItXCIpO1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbnRlbnQgPSByZWFkRmlsZVN5bmMocGF0aCkudG9TdHJpbmcoXCJ1dGY4XCIpO1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMudGVtcGxhdGVzW2Z1bGxOYW1lXSA9IEhhbmRsZWJhcnMuY29tcGlsZShjb250ZW50KTtcclxuICAgICAgICAgICAgICAgICAgICBIYW5kbGViYXJzLnJlZ2lzdGVyUGFydGlhbChmdWxsTmFtZSwgY29udGVudCk7XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICByZXNvbHZlKCk7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHJlbmRlcih0ZW1wbGF0ZTogc3RyaW5nLCBsYW5ndWFnZTogc3RyaW5nLCBjb250ZXh0PzogYW55KTogUHJvbWlzZTxzdHJpbmc+IHtcclxuICAgICAgICBhd2FpdCB0aGlzLmluaXQoKTtcclxuICAgICAgICBhd2FpdCB0aGlzLnRyYW5zbGF0b3IuZ2V0RGljdGlvbmFyeShsYW5ndWFnZSk7XHJcbiAgICAgICAgaWYgKCF0aGlzLnRlbXBsYXRlc1t0ZW1wbGF0ZV0pIHtcclxuICAgICAgICAgICAgcmV0dXJuIFByb21pc2UucmVqZWN0KGBUZW1wbGF0ZSBub3QgZm91bmQgd2l0aCBuYW1lOiAke3RlbXBsYXRlfWApO1xyXG4gICAgICAgIH1cclxuICAgICAgICBjb250ZXh0ID0gT2JqZWN0LmFzc2lnbih7bGFuZ3VhZ2V9LCBjb250ZXh0IHx8IHt9KTtcclxuICAgICAgICBjb25zdCByZXMgPSB0aGlzLnRlbXBsYXRlc1t0ZW1wbGF0ZV0oY29udGV4dCk7XHJcbiAgICAgICAgcmV0dXJuIHJlcyBpbnN0YW5jZW9mIEVycm9yID8gYXdhaXQgUHJvbWlzZS5yZWplY3QocmVzKSA6IHJlcztcclxuICAgIH1cclxufVxyXG4iXX0=
88
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVtcGxhdGUtcmVuZGVyZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmljZXMvdGVtcGxhdGUtcmVuZGVyZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsT0FBTyxFQUFDLFNBQVMsRUFBQyxNQUFNLFVBQVUsQ0FBQztBQUNuQyxPQUFPLEVBQUMsU0FBUyxFQUFFLE9BQU8sRUFBRSxZQUFZLEVBQUMsTUFBTSxJQUFJLENBQUM7QUFDcEQsT0FBTyxFQUFDLElBQUksRUFBQyxNQUFNLE1BQU0sQ0FBQztBQUMxQixPQUFPLEtBQUssVUFBVSxNQUFNLFlBQVksQ0FBQztBQUV6QyxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0sY0FBYyxDQUFDO0FBQ3hDLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztJQUdqQyxnQkFBZ0IsU0FBaEIsZ0JBQWdCO0lBTXpCLFlBQXFCLFVBQXNCLEVBQVcsTUFBcUI7UUFBdEQsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUFXLFdBQU0sR0FBTixNQUFNLENBQWU7UUFDdkUsSUFBSSxDQUFDLFNBQVMsR0FBRyxFQUFFLENBQUM7UUFDcEIsVUFBVSxDQUFDLGNBQWMsQ0FBQyxRQUFRLEVBQUUsVUFBUyxFQUFDLElBQUksRUFBQztZQUMvQyxPQUFPLElBQUksQ0FBQztRQUNoQixDQUFDLENBQUMsQ0FBQztRQUNILFVBQVUsQ0FBQyxjQUFjLENBQUMsS0FBSyxFQUFFO1lBQzdCLE9BQU8sSUFBSSxJQUFJLEVBQUUsQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUNoQyxDQUFDLENBQUMsQ0FBQztRQUNILFVBQVUsQ0FBQyxjQUFjLENBQUMsTUFBTSxFQUFFLFVBQVMsR0FBUTtZQUMvQyxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFD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QUNsRSxDQUFDO0tBQUE7Q0FDSixDQUFBO0FBM0RZLGdCQUFnQjtJQUQ1QixTQUFTLEVBQUU7cUNBT3lCLFVBQVUsRUFBbUIsYUFBYTtHQU5sRSxnQkFBZ0IsQ0EyRDVCO1NBM0RZLGdCQUFnQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7c2luZ2xldG9ufSBmcm9tIFwidHN5cmluZ2VcIjtcclxuaW1wb3J0IHtsc3RhdFN5bmMsIHJlYWRkaXIsIHJlYWRGaWxlU3luY30gZnJvbSBcImZzXCI7XHJcbmltcG9ydCB7am9pbn0gZnJvbSBcInBhdGhcIjtcclxuaW1wb3J0ICogYXMgSGFuZGxlYmFycyBmcm9tIFwiaGFuZGxlYmFyc1wiO1xyXG5cclxuaW1wb3J0IHtUcmFuc2xhdG9yfSBmcm9tIFwiLi90cmFuc2xhdG9yXCI7XHJcbmltcG9ydCB7Q29uZmlndXJhdGlvbn0gZnJvbSBcIi4vY29uZmlndXJhdGlvblwiO1xyXG5cclxuQHNpbmdsZXRvbigpXHJcbmV4cG9ydCBjbGFzcyBUZW1wbGF0ZVJlbmRlcmVyIHtcclxuXHJcbiAgICB0ZW1wbGF0ZXM6IHtbbmFtZTogc3RyaW5nXTogRnVuY3Rpb259O1xyXG5cclxuICAgIHByb3RlY3RlZCBpbml0UHJvbWlzZTogUHJvbWlzZTxhbnk+O1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKHJlYWRvbmx5IHRyYW5zbGF0b3I6IFRyYW5zbGF0b3IsIHJlYWRvbmx5IGNvbmZpZzogQ29uZmlndXJhdGlvbikge1xyXG4gICAgICAgIHRoaXMudGVtcGxhdGVzID0ge307XHJcbiAgICAgICAgSGFuZGxlYmFycy5yZWdpc3RlckhlbHBlcihgb2JqZWN0YCwgZnVuY3Rpb24oe2hhc2h9KSB7XHJcbiAgICAgICAgICAgIHJldHVybiBoYXNoO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIEhhbmRsZWJhcnMucmVnaXN0ZXJIZWxwZXIoYG5vd2AsIGZ1bmN0aW9uKCkge1xyXG4gICAgICAgICAgICByZXR1cm4gbmV3IERhdGUoKS5nZXRUaW1lKCk7XHJcbiAgICAgICAgfSk7XHJcbiAgICAgICAgSGFuZGxlYmFycy5yZWdpc3RlckhlbHBlcihga2V5c2AsIGZ1bmN0aW9uKG9iajogYW55KSB7XHJcbiAgICAgICAgICAgIHJldHVybiAhb2JqID8gW10gOiBPYmplY3Qua2V5cyhvYmopO1xyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIEhhbmRsZWJhcnMucmVnaXN0ZXJIZWxwZXIoYHRyYW5zbGF0ZWAsIGZ1bmN0aW9uIChrZXk6IHN0cmluZywgcGFyYW1zOiBhbnkpIHtcclxuICAgICAgICAgICAgcmV0dXJuIHRyYW5zbGF0b3IuZ2V0VHJhbnNsYXRpb25TeW5jKHRoaXMubGFuZ3VhZ2UsIGtleSwgcGFyYW1zKTtcclxuICAgICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBwcm90ZWN0ZWQgaW5pdCgpOiBQcm9taXNlPGFueT4ge1xyXG4gICAgICAgIHRoaXMuaW5pdFByb21pc2UgPSB0aGlzLmluaXRQcm9taXNlIHx8IHRoaXMucGFyc2VUZW1wbGF0ZXModGhpcy5jb25maWcucmVzb2x2ZShcInRlbXBsYXRlc0RpclwiKSwgW10pO1xyXG4gICAgICAgIHJldHVybiB0aGlzLmluaXRQcm9taXNlO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHBhcnNlVGVtcGxhdGVzKGRpcjogc3RyaW5nLCBkaXJQYXRoOiBzdHJpbmdbXSk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgICAgIHJldHVybiBuZXcgUHJvbWlzZTx2b2lkPihyZXNvbHZlID0+IHtcclxuICAgICAgICAgICAgcmVhZGRpcihkaXIsIGFzeW5jIChlcnIsIGZpbGVzKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBmb3IgKGxldCBmaWxlIG9mIGZpbGVzKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgcGF0aCA9IGpvaW4oZGlyLCBmaWxlKTtcclxuICAgICAgICAgICAgICAgICAgICBpZiAobHN0YXRTeW5jKHBhdGgpLmlzRGlyZWN0b3J5KCkpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgYXdhaXQgdGhpcy5wYXJzZVRlbXBsYXRlcyhqb2luKGRpciwgZmlsZSksIGRpclBhdGguY29uY2F0KFtmaWxlXSkpO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250aW51ZTtcclxuICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgcGFydHMgPSBmaWxlLnNwbGl0KFwiLlwiKTtcclxuICAgICAgICAgICAgICAgICAgICBwYXJ0cy5wb3AoKTtcclxuICAgICAgICAgICAgICAgICAgICBjb25zdCBuYW1lID0gcGFydHMuam9pbihcIi5cIik7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZnVsbE5hbWUgPSBkaXJQYXRoLmNvbmNhdChbbmFtZV0pLmpvaW4oXCItXCIpO1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbnRlbnQgPSByZWFkRmlsZVN5bmMocGF0aCkudG9TdHJpbmcoXCJ1dGY4XCIpO1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMudGVtcGxhdGVzW2Z1bGxOYW1lXSA9IEhhbmRsZWJhcnMuY29tcGlsZShjb250ZW50KTtcclxuICAgICAgICAgICAgICAgICAgICBIYW5kbGViYXJzLnJlZ2lzdGVyUGFydGlhbChmdWxsTmFtZSwgY29udGVudCk7XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICByZXNvbHZlKCk7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHJlbmRlcih0ZW1wbGF0ZTogc3RyaW5nLCBsYW5ndWFnZTogc3RyaW5nLCBjb250ZXh0PzogYW55KTogUHJvbWlzZTxzdHJpbmc+IHtcclxuICAgICAgICBhd2FpdCB0aGlzLmluaXQoKTtcclxuICAgICAgICBhd2FpdCB0aGlzLnRyYW5zbGF0b3IuZ2V0RGljdGlvbmFyeShsYW5ndWFnZSk7XHJcbiAgICAgICAgaWYgKCF0aGlzLnRlbXBsYXRlc1t0ZW1wbGF0ZV0pIHtcclxuICAgICAgICAgICAgcmV0dXJuIFByb21pc2UucmVqZWN0KGBUZW1wbGF0ZSBub3QgZm91bmQgd2l0aCBuYW1lOiAke3RlbXBsYXRlfWApO1xyXG4gICAgICAgIH1cclxuICAgICAgICBjb250ZXh0ID0gT2JqZWN0LmFzc2lnbih7bGFuZ3VhZ2V9LCBjb250ZXh0IHx8IHt9KTtcclxuICAgICAgICBjb25zdCByZXMgPSB0aGlzLnRlbXBsYXRlc1t0ZW1wbGF0ZV0oY29udGV4dCk7XHJcbiAgICAgICAgcmV0dXJuIHJlcyBpbnN0YW5jZW9mIEVycm9yID8gYXdhaXQgUHJvbWlzZS5yZWplY3QocmVzKSA6IHJlcztcclxuICAgIH1cclxufVxyXG4iXX0=