@stemy/backend 3.1.3 → 3.1.6
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 +32 -23
- package/bundles/stemy-backend.umd.js.map +1 -1
- package/bundles/stemy-backend.umd.min.js +1 -1
- package/bundles/stemy-backend.umd.min.js.map +1 -1
- package/esm2015/services/job-manager.js +31 -22
- package/fesm2015/stemy-backend.js +29 -20
- package/fesm2015/stemy-backend.js.map +1 -1
- package/package.json +1 -1
- package/services/job-manager.d.ts +6 -7
- package/stemy-backend.metadata.json +1 -1
|
@@ -26,7 +26,7 @@ import { Subject } from "rxjs";
|
|
|
26
26
|
import { filter, map } from "rxjs/operators";
|
|
27
27
|
import { ObjectId } from "bson";
|
|
28
28
|
import { DI_CONTAINER, JOB } from "../common-types";
|
|
29
|
-
import { colorize, ConsoleColor, getConstructorName, isArray, isObject, jsonHighlight, promiseTimeout } from "../utils";
|
|
29
|
+
import { colorize, ConsoleColor, getConstructorName, getType, isArray, isObject, jsonHighlight, promiseTimeout } from "../utils";
|
|
30
30
|
import { Configuration } from "./configuration";
|
|
31
31
|
let JobManager = class JobManager {
|
|
32
32
|
constructor(config, container, jobTypes) {
|
|
@@ -34,18 +34,21 @@ let JobManager = class JobManager {
|
|
|
34
34
|
this.container = container;
|
|
35
35
|
this.jobTypes = jobTypes || [];
|
|
36
36
|
this.jobs = this.jobTypes.reduce((res, jobType) => {
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
37
|
+
const jobName = getConstructorName(jobType);
|
|
38
|
+
res[jobName] = (jobParams, uniqueId) => {
|
|
39
|
+
const job = this.resolveJobInstance(jobType, jobParams, uniqueId);
|
|
40
|
+
const messageBridge = {
|
|
41
|
+
sendMessage: (message, params) => {
|
|
42
|
+
params.uniqueId = uniqueId;
|
|
43
|
+
this.workerPush.send([message, JSON.stringify(params)]);
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
47
|
+
return job.process(messageBridge);
|
|
40
48
|
};
|
|
41
49
|
return res;
|
|
42
50
|
}, {});
|
|
43
51
|
this.messages = new Subject();
|
|
44
|
-
this.messageBridge = {
|
|
45
|
-
sendMessage: (message, params) => {
|
|
46
|
-
this.workerPush.send([message, JSON.stringify(params)]);
|
|
47
|
-
}
|
|
48
|
-
};
|
|
49
52
|
this.processing = false;
|
|
50
53
|
this.maxTimeout = this.config.resolve("jobTimeout");
|
|
51
54
|
}
|
|
@@ -117,22 +120,21 @@ let JobManager = class JobManager {
|
|
|
117
120
|
this.workerPull = socket("pull");
|
|
118
121
|
yield this.workerPull.connect(pullHost);
|
|
119
122
|
console.log(`Worker consumer connected to: ${pullHost}`);
|
|
120
|
-
this.workerPull.on("message", (name, args,
|
|
123
|
+
this.workerPull.on("message", (name, args, uniqId) => __awaiter(this, void 0, void 0, function* () {
|
|
121
124
|
try {
|
|
122
125
|
const jobName = name.toString("utf8");
|
|
123
126
|
const jobParams = JSON.parse(args.toString("utf8"));
|
|
124
|
-
const
|
|
125
|
-
console.time(
|
|
126
|
-
console.timeLog(
|
|
127
|
-
this.messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
127
|
+
const uniqueId = uniqId === null || uniqId === void 0 ? void 0 : uniqId.toString("utf8");
|
|
128
|
+
console.time(uniqueId);
|
|
129
|
+
console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \n${jsonHighlight(jobParams)}\n\n`);
|
|
128
130
|
try {
|
|
129
|
-
yield Promise.race([this.jobs[jobName](jobParams), promiseTimeout(this.maxTimeout, true)]);
|
|
130
|
-
console.timeLog(
|
|
131
|
+
yield Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);
|
|
132
|
+
console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\n\n`);
|
|
131
133
|
}
|
|
132
134
|
catch (e) {
|
|
133
|
-
console.timeLog(
|
|
135
|
+
console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\n${e}\n\n`);
|
|
134
136
|
}
|
|
135
|
-
console.timeEnd(
|
|
137
|
+
console.timeEnd(uniqueId);
|
|
136
138
|
}
|
|
137
139
|
catch (e) {
|
|
138
140
|
console.log(`Failed to start job: ${e.message}`);
|
|
@@ -176,25 +178,32 @@ let JobManager = class JobManager {
|
|
|
176
178
|
this.apiPull.on("message", (name, args) => {
|
|
177
179
|
const message = name.toString("utf8");
|
|
178
180
|
const params = JSON.parse((args === null || args === void 0 ? void 0 : args.toString("utf8")) || "{}");
|
|
179
|
-
|
|
181
|
+
const paramTypes = Object.keys(params).reduce((res, key) => {
|
|
182
|
+
res[key] = getType(params[key]);
|
|
183
|
+
return res;
|
|
184
|
+
}, {});
|
|
185
|
+
console.log(`Received a message from worker: "${colorize(message, ConsoleColor.FgCyan)}" with args: ${jsonHighlight(paramTypes)}\n\n`);
|
|
180
186
|
this.messages.next({ message, params });
|
|
181
187
|
});
|
|
182
188
|
console.log(`API consumer bound to port: ${backPort}`);
|
|
183
189
|
}
|
|
184
190
|
return this.tryResolve(jobType, params);
|
|
185
191
|
}
|
|
186
|
-
resolveJobInstance(jobType, params) {
|
|
192
|
+
resolveJobInstance(jobType, params, uniqueId = "") {
|
|
187
193
|
const container = this.container.createChildContainer();
|
|
188
194
|
Object.keys(params).map((name) => {
|
|
189
195
|
container.register(name, { useValue: params[name] });
|
|
190
196
|
});
|
|
197
|
+
container.register("uniqueId", { useValue: uniqueId });
|
|
191
198
|
container.register(jobType, jobType);
|
|
192
199
|
return container.resolve(jobType);
|
|
193
200
|
}
|
|
194
201
|
sendToWorkers(jobName, params) {
|
|
195
202
|
return __awaiter(this, void 0, void 0, function* () {
|
|
196
203
|
const publisher = yield this.apiPush;
|
|
197
|
-
|
|
204
|
+
const uniqueId = new ObjectId().toHexString();
|
|
205
|
+
yield publisher.send([jobName, JSON.stringify(params), uniqueId]);
|
|
206
|
+
return uniqueId;
|
|
198
207
|
});
|
|
199
208
|
}
|
|
200
209
|
};
|
|
@@ -205,4 +214,4 @@ JobManager = __decorate([
|
|
|
205
214
|
__metadata("design:paramtypes", [Configuration, Object, Array])
|
|
206
215
|
], JobManager);
|
|
207
216
|
export { JobManager };
|
|
208
|
-
//# 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 {colorize, ConsoleColor, getConstructorName, isArray, isObject, jsonHighlight, promiseTimeout} 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) => Promise<any>};\r\n    protected messages: Subject<ISocketMessage>;\r\n    protected messageBridge: IMessageBridge;\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            res[getConstructorName(jobType)] = (jobParams: JobParams) => {\r\n                const job = this.resolveJobInstance(jobType, jobParams);\r\n                return job.process(this.messageBridge);\r\n            }\r\n            return res;\r\n        }, {});\r\n        this.messages = new Subject<ISocketMessage>();\r\n        this.messageBridge = {\r\n            sendMessage: (message: string, params?: SocketParams) => {\r\n                this.workerPush.send([message, JSON.stringify(params)]);\r\n            }\r\n        };\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<any> {\r\n        return this.sendToWorkers(this.tryResolveFromName(name, params), params);\r\n    }\r\n\r\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\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, uniqueId: 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 timerId = uniqueId?.toString(\"utf8\");\r\n\r\n                console.time(timerId);\r\n                console.timeLog(timerId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \\n${jsonHighlight(jobParams)}\\n\\n`);\r\n\r\n                this.messageBridge.sendMessage(`job-started`, {name: jobName});\r\n                try {\r\n                    await Promise.race([this.jobs[jobName](jobParams), promiseTimeout(this.maxTimeout, true)]);\r\n                    console.timeLog(timerId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\\n\\n`);\r\n                } catch (e) {\r\n                    console.timeLog(timerId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\\n${e}\\n\\n`);\r\n                }\r\n                console.timeEnd(timerId);\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                console.log(`Received a message from worker: \"${colorize(message, ConsoleColor.FgCyan)}\" with args: ${jsonHighlight(params)}\\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): 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(jobType, jobType);\r\n        return container.resolve(jobType) as IJob;\r\n    }\r\n\r\n    protected async sendToWorkers(jobName: string, params: JobParams): Promise<any> {\r\n        const publisher = await this.apiPush;\r\n        await publisher.send([jobName, JSON.stringify(params), new ObjectId().toHexString()]);\r\n    }\r\n}\r\n"]}
|
|
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"]}
|
|
@@ -1581,18 +1581,21 @@ let JobManager = class JobManager {
|
|
|
1581
1581
|
this.container = container;
|
|
1582
1582
|
this.jobTypes = jobTypes || [];
|
|
1583
1583
|
this.jobs = this.jobTypes.reduce((res, jobType) => {
|
|
1584
|
-
|
|
1585
|
-
|
|
1586
|
-
|
|
1584
|
+
const jobName = getConstructorName(jobType);
|
|
1585
|
+
res[jobName] = (jobParams, uniqueId) => {
|
|
1586
|
+
const job = this.resolveJobInstance(jobType, jobParams, uniqueId);
|
|
1587
|
+
const messageBridge = {
|
|
1588
|
+
sendMessage: (message, params) => {
|
|
1589
|
+
params.uniqueId = uniqueId;
|
|
1590
|
+
this.workerPush.send([message, JSON.stringify(params)]);
|
|
1591
|
+
}
|
|
1592
|
+
};
|
|
1593
|
+
messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
1594
|
+
return job.process(messageBridge);
|
|
1587
1595
|
};
|
|
1588
1596
|
return res;
|
|
1589
1597
|
}, {});
|
|
1590
1598
|
this.messages = new Subject();
|
|
1591
|
-
this.messageBridge = {
|
|
1592
|
-
sendMessage: (message, params) => {
|
|
1593
|
-
this.workerPush.send([message, JSON.stringify(params)]);
|
|
1594
|
-
}
|
|
1595
|
-
};
|
|
1596
1599
|
this.processing = false;
|
|
1597
1600
|
this.maxTimeout = this.config.resolve("jobTimeout");
|
|
1598
1601
|
}
|
|
@@ -1664,22 +1667,21 @@ let JobManager = class JobManager {
|
|
|
1664
1667
|
this.workerPull = socket("pull");
|
|
1665
1668
|
yield this.workerPull.connect(pullHost);
|
|
1666
1669
|
console.log(`Worker consumer connected to: ${pullHost}`);
|
|
1667
|
-
this.workerPull.on("message", (name, args,
|
|
1670
|
+
this.workerPull.on("message", (name, args, uniqId) => __awaiter$o(this, void 0, void 0, function* () {
|
|
1668
1671
|
try {
|
|
1669
1672
|
const jobName = name.toString("utf8");
|
|
1670
1673
|
const jobParams = JSON.parse(args.toString("utf8"));
|
|
1671
|
-
const
|
|
1672
|
-
console.time(
|
|
1673
|
-
console.timeLog(
|
|
1674
|
-
this.messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
1674
|
+
const uniqueId = uniqId === null || uniqId === void 0 ? void 0 : uniqId.toString("utf8");
|
|
1675
|
+
console.time(uniqueId);
|
|
1676
|
+
console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \n${jsonHighlight(jobParams)}\n\n`);
|
|
1675
1677
|
try {
|
|
1676
|
-
yield Promise.race([this.jobs[jobName](jobParams), promiseTimeout(this.maxTimeout, true)]);
|
|
1677
|
-
console.timeLog(
|
|
1678
|
+
yield Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);
|
|
1679
|
+
console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\n\n`);
|
|
1678
1680
|
}
|
|
1679
1681
|
catch (e) {
|
|
1680
|
-
console.timeLog(
|
|
1682
|
+
console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\n${e}\n\n`);
|
|
1681
1683
|
}
|
|
1682
|
-
console.timeEnd(
|
|
1684
|
+
console.timeEnd(uniqueId);
|
|
1683
1685
|
}
|
|
1684
1686
|
catch (e) {
|
|
1685
1687
|
console.log(`Failed to start job: ${e.message}`);
|
|
@@ -1723,25 +1725,32 @@ let JobManager = class JobManager {
|
|
|
1723
1725
|
this.apiPull.on("message", (name, args) => {
|
|
1724
1726
|
const message = name.toString("utf8");
|
|
1725
1727
|
const params = JSON.parse((args === null || args === void 0 ? void 0 : args.toString("utf8")) || "{}");
|
|
1726
|
-
|
|
1728
|
+
const paramTypes = Object.keys(params).reduce((res, key) => {
|
|
1729
|
+
res[key] = getType(params[key]);
|
|
1730
|
+
return res;
|
|
1731
|
+
}, {});
|
|
1732
|
+
console.log(`Received a message from worker: "${colorize(message, ConsoleColor.FgCyan)}" with args: ${jsonHighlight(paramTypes)}\n\n`);
|
|
1727
1733
|
this.messages.next({ message, params });
|
|
1728
1734
|
});
|
|
1729
1735
|
console.log(`API consumer bound to port: ${backPort}`);
|
|
1730
1736
|
}
|
|
1731
1737
|
return this.tryResolve(jobType, params);
|
|
1732
1738
|
}
|
|
1733
|
-
resolveJobInstance(jobType, params) {
|
|
1739
|
+
resolveJobInstance(jobType, params, uniqueId = "") {
|
|
1734
1740
|
const container = this.container.createChildContainer();
|
|
1735
1741
|
Object.keys(params).map((name) => {
|
|
1736
1742
|
container.register(name, { useValue: params[name] });
|
|
1737
1743
|
});
|
|
1744
|
+
container.register("uniqueId", { useValue: uniqueId });
|
|
1738
1745
|
container.register(jobType, jobType);
|
|
1739
1746
|
return container.resolve(jobType);
|
|
1740
1747
|
}
|
|
1741
1748
|
sendToWorkers(jobName, params) {
|
|
1742
1749
|
return __awaiter$o(this, void 0, void 0, function* () {
|
|
1743
1750
|
const publisher = yield this.apiPush;
|
|
1744
|
-
|
|
1751
|
+
const uniqueId = new ObjectId().toHexString();
|
|
1752
|
+
yield publisher.send([jobName, JSON.stringify(params), uniqueId]);
|
|
1753
|
+
return uniqueId;
|
|
1745
1754
|
});
|
|
1746
1755
|
}
|
|
1747
1756
|
};
|