@stemy/backend 3.4.9 → 3.5.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.
- package/bundles/stemy-backend.umd.js +203 -142
- package/bundles/stemy-backend.umd.js.map +1 -1
- package/esm2015/public_api.js +4 -1
- package/esm2015/rest-middlewares/request-ended.middleware.js +12 -4
- package/esm2015/rest-middlewares/request-started.middleware.js +11 -3
- package/esm2015/services/configuration.js +19 -13
- package/esm2015/services/entities/lazy-asset.js +5 -4
- package/esm2015/services/job-manager.js +18 -14
- package/esm2015/services/lazy-assets.js +7 -4
- package/esm2015/services/logger.js +29 -0
- package/esm2015/stemy-backend.js +2 -1
- package/esm2015/utilities/base-doc.js +1 -2
- package/fesm2015/stemy-backend.js +193 -140
- package/fesm2015/stemy-backend.js.map +1 -1
- package/package.json +1 -1
- package/rest-middlewares/request-ended.middleware.d.ts +3 -0
- package/rest-middlewares/request-started.middleware.d.ts +3 -0
- package/services/configuration.d.ts +4 -0
- package/services/entities/lazy-asset.d.ts +3 -1
- package/services/job-manager.d.ts +3 -1
- package/services/lazy-assets.d.ts +3 -1
- package/services/logger.d.ts +7 -0
- package/stemy-backend.d.ts +1 -0
- package/stemy-backend.metadata.json +1 -1
- package/utilities/base-doc.d.ts +1 -1
|
@@ -28,9 +28,11 @@ import { ObjectId } from "bson";
|
|
|
28
28
|
import { DI_CONTAINER, JOB } from "../common-types";
|
|
29
29
|
import { colorize, ConsoleColor, getConstructorName, getType, isArray, isObject, jsonHighlight, promiseTimeout } from "../utils";
|
|
30
30
|
import { Configuration } from "./configuration";
|
|
31
|
+
import { Logger } from "./logger";
|
|
31
32
|
let JobManager = class JobManager {
|
|
32
|
-
constructor(config, container, jobTypes) {
|
|
33
|
+
constructor(config, logger, container, jobTypes) {
|
|
33
34
|
this.config = config;
|
|
35
|
+
this.logger = logger;
|
|
34
36
|
this.container = container;
|
|
35
37
|
this.jobTypes = jobTypes || [];
|
|
36
38
|
this.jobs = this.jobTypes.reduce((res, jobType) => {
|
|
@@ -65,7 +67,7 @@ let JobManager = class JobManager {
|
|
|
65
67
|
}
|
|
66
68
|
catch (e) {
|
|
67
69
|
const jobName = getConstructorName(jobType);
|
|
68
|
-
throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}
|
|
70
|
+
throw new Error(`Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`);
|
|
69
71
|
}
|
|
70
72
|
return instance.process();
|
|
71
73
|
});
|
|
@@ -93,12 +95,12 @@ let JobManager = class JobManager {
|
|
|
93
95
|
}).join(" ");
|
|
94
96
|
const jobName = getConstructorName(jobType);
|
|
95
97
|
if (!validate(expression)) {
|
|
96
|
-
|
|
98
|
+
this.logger.log("job-manager", `Can't schedule the task: '${jobName}' because time expression is invalid.`);
|
|
97
99
|
return null;
|
|
98
100
|
}
|
|
99
101
|
return schedule(expression, () => {
|
|
100
102
|
this.enqueue(jobType, params).catch(e => {
|
|
101
|
-
|
|
103
|
+
this.logger.log("job-manager", `Can't enqueue job: '${jobName}' because: ${e}`);
|
|
102
104
|
});
|
|
103
105
|
});
|
|
104
106
|
}
|
|
@@ -108,18 +110,18 @@ let JobManager = class JobManager {
|
|
|
108
110
|
return null;
|
|
109
111
|
this.processing = true;
|
|
110
112
|
if (!this.config.resolve("isWorker")) {
|
|
111
|
-
|
|
113
|
+
this.logger.log("job-manager", colorize(`Processing can not be started because this is NOT a worker process!`, ConsoleColor.FgRed));
|
|
112
114
|
return null;
|
|
113
115
|
}
|
|
114
116
|
const host = this.config.resolve("zmqRemoteHost");
|
|
115
117
|
const pushHost = `${host}:${this.config.resolve("zmqBackPort")}`;
|
|
116
118
|
this.workerPush = socket("push");
|
|
117
119
|
yield this.workerPush.connect(pushHost);
|
|
118
|
-
|
|
120
|
+
this.logger.log("job-manager", `Worker producer connected to: ${pushHost}`);
|
|
119
121
|
const pullHost = `${host}:${this.config.resolve("zmqPort")}`;
|
|
120
122
|
this.workerPull = socket("pull");
|
|
121
123
|
yield this.workerPull.connect(pullHost);
|
|
122
|
-
|
|
124
|
+
this.logger.log("job-manager", `Worker consumer connected to: ${pullHost}`);
|
|
123
125
|
this.workerPull.on("message", (name, args, uniqId) => __awaiter(this, void 0, void 0, function* () {
|
|
124
126
|
try {
|
|
125
127
|
const jobName = name.toString("utf8");
|
|
@@ -137,7 +139,7 @@ let JobManager = class JobManager {
|
|
|
137
139
|
console.timeEnd(uniqueId);
|
|
138
140
|
}
|
|
139
141
|
catch (e) {
|
|
140
|
-
|
|
142
|
+
this.logger.log("job-manager", `Failed to start job: ${e.message}`);
|
|
141
143
|
}
|
|
142
144
|
}));
|
|
143
145
|
});
|
|
@@ -169,7 +171,7 @@ let JobManager = class JobManager {
|
|
|
169
171
|
const port = this.config.resolve("zmqPort");
|
|
170
172
|
this.apiPush = socket("push");
|
|
171
173
|
this.apiPush.bind(`tcp://0.0.0.0:${port}`);
|
|
172
|
-
|
|
174
|
+
this.logger.log("job-manager", `API producer bound to port: ${port}`);
|
|
173
175
|
}
|
|
174
176
|
if (!this.apiPull) {
|
|
175
177
|
const backPort = this.config.resolve("zmqBackPort");
|
|
@@ -182,10 +184,10 @@ let JobManager = class JobManager {
|
|
|
182
184
|
res[key] = getType(params[key]);
|
|
183
185
|
return res;
|
|
184
186
|
}, {});
|
|
185
|
-
|
|
187
|
+
this.logger.log("job-manager", `Received a message from worker: "${colorize(message, ConsoleColor.FgCyan)}" with args: ${jsonHighlight(paramTypes)}\n\n`);
|
|
186
188
|
this.messages.next({ message, params });
|
|
187
189
|
});
|
|
188
|
-
|
|
190
|
+
this.logger.log("job-manager", `API consumer bound to port: ${backPort}`);
|
|
189
191
|
}
|
|
190
192
|
return this.tryResolve(jobType, params);
|
|
191
193
|
}
|
|
@@ -210,8 +212,10 @@ let JobManager = class JobManager {
|
|
|
210
212
|
JobManager = __decorate([
|
|
211
213
|
injectable(),
|
|
212
214
|
scoped(Lifecycle.ContainerScoped),
|
|
213
|
-
__param(
|
|
214
|
-
|
|
215
|
+
__param(2, inject(DI_CONTAINER)),
|
|
216
|
+
__param(3, injectAll(JOB)),
|
|
217
|
+
__metadata("design:paramtypes", [Configuration,
|
|
218
|
+
Logger, Object, Array])
|
|
215
219
|
], JobManager);
|
|
216
220
|
export { JobManager };
|
|
217
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"job-manager.js","sourceRoot":"","sources":["../../../src/services/job-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import {DependencyContainer, inject, injectable, injectAll, Lifecycle, scoped} from \"tsyringe\";\r\nimport {schedule, validate} from \"node-cron\";\r\nimport {socket, Socket} from \"zeromq\";\r\nimport {Subject, Subscription} from \"rxjs\";\r\nimport {filter, map} from \"rxjs/operators\";\r\nimport {ObjectId} from \"bson\";\r\nimport {\r\n    DI_CONTAINER,\r\n    IJob,\r\n    IJobTask,\r\n    IMessageBridge,\r\n    ISocketMessage,\r\n    JOB,\r\n    JobParams,\r\n    JobScheduleRange,\r\n    JobScheduleTime,\r\n    SocketParams,\r\n    Type\r\n} from \"../common-types\";\r\nimport {\r\n    colorize,\r\n    ConsoleColor,\r\n    getConstructorName,\r\n    getType,\r\n    isArray,\r\n    isObject,\r\n    jsonHighlight,\r\n    promiseTimeout\r\n} from \"../utils\";\r\nimport {Configuration} from \"./configuration\";\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class JobManager {\r\n\r\n    protected jobTypes: Type<IJob>[];\r\n    protected jobs: { [name: string]: (jobParams: JobParams, uniqueId: string) => Promise<any> };\r\n    protected messages: Subject<ISocketMessage>;\r\n    protected processing: boolean;\r\n\r\n    protected apiPush: Socket;\r\n    protected apiPull: Socket;\r\n    protected workerPush: Socket;\r\n    protected workerPull: Socket;\r\n\r\n    readonly maxTimeout: number;\r\n\r\n    constructor(readonly config: Configuration, @inject(DI_CONTAINER) readonly container: DependencyContainer, @injectAll(JOB) jobTypes: Type<IJob>[]) {\r\n        this.jobTypes = jobTypes || [];\r\n        this.jobs = this.jobTypes.reduce((res, jobType) => {\r\n            const jobName = getConstructorName(jobType);\r\n            res[jobName] = (jobParams: JobParams, uniqueId: string) => {\r\n                const job = this.resolveJobInstance(jobType, jobParams, uniqueId);\r\n                const messageBridge: IMessageBridge = {\r\n                    sendMessage: (message: string, params?: SocketParams) => {\r\n                        params.uniqueId = uniqueId;\r\n                        this.workerPush.send([message, JSON.stringify(params)]);\r\n                    }\r\n                };\r\n                messageBridge.sendMessage(`job-started`, {name: jobName});\r\n                return job.process(messageBridge);\r\n            }\r\n            return res;\r\n        }, {});\r\n        this.messages = new Subject<ISocketMessage>();\r\n        this.processing = false;\r\n        this.maxTimeout = this.config.resolve(\"jobTimeout\");\r\n    }\r\n\r\n    on(message: string, cb: (params: SocketParams) => any): Subscription {\r\n        return this.messages\r\n            .pipe(filter(t => t.message === message))\r\n            .pipe(map(t => t.params)).subscribe(cb);\r\n    }\r\n\r\n    async process(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\r\n        let instance: IJob = null;\r\n        try {\r\n            instance = this.resolveJobInstance(jobType, params);\r\n        } catch (e) {\r\n            const jobName = getConstructorName(jobType);\r\n            throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`;\r\n        }\r\n        return instance.process();\r\n    }\r\n\r\n    async enqueueWithName(name: string, params: JobParams = {}): Promise<string> {\r\n        return this.sendToWorkers(this.tryResolveFromName(name, params), params);\r\n    }\r\n\r\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<string> {\r\n        return this.sendToWorkers(this.tryResolveAndInit(jobType, params), params);\r\n    }\r\n\r\n    schedule(minute: JobScheduleTime, hour: JobScheduleTime, dayOfMonth: JobScheduleTime, month: JobScheduleTime, dayOfWeek: JobScheduleTime, jobType: Type<IJob>, params: JobParams = {}): IJobTask {\r\n        const expression = [minute, hour, dayOfMonth, month, dayOfWeek].map(t => {\r\n            if (isObject(t)) {\r\n                const range = t as JobScheduleRange;\r\n                return `${range.min || 0}-${range.max || 0}`;\r\n            }\r\n            if (isArray(t)) {\r\n                return t.join(\",\");\r\n            }\r\n            return `${t}`;\r\n        }).join(\" \");\r\n        const jobName = getConstructorName(jobType);\r\n        if (!validate(expression)) {\r\n            console.log(`Can't schedule the task: '${jobName}' because time expression is invalid.`);\r\n            return null;\r\n        }\r\n        return schedule(expression, () => {\r\n            this.enqueue(jobType, params).catch(e => {\r\n                console.log(`Can't enqueue job: '${jobName}' because: ${e}`);\r\n            });\r\n        });\r\n    }\r\n\r\n    async startProcessing(): Promise<any> {\r\n        if (this.processing) return null;\r\n        this.processing = true;\r\n\r\n        if (!this.config.resolve(\"isWorker\")) {\r\n            console.log(colorize(`Processing can not be started because this is NOT a worker process!`, ConsoleColor.FgRed));\r\n            return null;\r\n        }\r\n\r\n        const host = this.config.resolve(\"zmqRemoteHost\");\r\n        const pushHost = `${host}:${this.config.resolve(\"zmqBackPort\")}`;\r\n        this.workerPush = socket(\"push\");\r\n        await this.workerPush.connect(pushHost);\r\n        console.log(`Worker producer connected to: ${pushHost}`);\r\n\r\n        const pullHost = `${host}:${this.config.resolve(\"zmqPort\")}`;\r\n        this.workerPull = socket(\"pull\");\r\n        await this.workerPull.connect(pullHost);\r\n        console.log(`Worker consumer connected to: ${pullHost}`);\r\n\r\n        this.workerPull.on(\"message\", async (name: Buffer, args: Buffer, uniqId: Buffer) => {\r\n            try {\r\n                const jobName = name.toString(\"utf8\");\r\n                const jobParams = JSON.parse(args.toString(\"utf8\")) as JobParams;\r\n                const uniqueId = uniqId?.toString(\"utf8\");\r\n\r\n                console.time(uniqueId);\r\n                console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \\n${jsonHighlight(jobParams)}\\n\\n`);\r\n\r\n                try {\r\n                    await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);\r\n                    console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\\n\\n`);\r\n                } catch (e) {\r\n                    console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\\n${e}\\n\\n`);\r\n                }\r\n                console.timeEnd(uniqueId);\r\n            } catch (e) {\r\n                console.log(`Failed to start job: ${e.message}`);\r\n            }\r\n        });\r\n    }\r\n\r\n    tryResolve(jobType: Type<IJob>, params: JobParams): string {\r\n        const jobName = getConstructorName(jobType);\r\n        if (!this.jobs[jobName]) {\r\n            throw `Can't find job with name: ${jobName} so it can't be enqueued!`;\r\n        }\r\n        try {\r\n            this.resolveJobInstance(jobType, params);\r\n        } catch (e) {\r\n            throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`;\r\n        }\r\n        return jobName;\r\n    }\r\n\r\n    protected tryResolveFromName(jobName: string, params: JobParams): string {\r\n        const jobType = this.jobTypes.find(type => {\r\n            return getConstructorName(type) == jobName;\r\n        });\r\n        if (!jobType) {\r\n            throw `Can't find job type with name: ${jobName} so it can't be enqueued!`;\r\n        }\r\n        return this.tryResolveAndInit(jobType, params);\r\n    }\r\n\r\n    protected tryResolveAndInit(jobType: Type<IJob>, params: JobParams): string {\r\n        if (!this.apiPush) {\r\n            const port = this.config.resolve(\"zmqPort\");\r\n            this.apiPush = socket(\"push\");\r\n            this.apiPush.bind(`tcp://0.0.0.0:${port}`);\r\n            console.log(`API producer bound to port: ${port}`);\r\n        }\r\n        if (!this.apiPull) {\r\n            const backPort = this.config.resolve(\"zmqBackPort\");\r\n            this.apiPull = socket(\"pull\");\r\n            this.apiPull.bind(`tcp://0.0.0.0:${backPort}`);\r\n            this.apiPull.on(\"message\", (name: Buffer, args?: Buffer) => {\r\n                const message = name.toString(\"utf8\");\r\n                const params = JSON.parse(args?.toString(\"utf8\") || \"{}\") as JobParams;\r\n                const paramTypes = Object.keys(params).reduce((res, key) => {\r\n                    res[key] = getType(params[key]);\r\n                    return res;\r\n                }, {});\r\n                console.log(`Received a message from worker: \"${colorize(message, ConsoleColor.FgCyan)}\" with args: ${jsonHighlight(paramTypes)}\\n\\n`);\r\n                this.messages.next({message, params});\r\n            });\r\n            console.log(`API consumer bound to port: ${backPort}`);\r\n        }\r\n        return this.tryResolve(jobType, params);\r\n    }\r\n\r\n    protected resolveJobInstance(jobType: Type<IJob>, params: JobParams, uniqueId: string = \"\"): IJob {\r\n        const container = this.container.createChildContainer();\r\n        Object.keys(params).map((name) => {\r\n            container.register(name, {useValue: params[name]});\r\n        });\r\n        container.register(\"uniqueId\", {useValue: uniqueId});\r\n        container.register(jobType, jobType);\r\n        return container.resolve(jobType) as IJob;\r\n    }\r\n\r\n    protected async sendToWorkers(jobName: string, params: JobParams): Promise<string> {\r\n        const publisher = await this.apiPush;\r\n        const uniqueId = new ObjectId().toHexString();\r\n        await publisher.send([jobName, JSON.stringify(params), uniqueId]);\r\n        return uniqueId;\r\n    }\r\n}\r\n"]}
|
|
221
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"job-manager.js","sourceRoot":"","sources":["../../../src/services/job-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import {DependencyContainer, inject, injectable, injectAll, Lifecycle, scoped} from \"tsyringe\";\r\nimport {schedule, validate} from \"node-cron\";\r\nimport {socket, Socket} from \"zeromq\";\r\nimport {Subject, Subscription} from \"rxjs\";\r\nimport {filter, map} from \"rxjs/operators\";\r\nimport {ObjectId} from \"bson\";\r\nimport {\r\n    DI_CONTAINER,\r\n    IJob,\r\n    IJobTask,\r\n    IMessageBridge,\r\n    ISocketMessage,\r\n    JOB,\r\n    JobParams,\r\n    JobScheduleRange,\r\n    JobScheduleTime,\r\n    SocketParams,\r\n    Type\r\n} from \"../common-types\";\r\nimport {\r\n    colorize,\r\n    ConsoleColor,\r\n    getConstructorName,\r\n    getType,\r\n    isArray,\r\n    isObject,\r\n    jsonHighlight,\r\n    promiseTimeout\r\n} from \"../utils\";\r\nimport {Configuration} from \"./configuration\";\r\nimport {Logger} from \"./logger\";\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class JobManager {\r\n\r\n    protected jobTypes: Type<IJob>[];\r\n    protected jobs: { [name: string]: (jobParams: JobParams, uniqueId: string) => Promise<any> };\r\n    protected messages: Subject<ISocketMessage>;\r\n    protected processing: boolean;\r\n\r\n    protected apiPush: Socket;\r\n    protected apiPull: Socket;\r\n    protected workerPush: Socket;\r\n    protected workerPull: Socket;\r\n\r\n    readonly maxTimeout: number;\r\n\r\n    constructor(readonly config: Configuration,\r\n                readonly logger: Logger,\r\n                @inject(DI_CONTAINER) readonly container: DependencyContainer,\r\n                @injectAll(JOB) jobTypes: Type<IJob>[]) {\r\n        this.jobTypes = jobTypes || [];\r\n        this.jobs = this.jobTypes.reduce((res, jobType) => {\r\n            const jobName = getConstructorName(jobType);\r\n            res[jobName] = (jobParams: JobParams, uniqueId: string) => {\r\n                const job = this.resolveJobInstance(jobType, jobParams, uniqueId);\r\n                const messageBridge: IMessageBridge = {\r\n                    sendMessage: (message: string, params?: SocketParams) => {\r\n                        params.uniqueId = uniqueId;\r\n                        this.workerPush.send([message, JSON.stringify(params)]);\r\n                    }\r\n                };\r\n                messageBridge.sendMessage(`job-started`, {name: jobName});\r\n                return job.process(messageBridge);\r\n            }\r\n            return res;\r\n        }, {});\r\n        this.messages = new Subject<ISocketMessage>();\r\n        this.processing = false;\r\n        this.maxTimeout = this.config.resolve(\"jobTimeout\");\r\n    }\r\n\r\n    on(message: string, cb: (params: SocketParams) => any): Subscription {\r\n        return this.messages\r\n            .pipe(filter(t => t.message === message))\r\n            .pipe(map(t => t.params)).subscribe(cb);\r\n    }\r\n\r\n    async process(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\r\n        let instance: IJob = null;\r\n        try {\r\n            instance = this.resolveJobInstance(jobType, params);\r\n        } catch (e) {\r\n            const jobName = getConstructorName(jobType);\r\n            throw new Error(`Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`);\r\n        }\r\n        return instance.process();\r\n    }\r\n\r\n    async enqueueWithName(name: string, params: JobParams = {}): Promise<string> {\r\n        return this.sendToWorkers(this.tryResolveFromName(name, params), params);\r\n    }\r\n\r\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<string> {\r\n        return this.sendToWorkers(this.tryResolveAndInit(jobType, params), params);\r\n    }\r\n\r\n    schedule(minute: JobScheduleTime, hour: JobScheduleTime, dayOfMonth: JobScheduleTime, month: JobScheduleTime, dayOfWeek: JobScheduleTime, jobType: Type<IJob>, params: JobParams = {}): IJobTask {\r\n        const expression = [minute, hour, dayOfMonth, month, dayOfWeek].map(t => {\r\n            if (isObject(t)) {\r\n                const range = t as JobScheduleRange;\r\n                return `${range.min || 0}-${range.max || 0}`;\r\n            }\r\n            if (isArray(t)) {\r\n                return t.join(\",\");\r\n            }\r\n            return `${t}`;\r\n        }).join(\" \");\r\n        const jobName = getConstructorName(jobType);\r\n        if (!validate(expression)) {\r\n            this.logger.log(\"job-manager\", `Can't schedule the task: '${jobName}' because time expression is invalid.`);\r\n            return null;\r\n        }\r\n        return schedule(expression, () => {\r\n            this.enqueue(jobType, params).catch(e => {\r\n                this.logger.log(\"job-manager\", `Can't enqueue job: '${jobName}' because: ${e}`);\r\n            });\r\n        });\r\n    }\r\n\r\n    async startProcessing(): Promise<any> {\r\n        if (this.processing) return null;\r\n        this.processing = true;\r\n\r\n        if (!this.config.resolve(\"isWorker\")) {\r\n            this.logger.log(\"job-manager\", colorize(`Processing can not be started because this is NOT a worker process!`, ConsoleColor.FgRed));\r\n            return null;\r\n        }\r\n\r\n        const host = this.config.resolve(\"zmqRemoteHost\");\r\n        const pushHost = `${host}:${this.config.resolve(\"zmqBackPort\")}`;\r\n        this.workerPush = socket(\"push\");\r\n        await this.workerPush.connect(pushHost);\r\n        this.logger.log(\"job-manager\", `Worker producer connected to: ${pushHost}`);\r\n\r\n        const pullHost = `${host}:${this.config.resolve(\"zmqPort\")}`;\r\n        this.workerPull = socket(\"pull\");\r\n        await this.workerPull.connect(pullHost);\r\n        this.logger.log(\"job-manager\", `Worker consumer connected to: ${pullHost}`);\r\n\r\n        this.workerPull.on(\"message\", async (name: Buffer, args: Buffer, uniqId: Buffer) => {\r\n            try {\r\n                const jobName = name.toString(\"utf8\");\r\n                const jobParams = JSON.parse(args.toString(\"utf8\")) as JobParams;\r\n                const uniqueId = uniqId?.toString(\"utf8\");\r\n\r\n                console.time(uniqueId);\r\n                console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \\n${jsonHighlight(jobParams)}\\n\\n`);\r\n\r\n                try {\r\n                    await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);\r\n                    console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\\n\\n`);\r\n                } catch (e) {\r\n                    console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\\n${e}\\n\\n`);\r\n                }\r\n                console.timeEnd(uniqueId);\r\n            } catch (e) {\r\n                this.logger.log(\"job-manager\", `Failed to start job: ${e.message}`);\r\n            }\r\n        });\r\n    }\r\n\r\n    tryResolve(jobType: Type<IJob>, params: JobParams): string {\r\n        const jobName = getConstructorName(jobType);\r\n        if (!this.jobs[jobName]) {\r\n            throw `Can't find job with name: ${jobName} so it can't be enqueued!`;\r\n        }\r\n        try {\r\n            this.resolveJobInstance(jobType, params);\r\n        } catch (e) {\r\n            throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`;\r\n        }\r\n        return jobName;\r\n    }\r\n\r\n    protected tryResolveFromName(jobName: string, params: JobParams): string {\r\n        const jobType = this.jobTypes.find(type => {\r\n            return getConstructorName(type) == jobName;\r\n        });\r\n        if (!jobType) {\r\n            throw `Can't find job type with name: ${jobName} so it can't be enqueued!`;\r\n        }\r\n        return this.tryResolveAndInit(jobType, params);\r\n    }\r\n\r\n    protected tryResolveAndInit(jobType: Type<IJob>, params: JobParams): string {\r\n        if (!this.apiPush) {\r\n            const port = this.config.resolve(\"zmqPort\");\r\n            this.apiPush = socket(\"push\");\r\n            this.apiPush.bind(`tcp://0.0.0.0:${port}`);\r\n            this.logger.log(\"job-manager\", `API producer bound to port: ${port}`);\r\n        }\r\n        if (!this.apiPull) {\r\n            const backPort = this.config.resolve(\"zmqBackPort\");\r\n            this.apiPull = socket(\"pull\");\r\n            this.apiPull.bind(`tcp://0.0.0.0:${backPort}`);\r\n            this.apiPull.on(\"message\", (name: Buffer, args?: Buffer) => {\r\n                const message = name.toString(\"utf8\");\r\n                const params = JSON.parse(args?.toString(\"utf8\") || \"{}\") as JobParams;\r\n                const paramTypes = Object.keys(params).reduce((res, key) => {\r\n                    res[key] = getType(params[key]);\r\n                    return res;\r\n                }, {});\r\n                this.logger.log(\"job-manager\", `Received a message from worker: \"${colorize(message, ConsoleColor.FgCyan)}\" with args: ${jsonHighlight(paramTypes)}\\n\\n`);\r\n                this.messages.next({message, params});\r\n            });\r\n            this.logger.log(\"job-manager\", `API consumer bound to port: ${backPort}`);\r\n        }\r\n        return this.tryResolve(jobType, params);\r\n    }\r\n\r\n    protected resolveJobInstance(jobType: Type<IJob>, params: JobParams, uniqueId: string = \"\"): IJob {\r\n        const container = this.container.createChildContainer();\r\n        Object.keys(params).map((name) => {\r\n            container.register(name, {useValue: params[name]});\r\n        });\r\n        container.register(\"uniqueId\", {useValue: uniqueId});\r\n        container.register(jobType, jobType);\r\n        return container.resolve(jobType) as IJob;\r\n    }\r\n\r\n    protected async sendToWorkers(jobName: string, params: JobParams): Promise<string> {\r\n        const publisher = await this.apiPush;\r\n        const uniqueId = new ObjectId().toHexString();\r\n        await publisher.send([jobName, JSON.stringify(params), uniqueId]);\r\n        return uniqueId;\r\n    }\r\n}\r\n"]}
|
|
@@ -22,12 +22,14 @@ import { MongoConnector } from "./mongo-connector";
|
|
|
22
22
|
import { Assets } from "./assets";
|
|
23
23
|
import { LazyAsset } from "./entities/lazy-asset";
|
|
24
24
|
import { JobManager } from "./job-manager";
|
|
25
|
+
import { Logger } from "./logger";
|
|
25
26
|
import { Progresses } from "./progresses";
|
|
26
27
|
let LazyAssets = class LazyAssets {
|
|
27
|
-
constructor(connector, assets, progresses, jobMan) {
|
|
28
|
+
constructor(connector, assets, progresses, logger, jobMan) {
|
|
28
29
|
this.connector = connector;
|
|
29
30
|
this.assets = assets;
|
|
30
31
|
this.progresses = progresses;
|
|
32
|
+
this.logger = logger;
|
|
31
33
|
this.jobMan = jobMan;
|
|
32
34
|
this.collection = connector.database.collection("lazyassets");
|
|
33
35
|
}
|
|
@@ -43,7 +45,7 @@ let LazyAssets = class LazyAssets {
|
|
|
43
45
|
if (existingAsset)
|
|
44
46
|
return existingAsset;
|
|
45
47
|
const res = yield this.collection.insertOne(data);
|
|
46
|
-
return new LazyAsset(res.insertedId, data, this.collection, this.assets, this.progresses);
|
|
48
|
+
return new LazyAsset(res.insertedId, data, this.collection, this.logger, this.assets, this.progresses);
|
|
47
49
|
});
|
|
48
50
|
}
|
|
49
51
|
read(id) {
|
|
@@ -56,7 +58,7 @@ let LazyAssets = class LazyAssets {
|
|
|
56
58
|
const data = yield this.collection.findOne(where);
|
|
57
59
|
return !data
|
|
58
60
|
? null
|
|
59
|
-
: new LazyAsset(data._id, data, this.collection, this.assets, this.progresses);
|
|
61
|
+
: new LazyAsset(data._id, data, this.collection, this.logger, this.assets, this.progresses);
|
|
60
62
|
});
|
|
61
63
|
}
|
|
62
64
|
unlink(id) {
|
|
@@ -74,7 +76,8 @@ LazyAssets = __decorate([
|
|
|
74
76
|
__metadata("design:paramtypes", [MongoConnector,
|
|
75
77
|
Assets,
|
|
76
78
|
Progresses,
|
|
79
|
+
Logger,
|
|
77
80
|
JobManager])
|
|
78
81
|
], LazyAssets);
|
|
79
82
|
export { LazyAssets };
|
|
80
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGF6eS1hc3NldHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2VydmljZXMvbGF6eS1hc3NldHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsT0FBTyxFQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFDLE1BQU0sVUFBVSxDQUFDO0FBQ3ZELE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFJOUIsT0FBTyxFQUFDLGNBQWMsRUFBQyxNQUFNLG1CQUFtQixDQUFDO0FBQ2pELE9BQU8sRUFBQyxNQUFNLEVBQUMsTUFBTSxVQUFVLENBQUM7QUFDaEMsT0FBTyxFQUFDLFNBQVMsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ2hELE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFDLE1BQU0sRUFBQyxNQUFNLFVBQVUsQ0FBQztBQUNoQyxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0sY0FBYyxDQUFDO0lBSTNCLFVBQVUsU0FBVixVQUFVO0lBSW5CLFlBQXFCLFNBQXlCLEVBQ3pCLE1BQWMsRUFDZCxVQUFzQixFQUN0QixNQUFjLEVBQ2QsTUFBa0I7UUFKbEIsY0FBUyxHQUFULFNBQVMsQ0FBZ0I7UUFDekIsV0FBTSxHQUFOLE1BQU0sQ0FBUTtRQUNkLGVBQVUsR0FBVixVQUFVLENBQVk7UUFDdEIsV0FBTSxHQUFOLE1BQU0sQ0FBUTtRQUNkLFdBQU0sR0FBTixNQUFNLENBQVk7UUFDbkMsSUFBSSxDQUFDLFVBQVUsR0FBRyxTQUFTLENBQUMsUUFBUSxDQUFDLFVBQVUsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUNsRSxDQUFDO0lBRUssTUFBTSxDQUFDLE9BQW1CLEVBQUUsWUFBdUIsRUFBRSxFQUFFLFNBQWlCLE1BQU07O1lBQ2hGLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLE9BQU8sa0NBQU0sU0FBUyxLQUFFLE1BQU0sRUFBRSxFQUFFLElBQUUsQ0FBQztZQUM1RSxNQUFNLElBQUksR0FBRztnQkFDVCxPQUFPO2dCQUNQLFNBQVM7Z0JBQ1QsTUFBTTthQUNULENBQUM7WUFDRixNQUFNLGFBQWEsR0FBRyxNQUFNLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDNUMsSUFBSSxhQUFhO2dCQUFFLE9BQU8sYUFBYSxDQUFDO1lBQ3hDLE1BQU0sR0FBRyxHQUFHLE1BQU0sSUFBSSxDQUFDLFVBQVUsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDbEQsT0FBTyxJQUFJLFNBQVMsQ0FBQyxHQUFHLENBQUMsVUFBVSxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsVUFBVSxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDM0csQ0FBQztLQUFBO0lBRUssSUFBSSxDQUFDLEVBQVU7O1lBQ2pCLE9BQU8sQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFDLEdBQUcsRUFBRSxJQUFJLFFBQVEsQ0FBQyxFQUFFLENBQUMsRUFBQyxDQUFDLENBQUM7UUFDM0QsQ0FBQztLQUFBO0lBRUssSUFBSSxDQUFDLEtBQThCOztZQUNyQyxNQUFNLElBQUksR0FBRyxNQUFNLElBQUksQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQ2xELE9BQU8sQ0FBQyxJQUFJO2dCQUNSLENBQUMsQ0FBQyxJQUFJO2dCQUNOLENBQUMsQ0FBQyxJQUFJLFNBQVMsQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsVUFBVSxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDcEcsQ0FBQztLQUFBO0lBRUssTUFBTSxDQUFDLEVBQVU7O1lBQ25CLE1BQU0sS0FBSyxHQUFHLE1BQU0sSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUNsQyxJQUFJLENBQUMsS0FBSztnQkFBRSxPQUFPLElBQUksQ0FBQztZQUN4QixPQUFPLEtBQUssQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUMxQixDQUFDO0tBQUE7Q0FDSixDQUFBO0FBekNZLFVBQVU7SUFGdEIsVUFBVSxFQUFFO0lBQ1osTUFBTSxDQUFDLFNBQVMsQ0FBQyxlQUFlLENBQUM7cUNBS0UsY0FBYztRQUNqQixNQUFNO1FBQ0YsVUFBVTtRQUNkLE1BQU07UUFDTixVQUFVO0dBUjlCLFVBQVUsQ0F5Q3RCO1NBekNZLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge2luamVjdGFibGUsIExpZmVjeWNsZSwgc2NvcGVkfSBmcm9tIFwidHN5cmluZ2VcIjtcclxuaW1wb3J0IHtPYmplY3RJZH0gZnJvbSBcImJzb25cIjtcclxuaW1wb3J0IHtDb2xsZWN0aW9ufSBmcm9tIFwibW9uZ29kYlwiO1xyXG5pbXBvcnQge0ZpbHRlclF1ZXJ5fSBmcm9tIFwibW9uZ29vc2VcIjtcclxuaW1wb3J0IHtJSm9iLCBJTGF6eUFzc2V0LCBKb2JQYXJhbXMsIFR5cGV9IGZyb20gXCIuLi9jb21tb24tdHlwZXNcIjtcclxuaW1wb3J0IHtNb25nb0Nvbm5lY3Rvcn0gZnJvbSBcIi4vbW9uZ28tY29ubmVjdG9yXCI7XHJcbmltcG9ydCB7QXNzZXRzfSBmcm9tIFwiLi9hc3NldHNcIjtcclxuaW1wb3J0IHtMYXp5QXNzZXR9IGZyb20gXCIuL2VudGl0aWVzL2xhenktYXNzZXRcIjtcclxuaW1wb3J0IHtKb2JNYW5hZ2VyfSBmcm9tIFwiLi9qb2ItbWFuYWdlclwiO1xyXG5pbXBvcnQge0xvZ2dlcn0gZnJvbSBcIi4vbG9nZ2VyXCI7XHJcbmltcG9ydCB7UHJvZ3Jlc3Nlc30gZnJvbSBcIi4vcHJvZ3Jlc3Nlc1wiO1xyXG5cclxuQGluamVjdGFibGUoKVxyXG5Ac2NvcGVkKExpZmVjeWNsZS5Db250YWluZXJTY29wZWQpXHJcbmV4cG9ydCBjbGFzcyBMYXp5QXNzZXRzIHtcclxuXHJcbiAgICBwcm90ZWN0ZWQgY29sbGVjdGlvbjogQ29sbGVjdGlvbjxQYXJ0aWFsPElMYXp5QXNzZXQ+PjtcclxuXHJcbiAgICBjb25zdHJ1Y3RvcihyZWFkb25seSBjb25uZWN0b3I6IE1vbmdvQ29ubmVjdG9yLFxyXG4gICAgICAgICAgICAgICAgcmVhZG9ubHkgYXNzZXRzOiBBc3NldHMsXHJcbiAgICAgICAgICAgICAgICByZWFkb25seSBwcm9ncmVzc2VzOiBQcm9ncmVzc2VzLFxyXG4gICAgICAgICAgICAgICAgcmVhZG9ubHkgbG9nZ2VyOiBMb2dnZXIsXHJcbiAgICAgICAgICAgICAgICByZWFkb25seSBqb2JNYW46IEpvYk1hbmFnZXIpIHtcclxuICAgICAgICB0aGlzLmNvbGxlY3Rpb24gPSBjb25uZWN0b3IuZGF0YWJhc2UuY29sbGVjdGlvbihcImxhenlhc3NldHNcIik7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgY3JlYXRlKGpvYlR5cGU6IFR5cGU8SUpvYj4sIGpvYlBhcmFtczogSm9iUGFyYW1zID0ge30sIGpvYlF1ZTogc3RyaW5nID0gXCJtYWluXCIpOiBQcm9taXNlPElMYXp5QXNzZXQ+IHtcclxuICAgICAgICBjb25zdCBqb2JOYW1lID0gdGhpcy5qb2JNYW4udHJ5UmVzb2x2ZShqb2JUeXBlLCB7Li4uam9iUGFyYW1zLCBsYXp5SWQ6IFwiXCJ9KTtcclxuICAgICAgICBjb25zdCBkYXRhID0ge1xyXG4gICAgICAgICAgICBqb2JOYW1lLFxyXG4gICAgICAgICAgICBqb2JQYXJhbXMsXHJcbiAgICAgICAgICAgIGpvYlF1ZVxyXG4gICAgICAgIH07XHJcbiAgICAgICAgY29uc3QgZXhpc3RpbmdBc3NldCA9IGF3YWl0IHRoaXMuZmluZChkYXRhKTtcclxuICAgICAgICBpZiAoZXhpc3RpbmdBc3NldCkgcmV0dXJuIGV4aXN0aW5nQXNzZXQ7XHJcbiAgICAgICAgY29uc3QgcmVzID0gYXdhaXQgdGhpcy5jb2xsZWN0aW9uLmluc2VydE9uZShkYXRhKTtcclxuICAgICAgICByZXR1cm4gbmV3IExhenlBc3NldChyZXMuaW5zZXJ0ZWRJZCwgZGF0YSwgdGhpcy5jb2xsZWN0aW9uLCB0aGlzLmxvZ2dlciwgdGhpcy5hc3NldHMsIHRoaXMucHJvZ3Jlc3Nlcyk7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgcmVhZChpZDogc3RyaW5nKTogUHJvbWlzZTxJTGF6eUFzc2V0PiB7XHJcbiAgICAgICAgcmV0dXJuICFpZCA/IG51bGwgOiB0aGlzLmZpbmQoe19pZDogbmV3IE9iamVjdElkKGlkKX0pO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIGZpbmQod2hlcmU6IEZpbHRlclF1ZXJ5PElMYXp5QXNzZXQ+KTogUHJvbWlzZTxJTGF6eUFzc2V0PiB7XHJcbiAgICAgICAgY29uc3QgZGF0YSA9IGF3YWl0IHRoaXMuY29sbGVjdGlvbi5maW5kT25lKHdoZXJlKTtcclxuICAgICAgICByZXR1cm4gIWRhdGFcclxuICAgICAgICAgICAgPyBudWxsXHJcbiAgICAgICAgICAgIDogbmV3IExhenlBc3NldChkYXRhLl9pZCwgZGF0YSwgdGhpcy5jb2xsZWN0aW9uLCB0aGlzLmxvZ2dlciwgdGhpcy5hc3NldHMsIHRoaXMucHJvZ3Jlc3Nlcyk7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgdW5saW5rKGlkOiBzdHJpbmcpOiBQcm9taXNlPGFueT4ge1xyXG4gICAgICAgIGNvbnN0IGFzc2V0ID0gYXdhaXQgdGhpcy5yZWFkKGlkKTtcclxuICAgICAgICBpZiAoIWFzc2V0KSByZXR1cm4gbnVsbDtcclxuICAgICAgICByZXR1cm4gYXNzZXQudW5saW5rKCk7XHJcbiAgICB9XHJcbn1cclxuIl19
|
|
@@ -0,0 +1,29 @@
|
|
|
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 { singleton } from "tsyringe";
|
|
11
|
+
import { Configuration } from "./configuration";
|
|
12
|
+
let Logger = class Logger {
|
|
13
|
+
constructor(config) {
|
|
14
|
+
this.config = config;
|
|
15
|
+
console.log("Logger created");
|
|
16
|
+
this.tags = this.config.resolve("logTags");
|
|
17
|
+
}
|
|
18
|
+
log(tag, ...params) {
|
|
19
|
+
if (!this.tags.includes(tag)) {
|
|
20
|
+
console.log(`[${tag}]`, ...params);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
Logger = __decorate([
|
|
25
|
+
singleton(),
|
|
26
|
+
__metadata("design:paramtypes", [Configuration])
|
|
27
|
+
], Logger);
|
|
28
|
+
export { Logger };
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3NlcnZpY2VzL2xvZ2dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFBQSxPQUFPLEVBQUMsU0FBUyxFQUFDLE1BQU0sVUFBVSxDQUFDO0FBQ25DLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQztJQUdqQyxNQUFNLFNBQU4sTUFBTTtJQUlmLFlBQXFCLE1BQXFCO1FBQXJCLFdBQU0sR0FBTixNQUFNLENBQWU7UUFDdEMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO1FBQzlCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDL0MsQ0FBQztJQUVELEdBQUcsQ0FBQyxHQUFXLEVBQUUsR0FBRyxNQUFhO1FBQzdCLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsRUFBRTtZQUMxQixPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksR0FBRyxHQUFHLEVBQUUsR0FBRyxNQUFNLENBQUMsQ0FBQztTQUN0QztJQUNMLENBQUM7Q0FDSixDQUFBO0FBZFksTUFBTTtJQURsQixTQUFTLEVBQUU7cUNBS3FCLGFBQWE7R0FKakMsTUFBTSxDQWNsQjtTQWRZLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge3NpbmdsZXRvbn0gZnJvbSBcInRzeXJpbmdlXCI7XHJcbmltcG9ydCB7Q29uZmlndXJhdGlvbn0gZnJvbSBcIi4vY29uZmlndXJhdGlvblwiO1xyXG5cclxuQHNpbmdsZXRvbigpXHJcbmV4cG9ydCBjbGFzcyBMb2dnZXIge1xyXG5cclxuICAgIHByb3RlY3RlZCB0YWdzOiBzdHJpbmdbXTtcclxuXHJcbiAgICBjb25zdHJ1Y3RvcihyZWFkb25seSBjb25maWc6IENvbmZpZ3VyYXRpb24pIHtcclxuICAgICAgICBjb25zb2xlLmxvZyhcIkxvZ2dlciBjcmVhdGVkXCIpO1xyXG4gICAgICAgIHRoaXMudGFncyA9IHRoaXMuY29uZmlnLnJlc29sdmUoXCJsb2dUYWdzXCIpO1xyXG4gICAgfVxyXG5cclxuICAgIGxvZyh0YWc6IHN0cmluZywgLi4ucGFyYW1zOiBhbnlbXSk6IHZvaWQge1xyXG4gICAgICAgIGlmICghdGhpcy50YWdzLmluY2x1ZGVzKHRhZykpIHtcclxuICAgICAgICAgICAgY29uc29sZS5sb2coYFske3RhZ31dYCwgLi4ucGFyYW1zKTtcclxuICAgICAgICB9XHJcbiAgICB9XHJcbn1cclxuIl19
|
package/esm2015/stemy-backend.js
CHANGED
|
@@ -2,4 +2,5 @@
|
|
|
2
2
|
* Generated bundle index. Do not edit.
|
|
3
3
|
*/
|
|
4
4
|
export * from './public_api';
|
|
5
|
-
|
|
5
|
+
export { Logger as ɵa } from './services/logger';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlbXktYmFja2VuZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdGVteS1iYWNrZW5kLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUM7QUFFN0IsT0FBTyxFQUFDLE1BQU0sSUFBSSxFQUFFLEVBQUMsTUFBTSxtQkFBbUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWNfYXBpJztcblxuZXhwb3J0IHtMb2dnZXIgYXMgybVhfSBmcm9tICcuL3NlcnZpY2VzL2xvZ2dlcic7Il19
|
|
@@ -15,7 +15,6 @@ export class BaseDoc {
|
|
|
15
15
|
return getModelForClass(type);
|
|
16
16
|
}
|
|
17
17
|
}
|
|
18
|
-
// @ts-ignore
|
|
19
18
|
export const PrimitiveArray = Types.Array;
|
|
20
19
|
export const DocumentArray = Types.DocumentArray;
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1kb2MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdXRpbGl0aWVzL2Jhc2UtZG9jLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBVSxLQUFLLEVBQUMsTUFBTSxVQUFVLENBQUM7QUFDeEMsT0FBTyxFQUFlLGdCQUFnQixFQUFrQixNQUFNLHNCQUFzQixDQUFDO0FBR3JGLE1BQU0sT0FBZ0IsT0FBTztJQTZCekI7O09BRUc7SUFDSCxJQUFJO1FBQ0EsT0FBTyxJQUFXLENBQUM7SUFDdkIsQ0FBQztJQUVEOzs7T0FHRztJQUNILEtBQUssQ0FBNkIsSUFBTztRQUNyQyxPQUFPLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ2xDLENBQUM7Q0FDSjtBQUlELE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDO0FBSTFDLE1BQU0sQ0FBQyxNQUFNLGFBQWEsR0FBRyxLQUFLLENBQUMsYUFBYSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtSZWZUeXBlLCBUeXBlc30gZnJvbSBcIm1vbmdvb3NlXCI7XHJcbmltcG9ydCB7RG9jdW1lbnRUeXBlLCBnZXRNb2RlbEZvckNsYXNzLCBSZXR1cm5Nb2RlbFR5cGV9IGZyb20gXCJAdHlwZWdvb3NlL3R5cGVnb29zZVwiO1xyXG5pbXBvcnQge0NvbnN0cnVjdG9yfSBmcm9tIFwiLi4vY29tbW9uLXR5cGVzXCI7XHJcblxyXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZURvYzxJRFR5cGUgZXh0ZW5kcyBSZWZUeXBlID0gVHlwZXMuT2JqZWN0SWQ+IHtcclxuXHJcbiAgICBfaWQ/OiBJRFR5cGU7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBUaGlzIGdldHRlci9zZXR0ZXIgZG9lc24ndCBleGlzdCBpZiBcInNjaGVtYU9wdGlvbnMuaWRcIiBiZWluZyBzZXQgdG8gXCJmYWxzZVwiXHJcbiAgICAgKi9cclxuICAgIGlkOiBzdHJpbmc7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBUaGlzIGdldHRlciBkb2Vzbid0IGV4aXN0IGlmIFwic2NoZW1hT3B0aW9ucy50aW1lc3RhbXBzXCIgYmVpbmcgc2V0IHRvIFwiZmFsc2VcIlxyXG4gICAgICovXHJcbiAgICBjcmVhdGVkQXQ/OiBEYXRlO1xyXG5cclxuICAgIC8qKlxyXG4gICAgICogVGhpcyBnZXR0ZXIgZG9lc24ndCBleGlzdCBpZiBcInNjaGVtYU9wdGlvbnMudGltZXN0YW1wc1wiIGJlaW5nIHNldCB0byBcImZhbHNlXCJcclxuICAgICAqL1xyXG4gICAgdXBkYXRlZEF0PzogRGF0ZTtcclxuXHJcbiAgICAvKipcclxuICAgICAqIFJldHVybnMgdGhlIERvY3VtZW50IGFzIEpTT05cclxuICAgICAqL1xyXG4gICAgdG9KU09OPygpOiBhbnk7XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBSZXR1cm5zIHRoZSBEb2N1bWVudCBhcyBhbiBPYmplY3RcclxuICAgICAqL1xyXG4gICAgdG9PYmplY3Q/KCk6IGFueTtcclxuXHJcbiAgICAvKipcclxuICAgICAqIENhc3RzIHRoaXMgdG8gRG9jdW1lbnRUeXBlPHRoaXM+IHRvIGFsbG93IHVzaW5nIGRvY3VtZW50IG1ldGhvZHMgaW4gZ2V0L3NldC1zXHJcbiAgICAgKi9cclxuICAgIGNhc3QoKTogRG9jdW1lbnRUeXBlPHRoaXM+IHtcclxuICAgICAgICByZXR1cm4gdGhpcyBhcyBhbnk7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBHZXRzIGEgcHJlLWNvbXBpbGVkIG1vZGVsIGZyb20gdHlwZWdvb3NlIGNhY2hlIGJ5IGl0cyBjbGFzcyB0eXBlXHJcbiAgICAgKiBAcGFyYW0gdHlwZVxyXG4gICAgICovXHJcbiAgICBtb2RlbDxUIGV4dGVuZHMgQ29uc3RydWN0b3I8YW55Pj4odHlwZTogVCk6IFJldHVybk1vZGVsVHlwZTxUPiB7XHJcbiAgICAgICAgcmV0dXJuIGdldE1vZGVsRm9yQ2xhc3ModHlwZSk7XHJcbiAgICB9XHJcbn1cclxuXHJcbmV4cG9ydCB0eXBlIFByaW1pdGl2ZUFycmF5PFQ+ID0gVHlwZXMuQXJyYXk8VD47XHJcblxyXG5leHBvcnQgY29uc3QgUHJpbWl0aXZlQXJyYXkgPSBUeXBlcy5BcnJheTtcclxuXHJcbmV4cG9ydCB0eXBlIERvY3VtZW50QXJyYXk8VCBleHRlbmRzIEJhc2VEb2M+ID0gVHlwZXMuRG9jdW1lbnRBcnJheTxEb2N1bWVudFR5cGU8VD4+O1xyXG5cclxuZXhwb3J0IGNvbnN0IERvY3VtZW50QXJyYXkgPSBUeXBlcy5Eb2N1bWVudEFycmF5O1xyXG4iXX0=
|