@stemy/backend 6.0.2 → 6.0.3
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.
|
@@ -16,7 +16,6 @@ let JobManager = class JobManager {
|
|
|
16
16
|
jobTypes;
|
|
17
17
|
jobs;
|
|
18
18
|
messages;
|
|
19
|
-
processing;
|
|
20
19
|
apiPush;
|
|
21
20
|
apiPull;
|
|
22
21
|
workerPush;
|
|
@@ -34,7 +33,9 @@ let JobManager = class JobManager {
|
|
|
34
33
|
const messageBridge = {
|
|
35
34
|
sendMessage: (message, params) => {
|
|
36
35
|
params.uniqueId = uniqueId;
|
|
37
|
-
this.workerPush.
|
|
36
|
+
this.workerPush.then(sock => {
|
|
37
|
+
sock.send([message, JSON.stringify(params)]);
|
|
38
|
+
});
|
|
38
39
|
}
|
|
39
40
|
};
|
|
40
41
|
messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
@@ -43,7 +44,6 @@ let JobManager = class JobManager {
|
|
|
43
44
|
return res;
|
|
44
45
|
}, {});
|
|
45
46
|
this.messages = new Subject();
|
|
46
|
-
this.processing = null;
|
|
47
47
|
this.maxTimeout = this.config.resolve("jobTimeout");
|
|
48
48
|
}
|
|
49
49
|
on(message, cb) {
|
|
@@ -90,40 +90,46 @@ let JobManager = class JobManager {
|
|
|
90
90
|
});
|
|
91
91
|
});
|
|
92
92
|
}
|
|
93
|
-
async
|
|
93
|
+
async startProcessing() {
|
|
94
94
|
const host = this.config.resolve("zmqRemoteHost");
|
|
95
95
|
const pushHost = `${host}:${this.config.resolve("zmqBackPort")}`;
|
|
96
|
-
this.workerPush = socket("push");
|
|
97
|
-
this.workerPush.connect(pushHost);
|
|
98
|
-
this.logger.log("job-manager", `Worker producer connected to: ${pushHost}`);
|
|
99
96
|
const pullHost = `${host}:${this.config.resolve("zmqPort")}`;
|
|
100
|
-
this.
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
97
|
+
this.workerPush = this.workerPush || new Promise(resolve => {
|
|
98
|
+
const sock = socket("push");
|
|
99
|
+
sock.connect(pushHost);
|
|
100
|
+
this.logger.log("job-manager", `Worker producer connected to: ${pushHost}`);
|
|
101
|
+
resolve(sock);
|
|
102
|
+
});
|
|
103
|
+
this.workerPull = this.workerPull || new Promise(resolve => {
|
|
104
|
+
const sock = socket("pull");
|
|
105
|
+
sock.connect(pullHost);
|
|
106
|
+
sock.on("message", async (name, args, uniqId) => {
|
|
110
107
|
try {
|
|
111
|
-
|
|
112
|
-
|
|
108
|
+
const jobName = name.toString("utf8");
|
|
109
|
+
const jobParams = JSON.parse(args.toString("utf8"));
|
|
110
|
+
const uniqueId = uniqId?.toString("utf8");
|
|
111
|
+
console.time(uniqueId);
|
|
112
|
+
console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \n${jsonHighlight(jobParams)}\n\n`);
|
|
113
|
+
try {
|
|
114
|
+
await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);
|
|
115
|
+
console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\n\n`);
|
|
116
|
+
}
|
|
117
|
+
catch (e) {
|
|
118
|
+
console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\n${e}\n\n`);
|
|
119
|
+
}
|
|
120
|
+
console.timeEnd(uniqueId);
|
|
113
121
|
}
|
|
114
122
|
catch (e) {
|
|
115
|
-
|
|
123
|
+
this.logger.log("job-manager", `Failed to start job: ${e.message}`);
|
|
116
124
|
}
|
|
117
|
-
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
this.logger.log("job-manager", `Failed to start job: ${e.message}`);
|
|
121
|
-
}
|
|
125
|
+
});
|
|
126
|
+
this.logger.log("job-manager", `Worker consumer connected to: ${pullHost}`);
|
|
127
|
+
resolve(sock);
|
|
122
128
|
});
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
129
|
+
await Promise.allSettled([
|
|
130
|
+
this.workerPush,
|
|
131
|
+
this.workerPull,
|
|
132
|
+
]);
|
|
127
133
|
}
|
|
128
134
|
tryResolve(jobType, params) {
|
|
129
135
|
const jobName = getConstructorName(jobType);
|
|
@@ -148,28 +154,32 @@ let JobManager = class JobManager {
|
|
|
148
154
|
return this.tryResolveAndInit(jobType, params);
|
|
149
155
|
}
|
|
150
156
|
tryResolveAndInit(jobType, params) {
|
|
151
|
-
|
|
157
|
+
this.apiPush = this.apiPush || new Promise(resolve => {
|
|
152
158
|
const port = this.config.resolve("zmqPort");
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
159
|
+
const sock = socket("push");
|
|
160
|
+
sock.bind(`tcp://0.0.0.0:${port}`, () => {
|
|
161
|
+
this.logger.log("job-manager", `API producer bound to port: ${port}`);
|
|
162
|
+
resolve(sock);
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
this.apiPull = this.apiPush || new Promise(resolve => {
|
|
158
166
|
const backPort = this.config.resolve("zmqBackPort");
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
167
|
+
const sock = socket("pull");
|
|
168
|
+
sock.bind(`tcp://0.0.0.0:${backPort}`, () => {
|
|
169
|
+
sock.on("message", (name, args) => {
|
|
170
|
+
const message = name.toString("utf8");
|
|
171
|
+
const params = JSON.parse(args?.toString("utf8") || "{}");
|
|
172
|
+
const paramTypes = Object.keys(params).reduce((res, key) => {
|
|
173
|
+
res[key] = getType(params[key]);
|
|
174
|
+
return res;
|
|
175
|
+
}, {});
|
|
176
|
+
this.logger.log("job-manager", `Received a message from worker: "${colorize(message, ConsoleColor.FgCyan)}" with args: ${jsonHighlight(paramTypes)}\n\n`);
|
|
177
|
+
this.messages.next({ message, params });
|
|
178
|
+
});
|
|
179
|
+
this.logger.log("job-manager", `API consumer bound to port: ${backPort}`);
|
|
180
|
+
resolve(sock);
|
|
170
181
|
});
|
|
171
|
-
|
|
172
|
-
}
|
|
182
|
+
});
|
|
173
183
|
return this.tryResolve(jobType, params);
|
|
174
184
|
}
|
|
175
185
|
resolveJobInstance(jobType, params, uniqueId = "") {
|
|
@@ -183,7 +193,8 @@ let JobManager = class JobManager {
|
|
|
183
193
|
}
|
|
184
194
|
async sendToWorkers(jobName, params) {
|
|
185
195
|
const uniqueId = new ObjectId().toHexString();
|
|
186
|
-
this.apiPush
|
|
196
|
+
const sock = await this.apiPush;
|
|
197
|
+
sock.send([jobName, JSON.stringify(params), uniqueId]);
|
|
187
198
|
return uniqueId;
|
|
188
199
|
}
|
|
189
200
|
};
|
|
@@ -195,4 +206,4 @@ JobManager = __decorate([
|
|
|
195
206
|
Logger, Object, Array])
|
|
196
207
|
], JobManager);
|
|
197
208
|
export { JobManager };
|
|
198
|
-
//# 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, injectAll, singleton} from \"tsyringe\";\nimport cron from \"node-cron\";\nimport {socket, Socket} from \"zeromq/v5-compat\";\nimport {Subject, Subscription} from \"rxjs\";\nimport {filter, map} from \"rxjs/operators\";\nimport {ObjectId} from \"bson\";\nimport {\n    DI_CONTAINER,\n    IJob,\n    IJobTask,\n    IMessageBridge,\n    ISocketMessage,\n    JOB,\n    JobParams,\n    JobScheduleRange,\n    JobScheduleTime,\n    SocketParams,\n    Type\n} from \"../common-types\";\nimport {\n    colorize,\n    ConsoleColor,\n    getConstructorName,\n    getType,\n    isArray,\n    isObject,\n    jsonHighlight,\n    promiseTimeout\n} from \"../utils\";\nimport {Configuration} from \"./configuration\";\nimport {Logger} from \"./logger\";\n\n@singleton()\nexport class JobManager {\n\n    protected jobTypes: Type<IJob>[];\n    protected jobs: { [name: string]: (jobParams: JobParams, uniqueId: string) => Promise<any> };\n    protected messages: Subject<ISocketMessage>;\n    protected processing: Promise<any>;\n\n    protected apiPush: Socket;\n    protected apiPull: Socket;\n    protected workerPush: Socket;\n    protected workerPull: Socket;\n\n    readonly maxTimeout: number;\n\n    constructor(readonly config: Configuration,\n                readonly logger: Logger,\n                @inject(DI_CONTAINER) readonly container: DependencyContainer,\n                @injectAll(JOB) jobTypes: Type<IJob>[]) {\n        this.jobTypes = jobTypes || [];\n        this.jobs = this.jobTypes.reduce((res, jobType) => {\n            const jobName = getConstructorName(jobType);\n            res[jobName] = (jobParams: JobParams, uniqueId: string) => {\n                const job = this.resolveJobInstance(jobType, jobParams, uniqueId);\n                const messageBridge: IMessageBridge = {\n                    sendMessage: (message: string, params?: SocketParams) => {\n                        params.uniqueId = uniqueId;\n                        this.workerPush.send([message, JSON.stringify(params)]);\n                    }\n                };\n                messageBridge.sendMessage(`job-started`, {name: jobName});\n                return job.process(messageBridge);\n            }\n            return res;\n        }, {});\n        this.messages = new Subject<ISocketMessage>();\n        this.processing = null;\n        this.maxTimeout = this.config.resolve(\"jobTimeout\");\n    }\n\n    on(message: string, cb: (params: SocketParams) => any): Subscription {\n        return this.messages\n            .pipe(filter(t => t.message === message))\n            .pipe(map(t => t.params)).subscribe(cb);\n    }\n\n    async process(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\n        let instance: IJob = null;\n        try {\n            instance = this.resolveJobInstance(jobType, params);\n        } catch (e) {\n            const jobName = getConstructorName(jobType);\n            throw new Error(`Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`);\n        }\n        return instance.process();\n    }\n\n    async enqueueWithName(name: string, params: JobParams = {}): Promise<string> {\n        return this.sendToWorkers(this.tryResolveFromName(name, params), params);\n    }\n\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<string> {\n        return this.sendToWorkers(this.tryResolveAndInit(jobType, params), params);\n    }\n\n    schedule(minute: JobScheduleTime, hour: JobScheduleTime, dayOfMonth: JobScheduleTime, month: JobScheduleTime, dayOfWeek: JobScheduleTime, jobType: Type<IJob>, params: JobParams = {}): IJobTask {\n        const expression = [minute, hour, dayOfMonth, month, dayOfWeek].map(t => {\n            if (isObject(t)) {\n                const range = t as JobScheduleRange;\n                return `${range.min || 0}-${range.max || 0}`;\n            }\n            if (isArray(t)) {\n                return t.join(\",\");\n            }\n            return `${t}`;\n        }).join(\" \");\n        const jobName = getConstructorName(jobType);\n        if (!cron.validate(expression)) {\n            this.logger.log(\"job-manager\", `Can't schedule the task: '${jobName}' because time expression is invalid.`);\n            return null;\n        }\n        return cron.schedule(expression, () => {\n            this.enqueue(jobType, params).catch(e => {\n                this.logger.log(\"job-manager\", `Can't enqueue job: '${jobName}' because: ${e}`);\n            });\n        });\n    }\n\n    protected async initProcessing(): Promise<void> {\n        const host = this.config.resolve(\"zmqRemoteHost\");\n        const pushHost = `${host}:${this.config.resolve(\"zmqBackPort\")}`;\n        this.workerPush = socket(\"push\");\n        this.workerPush.connect(pushHost);\n        this.logger.log(\"job-manager\", `Worker producer connected to: ${pushHost}`);\n\n        const pullHost = `${host}:${this.config.resolve(\"zmqPort\")}`;\n        this.workerPull = socket(\"pull\");\n        this.workerPull.connect(pullHost);\n        this.logger.log(\"job-manager\", `Worker consumer connected to: ${pullHost}`);\n\n        this.workerPull.on(\"message\", async (name: Buffer, args: Buffer, uniqId: Buffer) => {\n            try {\n                const jobName = name.toString(\"utf8\");\n                const jobParams = JSON.parse(args.toString(\"utf8\")) as JobParams;\n                const uniqueId = uniqId?.toString(\"utf8\");\n\n                console.time(uniqueId);\n                console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \\n${jsonHighlight(jobParams)}\\n\\n`);\n\n                try {\n                    await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);\n                    console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\\n\\n`);\n                } catch (e) {\n                    console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\\n${e}\\n\\n`);\n                }\n                console.timeEnd(uniqueId);\n            } catch (e) {\n                this.logger.log(\"job-manager\", `Failed to start job: ${e.message}`);\n            }\n        });\n    }\n\n    startProcessing(): Promise<void> {\n        this.processing = this.processing || this.initProcessing();\n        return this.processing;\n    }\n\n    tryResolve(jobType: Type<IJob>, params: JobParams): string {\n        const jobName = getConstructorName(jobType);\n        if (!this.jobs[jobName]) {\n            throw `Can't find job with name: ${jobName} so it can't be enqueued!`;\n        }\n        try {\n            this.resolveJobInstance(jobType, params);\n        } catch (e) {\n            throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`;\n        }\n        return jobName;\n    }\n\n    protected tryResolveFromName(jobName: string, params: JobParams) {\n        const jobType = this.jobTypes.find(type => {\n            return getConstructorName(type) == jobName;\n        });\n        if (!jobType) {\n            throw `Can't find job type with name: ${jobName} so it can't be enqueued!`;\n        }\n        return this.tryResolveAndInit(jobType, params);\n    }\n\n    protected tryResolveAndInit(jobType: Type<IJob>, params: JobParams) {\n        if (!this.apiPush) {\n            const port = this.config.resolve(\"zmqPort\");\n            this.apiPush = socket(\"push\");\n            this.apiPush.bind(`tcp://0.0.0.0:${port}`);\n            this.logger.log(\"job-manager\", `API producer bound to port: ${port}`);\n        }\n        if (!this.apiPull) {\n            const backPort = this.config.resolve(\"zmqBackPort\");\n            this.apiPull = socket(\"pull\");\n            this.apiPull.bind(`tcp://0.0.0.0:${backPort}`);\n            this.apiPull.on(\"message\", (name: Buffer, args?: Buffer) => {\n                const message = name.toString(\"utf8\");\n                const params = JSON.parse(args?.toString(\"utf8\") || \"{}\") as JobParams;\n                const paramTypes = Object.keys(params).reduce((res, key) => {\n                    res[key] = getType(params[key]);\n                    return res;\n                }, {});\n                this.logger.log(\"job-manager\", `Received a message from worker: \"${colorize(message, ConsoleColor.FgCyan)}\" with args: ${jsonHighlight(paramTypes)}\\n\\n`);\n                this.messages.next({message, params});\n            });\n            this.logger.log(\"job-manager\", `API consumer bound to port: ${backPort}`);\n        }\n        return this.tryResolve(jobType, params);\n    }\n\n    protected resolveJobInstance(jobType: Type<IJob>, params: JobParams, uniqueId: string = \"\") {\n        const container = this.container.createChildContainer();\n        Object.keys(params).map((name) => {\n            container.register(name, {useValue: params[name]});\n        });\n        container.register(\"uniqueId\", {useValue: uniqueId});\n        container.register(jobType, jobType);\n        return container.resolve(jobType) as IJob;\n    }\n\n    protected async sendToWorkers(jobName: string, params: JobParams) {\n        const uniqueId = new ObjectId().toHexString();\n        this.apiPush.send([jobName, JSON.stringify(params), uniqueId]);\n        return uniqueId;\n    }\n}\n"]}
|
|
209
|
+
//# 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, injectAll, singleton} from \"tsyringe\";\nimport cron from \"node-cron\";\nimport {socket, Socket} from \"zeromq/v5-compat\";\nimport {Subject, Subscription} from \"rxjs\";\nimport {filter, map} from \"rxjs/operators\";\nimport {ObjectId} from \"bson\";\nimport {\n    DI_CONTAINER,\n    IJob,\n    IJobTask,\n    IMessageBridge,\n    ISocketMessage,\n    JOB,\n    JobParams,\n    JobScheduleRange,\n    JobScheduleTime,\n    SocketParams,\n    Type\n} from \"../common-types\";\nimport {\n    colorize,\n    ConsoleColor,\n    getConstructorName,\n    getType,\n    isArray,\n    isObject,\n    jsonHighlight,\n    promiseTimeout\n} from \"../utils\";\nimport {Configuration} from \"./configuration\";\nimport {Logger} from \"./logger\";\n\n@singleton()\nexport class JobManager {\n\n    protected jobTypes: Type<IJob>[];\n    protected jobs: { [name: string]: (jobParams: JobParams, uniqueId: string) => Promise<any> };\n    protected messages: Subject<ISocketMessage>;\n\n    protected apiPush: Promise<Socket>;\n    protected apiPull: Promise<Socket>;\n    protected workerPush: Promise<Socket>;\n    protected workerPull: Promise<Socket>;\n\n    readonly maxTimeout: number;\n\n    constructor(readonly config: Configuration,\n                readonly logger: Logger,\n                @inject(DI_CONTAINER) readonly container: DependencyContainer,\n                @injectAll(JOB) jobTypes: Type<IJob>[]) {\n        this.jobTypes = jobTypes || [];\n        this.jobs = this.jobTypes.reduce((res, jobType) => {\n            const jobName = getConstructorName(jobType);\n            res[jobName] = (jobParams: JobParams, uniqueId: string) => {\n                const job = this.resolveJobInstance(jobType, jobParams, uniqueId);\n                const messageBridge: IMessageBridge = {\n                    sendMessage: (message: string, params?: SocketParams) => {\n                        params.uniqueId = uniqueId;\n                        this.workerPush.then(sock => {\n                            sock.send([message, JSON.stringify(params)]);\n                        });\n                    }\n                };\n                messageBridge.sendMessage(`job-started`, {name: jobName});\n                return job.process(messageBridge);\n            }\n            return res;\n        }, {});\n        this.messages = new Subject<ISocketMessage>();\n        this.maxTimeout = this.config.resolve(\"jobTimeout\");\n    }\n\n    on(message: string, cb: (params: SocketParams) => any): Subscription {\n        return this.messages\n            .pipe(filter(t => t.message === message))\n            .pipe(map(t => t.params)).subscribe(cb);\n    }\n\n    async process(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\n        let instance: IJob = null;\n        try {\n            instance = this.resolveJobInstance(jobType, params);\n        } catch (e) {\n            const jobName = getConstructorName(jobType);\n            throw new Error(`Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`);\n        }\n        return instance.process();\n    }\n\n    async enqueueWithName(name: string, params: JobParams = {}): Promise<string> {\n        return this.sendToWorkers(this.tryResolveFromName(name, params), params);\n    }\n\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<string> {\n        return this.sendToWorkers(this.tryResolveAndInit(jobType, params), params);\n    }\n\n    schedule(minute: JobScheduleTime, hour: JobScheduleTime, dayOfMonth: JobScheduleTime, month: JobScheduleTime, dayOfWeek: JobScheduleTime, jobType: Type<IJob>, params: JobParams = {}): IJobTask {\n        const expression = [minute, hour, dayOfMonth, month, dayOfWeek].map(t => {\n            if (isObject(t)) {\n                const range = t as JobScheduleRange;\n                return `${range.min || 0}-${range.max || 0}`;\n            }\n            if (isArray(t)) {\n                return t.join(\",\");\n            }\n            return `${t}`;\n        }).join(\" \");\n        const jobName = getConstructorName(jobType);\n        if (!cron.validate(expression)) {\n            this.logger.log(\"job-manager\", `Can't schedule the task: '${jobName}' because time expression is invalid.`);\n            return null;\n        }\n        return cron.schedule(expression, () => {\n            this.enqueue(jobType, params).catch(e => {\n                this.logger.log(\"job-manager\", `Can't enqueue job: '${jobName}' because: ${e}`);\n            });\n        });\n    }\n\n    async startProcessing(): Promise<void> {\n        const host = this.config.resolve(\"zmqRemoteHost\");\n        const pushHost = `${host}:${this.config.resolve(\"zmqBackPort\")}`;\n        const pullHost = `${host}:${this.config.resolve(\"zmqPort\")}`;\n        this.workerPush = this.workerPush || new Promise(resolve => {\n            const sock = socket(\"push\");\n            sock.connect(pushHost);\n            this.logger.log(\"job-manager\", `Worker producer connected to: ${pushHost}`);\n            resolve(sock);\n        });\n        this.workerPull = this.workerPull || new Promise(resolve => {\n            const sock = socket(\"pull\");\n            sock.connect(pullHost);\n            sock.on(\"message\", async (name: Buffer, args: Buffer, uniqId: Buffer) => {\n                try {\n                    const jobName = name.toString(\"utf8\");\n                    const jobParams = JSON.parse(args.toString(\"utf8\")) as JobParams;\n                    const uniqueId = uniqId?.toString(\"utf8\");\n\n                    console.time(uniqueId);\n                    console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \\n${jsonHighlight(jobParams)}\\n\\n`);\n\n                    try {\n                        await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);\n                        console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\\n\\n`);\n                    } catch (e) {\n                        console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\\n${e}\\n\\n`);\n                    }\n                    console.timeEnd(uniqueId);\n                } catch (e) {\n                    this.logger.log(\"job-manager\", `Failed to start job: ${e.message}`);\n                }\n            });\n            this.logger.log(\"job-manager\", `Worker consumer connected to: ${pullHost}`);\n            resolve(sock);\n        });\n\n        await Promise.allSettled([\n            this.workerPush,\n            this.workerPull,\n        ]);\n    }\n\n    tryResolve(jobType: Type<IJob>, params: JobParams): string {\n        const jobName = getConstructorName(jobType);\n        if (!this.jobs[jobName]) {\n            throw `Can't find job with name: ${jobName} so it can't be enqueued!`;\n        }\n        try {\n            this.resolveJobInstance(jobType, params);\n        } catch (e) {\n            throw `Can't resolve params for job: ${jobName}, with params: ${JSON.stringify(params)}. Reason: ${e}`;\n        }\n        return jobName;\n    }\n\n    protected tryResolveFromName(jobName: string, params: JobParams) {\n        const jobType = this.jobTypes.find(type => {\n            return getConstructorName(type) == jobName;\n        });\n        if (!jobType) {\n            throw `Can't find job type with name: ${jobName} so it can't be enqueued!`;\n        }\n        return this.tryResolveAndInit(jobType, params);\n    }\n\n    protected tryResolveAndInit(jobType: Type<IJob>, params: JobParams) {\n        this.apiPush = this.apiPush || new Promise(resolve => {\n            const port = this.config.resolve(\"zmqPort\");\n            const sock = socket(\"push\");\n            sock.bind(`tcp://0.0.0.0:${port}`, () => {\n                this.logger.log(\"job-manager\", `API producer bound to port: ${port}`);\n                resolve(sock);\n            });\n        });\n        this.apiPull = this.apiPush || new Promise(resolve => {\n            const backPort = this.config.resolve(\"zmqBackPort\");\n            const sock = socket(\"pull\");\n            sock.bind(`tcp://0.0.0.0:${backPort}`, () => {\n                sock.on(\"message\", (name: Buffer, args?: Buffer) => {\n                    const message = name.toString(\"utf8\");\n                    const params = JSON.parse(args?.toString(\"utf8\") || \"{}\") as JobParams;\n                    const paramTypes = Object.keys(params).reduce((res, key) => {\n                        res[key] = getType(params[key]);\n                        return res;\n                    }, {});\n                    this.logger.log(\"job-manager\", `Received a message from worker: \"${colorize(message, ConsoleColor.FgCyan)}\" with args: ${jsonHighlight(paramTypes)}\\n\\n`);\n                    this.messages.next({message, params});\n                });\n                this.logger.log(\"job-manager\", `API consumer bound to port: ${backPort}`);\n                resolve(sock);\n            });\n        });\n        return this.tryResolve(jobType, params);\n    }\n\n    protected resolveJobInstance(jobType: Type<IJob>, params: JobParams, uniqueId: string = \"\") {\n        const container = this.container.createChildContainer();\n        Object.keys(params).map((name) => {\n            container.register(name, {useValue: params[name]});\n        });\n        container.register(\"uniqueId\", {useValue: uniqueId});\n        container.register(jobType, jobType);\n        return container.resolve(jobType) as IJob;\n    }\n\n    protected async sendToWorkers(jobName: string, params: JobParams) {\n        const uniqueId = new ObjectId().toHexString();\n        const sock = await this.apiPush;\n        sock.send([jobName, JSON.stringify(params), uniqueId]);\n        return uniqueId;\n    }\n}\n"]}
|
|
@@ -1355,7 +1355,6 @@ let JobManager = class JobManager {
|
|
|
1355
1355
|
jobTypes;
|
|
1356
1356
|
jobs;
|
|
1357
1357
|
messages;
|
|
1358
|
-
processing;
|
|
1359
1358
|
apiPush;
|
|
1360
1359
|
apiPull;
|
|
1361
1360
|
workerPush;
|
|
@@ -1373,7 +1372,9 @@ let JobManager = class JobManager {
|
|
|
1373
1372
|
const messageBridge = {
|
|
1374
1373
|
sendMessage: (message, params) => {
|
|
1375
1374
|
params.uniqueId = uniqueId;
|
|
1376
|
-
this.workerPush.
|
|
1375
|
+
this.workerPush.then(sock => {
|
|
1376
|
+
sock.send([message, JSON.stringify(params)]);
|
|
1377
|
+
});
|
|
1377
1378
|
}
|
|
1378
1379
|
};
|
|
1379
1380
|
messageBridge.sendMessage(`job-started`, { name: jobName });
|
|
@@ -1382,7 +1383,6 @@ let JobManager = class JobManager {
|
|
|
1382
1383
|
return res;
|
|
1383
1384
|
}, {});
|
|
1384
1385
|
this.messages = new Subject();
|
|
1385
|
-
this.processing = null;
|
|
1386
1386
|
this.maxTimeout = this.config.resolve("jobTimeout");
|
|
1387
1387
|
}
|
|
1388
1388
|
on(message, cb) {
|
|
@@ -1429,40 +1429,46 @@ let JobManager = class JobManager {
|
|
|
1429
1429
|
});
|
|
1430
1430
|
});
|
|
1431
1431
|
}
|
|
1432
|
-
async
|
|
1432
|
+
async startProcessing() {
|
|
1433
1433
|
const host = this.config.resolve("zmqRemoteHost");
|
|
1434
1434
|
const pushHost = `${host}:${this.config.resolve("zmqBackPort")}`;
|
|
1435
|
-
this.workerPush = socket("push");
|
|
1436
|
-
this.workerPush.connect(pushHost);
|
|
1437
|
-
this.logger.log("job-manager", `Worker producer connected to: ${pushHost}`);
|
|
1438
1435
|
const pullHost = `${host}:${this.config.resolve("zmqPort")}`;
|
|
1439
|
-
this.
|
|
1440
|
-
|
|
1441
|
-
|
|
1442
|
-
|
|
1443
|
-
|
|
1444
|
-
|
|
1445
|
-
|
|
1446
|
-
|
|
1447
|
-
|
|
1448
|
-
|
|
1436
|
+
this.workerPush = this.workerPush || new Promise(resolve => {
|
|
1437
|
+
const sock = socket("push");
|
|
1438
|
+
sock.connect(pushHost);
|
|
1439
|
+
this.logger.log("job-manager", `Worker producer connected to: ${pushHost}`);
|
|
1440
|
+
resolve(sock);
|
|
1441
|
+
});
|
|
1442
|
+
this.workerPull = this.workerPull || new Promise(resolve => {
|
|
1443
|
+
const sock = socket("pull");
|
|
1444
|
+
sock.connect(pullHost);
|
|
1445
|
+
sock.on("message", async (name, args, uniqId) => {
|
|
1449
1446
|
try {
|
|
1450
|
-
|
|
1451
|
-
|
|
1447
|
+
const jobName = name.toString("utf8");
|
|
1448
|
+
const jobParams = JSON.parse(args.toString("utf8"));
|
|
1449
|
+
const uniqueId = uniqId?.toString("utf8");
|
|
1450
|
+
console.time(uniqueId);
|
|
1451
|
+
console.timeLog(uniqueId, `Started working on background job: ${colorize(jobName, ConsoleColor.FgCyan)} with args: \n${jsonHighlight(jobParams)}\n\n`);
|
|
1452
|
+
try {
|
|
1453
|
+
await Promise.race([this.jobs[jobName](jobParams, uniqueId), promiseTimeout(this.maxTimeout, true)]);
|
|
1454
|
+
console.timeLog(uniqueId, `Finished working on background job: ${colorize(jobName, ConsoleColor.FgCyan)}\n\n`);
|
|
1455
|
+
}
|
|
1456
|
+
catch (e) {
|
|
1457
|
+
console.timeLog(uniqueId, `Background job failed: ${colorize(jobName, ConsoleColor.FgRed)}\n${e}\n\n`);
|
|
1458
|
+
}
|
|
1459
|
+
console.timeEnd(uniqueId);
|
|
1452
1460
|
}
|
|
1453
1461
|
catch (e) {
|
|
1454
|
-
|
|
1462
|
+
this.logger.log("job-manager", `Failed to start job: ${e.message}`);
|
|
1455
1463
|
}
|
|
1456
|
-
|
|
1457
|
-
}
|
|
1458
|
-
|
|
1459
|
-
this.logger.log("job-manager", `Failed to start job: ${e.message}`);
|
|
1460
|
-
}
|
|
1464
|
+
});
|
|
1465
|
+
this.logger.log("job-manager", `Worker consumer connected to: ${pullHost}`);
|
|
1466
|
+
resolve(sock);
|
|
1461
1467
|
});
|
|
1462
|
-
|
|
1463
|
-
|
|
1464
|
-
|
|
1465
|
-
|
|
1468
|
+
await Promise.allSettled([
|
|
1469
|
+
this.workerPush,
|
|
1470
|
+
this.workerPull,
|
|
1471
|
+
]);
|
|
1466
1472
|
}
|
|
1467
1473
|
tryResolve(jobType, params) {
|
|
1468
1474
|
const jobName = getConstructorName(jobType);
|
|
@@ -1487,28 +1493,32 @@ let JobManager = class JobManager {
|
|
|
1487
1493
|
return this.tryResolveAndInit(jobType, params);
|
|
1488
1494
|
}
|
|
1489
1495
|
tryResolveAndInit(jobType, params) {
|
|
1490
|
-
|
|
1496
|
+
this.apiPush = this.apiPush || new Promise(resolve => {
|
|
1491
1497
|
const port = this.config.resolve("zmqPort");
|
|
1492
|
-
|
|
1493
|
-
|
|
1494
|
-
|
|
1495
|
-
|
|
1496
|
-
|
|
1498
|
+
const sock = socket("push");
|
|
1499
|
+
sock.bind(`tcp://0.0.0.0:${port}`, () => {
|
|
1500
|
+
this.logger.log("job-manager", `API producer bound to port: ${port}`);
|
|
1501
|
+
resolve(sock);
|
|
1502
|
+
});
|
|
1503
|
+
});
|
|
1504
|
+
this.apiPull = this.apiPush || new Promise(resolve => {
|
|
1497
1505
|
const backPort = this.config.resolve("zmqBackPort");
|
|
1498
|
-
|
|
1499
|
-
|
|
1500
|
-
|
|
1501
|
-
|
|
1502
|
-
|
|
1503
|
-
|
|
1504
|
-
|
|
1505
|
-
|
|
1506
|
-
|
|
1507
|
-
|
|
1508
|
-
|
|
1506
|
+
const sock = socket("pull");
|
|
1507
|
+
sock.bind(`tcp://0.0.0.0:${backPort}`, () => {
|
|
1508
|
+
sock.on("message", (name, args) => {
|
|
1509
|
+
const message = name.toString("utf8");
|
|
1510
|
+
const params = JSON.parse(args?.toString("utf8") || "{}");
|
|
1511
|
+
const paramTypes = Object.keys(params).reduce((res, key) => {
|
|
1512
|
+
res[key] = getType(params[key]);
|
|
1513
|
+
return res;
|
|
1514
|
+
}, {});
|
|
1515
|
+
this.logger.log("job-manager", `Received a message from worker: "${colorize(message, ConsoleColor.FgCyan)}" with args: ${jsonHighlight(paramTypes)}\n\n`);
|
|
1516
|
+
this.messages.next({ message, params });
|
|
1517
|
+
});
|
|
1518
|
+
this.logger.log("job-manager", `API consumer bound to port: ${backPort}`);
|
|
1519
|
+
resolve(sock);
|
|
1509
1520
|
});
|
|
1510
|
-
|
|
1511
|
-
}
|
|
1521
|
+
});
|
|
1512
1522
|
return this.tryResolve(jobType, params);
|
|
1513
1523
|
}
|
|
1514
1524
|
resolveJobInstance(jobType, params, uniqueId = "") {
|
|
@@ -1522,7 +1532,8 @@ let JobManager = class JobManager {
|
|
|
1522
1532
|
}
|
|
1523
1533
|
async sendToWorkers(jobName, params) {
|
|
1524
1534
|
const uniqueId = new ObjectId$1().toHexString();
|
|
1525
|
-
this.apiPush
|
|
1535
|
+
const sock = await this.apiPush;
|
|
1536
|
+
sock.send([jobName, JSON.stringify(params), uniqueId]);
|
|
1526
1537
|
return uniqueId;
|
|
1527
1538
|
}
|
|
1528
1539
|
};
|