@stemy/backend 2.9.6 → 3.0.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.
@@ -46,9 +46,10 @@ export class Asset extends BaseEntity {
46
46
  params = params || {};
47
47
  // Get default crop info
48
48
  const crop = Asset.toCropRegion(meta.crop);
49
- // Return back the stream if there is no params and no default crop exists
50
- if (Object.keys(params).length == 0 && !crop)
49
+ // Return the stream if there is no params and no default crop exists
50
+ if ((meta === null || meta === void 0 ? void 0 : meta.extension) === "svg" || (Object.keys(params).length == 0 && !crop)) {
51
51
  return stream;
52
+ }
52
53
  // Parse params
53
54
  params.rotation = isNaN(params.rotation) ? 0 : Math.round(params.rotation / 90) * 90;
54
55
  params.canvasScaleX = isNaN(params.canvasScaleX) ? 1 : Number(params.canvasScaleX);
@@ -63,51 +64,43 @@ export class Asset extends BaseEntity {
63
64
  const cropBefore = Asset.toCropRegion(params.cropBefore || (params.crop ? meta.cropBefore : null));
64
65
  const cropAfter = Asset.toCropRegion(params.cropAfter || (params.crop ? meta.cropAfter : null));
65
66
  // Get metadata
66
- const imgMeta = yield sharp(buffer).metadata();
67
- let width = imgMeta.width;
68
- let height = imgMeta.height;
67
+ let img = sharp(buffer);
68
+ let { width, height } = yield img.metadata();
69
69
  // Crop before resize
70
70
  if (cropBefore) {
71
- buffer = yield sharp(buffer)
72
- .extract(cropBefore)
73
- .toBuffer();
74
71
  width = cropBefore.width;
75
72
  height = cropBefore.height;
73
+ img = img.extract(cropBefore);
76
74
  }
77
75
  else if (crop) {
78
- buffer = yield sharp(buffer)
79
- .extract(crop)
80
- .toBuffer();
81
76
  width = crop.width;
82
77
  height = crop.height;
78
+ img = img.extract(crop);
83
79
  }
84
80
  // Resize canvas
85
- if (params.canvasScaleX !== 1 || params.canvasScaleY !== 1) {
81
+ const canvasScaleX = (meta === null || meta === void 0 ? void 0 : meta.canvasScaleX) || 1;
82
+ const canvasScaleY = (meta === null || meta === void 0 ? void 0 : meta.canvasScaleY) || 1;
83
+ if (params.canvasScaleX !== canvasScaleX || params.canvasScaleY !== canvasScaleY) {
86
84
  width = Math.round(width * params.canvasScaleX);
87
85
  height = Math.round(height * params.canvasScaleY);
88
- buffer = yield sharp(buffer)
89
- .resize({ width, height, background: "#00000000", fit: "contain" })
90
- .toBuffer();
86
+ img = img.resize({ width, height, background: "#00000000", fit: "contain" });
91
87
  }
92
88
  // Resize image
93
89
  if (params.scaleX !== 1 || params.scaleY !== 1) {
94
90
  width = Math.round(width * params.scaleX);
95
91
  height = Math.round(height * params.scaleY);
96
- buffer = yield sharp(buffer)
97
- .resize({ width, height, background: "#00000000", fit: "fill" })
98
- .toBuffer();
92
+ img = img.resize({ width, height, background: "#00000000", fit: "fill" });
99
93
  }
100
94
  // Crop after resize
101
95
  if (cropAfter) {
102
- buffer = yield sharp(buffer)
103
- .extract(cropAfter)
104
- .toBuffer();
96
+ img = img.extract(cropAfter);
105
97
  }
106
98
  // Rotate
107
99
  if (params.rotation !== 0) {
108
- buffer = yield sharp(buffer).rotate(params.rotation).toBuffer();
100
+ buffer = yield img.toBuffer();
101
+ img = sharp(buffer).rotate(params.rotation);
109
102
  }
110
- return bufferToStream(buffer);
103
+ return bufferToStream(yield img.toBuffer());
111
104
  }
112
105
  catch (e) {
113
106
  console.log("Asset image conversion error", e);
@@ -158,4 +151,4 @@ export class Asset extends BaseEntity {
158
151
  });
159
152
  }
160
153
  }
161
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset.js","sourceRoot":"","sources":["../../../../src/services/entities/asset.ts"],"names":[],"mappings":"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","sourcesContent":["import sharp_, {Region} from \"sharp\";\r\nimport {Readable} from \"stream\";\r\nimport {Collection, GridFSBucket} from \"mongodb\";\r\nimport {ObjectId} from \"bson\";\r\n\r\nimport {IAsset, IAssetCropInfo, IAssetImageParams, IAssetMeta} from \"../../common-types\";\r\nimport {bufferToStream, deleteFromBucket, isBoolean, isInterface, isString, streamToBuffer} from \"../../utils\";\r\nimport {BaseEntity} from \"./base-entity\";\r\n\r\nconst sharp = sharp_;\r\nconst cropInterface = {\r\n    x: \"number\",\r\n    y: \"number\",\r\n    w: \"number\",\r\n    h: \"number\"\r\n};\r\n\r\nexport class Asset extends BaseEntity<IAsset> implements IAsset {\r\n\r\n    protected static toCropRegion(cropInfo: string | IAssetCropInfo): Region {\r\n        let crop = cropInfo as IAssetCropInfo;\r\n        if (isString(cropInfo)) {\r\n            try {\r\n                crop = JSON.parse(cropInfo as string);\r\n            } catch (e) {\r\n                return null;\r\n            }\r\n        }\r\n        if (!isInterface(crop, cropInterface)) return null;\r\n        return {\r\n            width: Math.round(crop.w),\r\n            height: Math.round(crop.h),\r\n            top: Math.round(crop.y),\r\n            left: Math.round(crop.x)\r\n        };\r\n    }\r\n\r\n    protected static async toImage(stream: Readable, meta?: IAssetMeta, params?: IAssetImageParams): Promise<Readable> {\r\n        params = params || {};\r\n\r\n        // Get default crop info\r\n        const crop = Asset.toCropRegion(meta.crop);\r\n\r\n        // Return back the stream if there is no params and no default crop exists\r\n        if (Object.keys(params).length == 0 && !crop) return stream;\r\n\r\n        // Parse params\r\n        params.rotation = isNaN(params.rotation) ? 0 : Math.round(params.rotation / 90) * 90;\r\n        params.canvasScaleX = isNaN(params.canvasScaleX) ? 1 : Number(params.canvasScaleX);\r\n        params.canvasScaleY = isNaN(params.canvasScaleY) ? 1 : Number(params.canvasScaleY);\r\n        params.scaleX = isNaN(params.scaleX) ? 1 : Number(params.scaleX);\r\n        params.scaleY = isNaN(params.scaleY) ? 1 : Number(params.scaleY);\r\n        params.crop = isBoolean(params.crop) ? params.crop : params.crop == \"true\";\r\n\r\n        // Try to modify image\r\n        let buffer = await streamToBuffer(stream);\r\n        try {\r\n            // Get crop info\r\n            const cropBefore = Asset.toCropRegion(params.cropBefore || (params.crop ? meta.cropBefore : null));\r\n            const cropAfter = Asset.toCropRegion(params.cropAfter || (params.crop ? meta.cropAfter : null));\r\n            // Get metadata\r\n            const imgMeta = await sharp(buffer).metadata();\r\n            let width = imgMeta.width;\r\n            let height = imgMeta.height;\r\n            // Crop before resize\r\n            if (cropBefore) {\r\n                buffer = await sharp(buffer)\r\n                    .extract(cropBefore)\r\n                    .toBuffer();\r\n                width = cropBefore.width;\r\n                height = cropBefore.height;\r\n            } else if (crop) {\r\n                buffer = await sharp(buffer)\r\n                    .extract(crop)\r\n                    .toBuffer();\r\n                width = crop.width;\r\n                height = crop.height;\r\n            }\r\n            // Resize canvas\r\n            if (params.canvasScaleX !== 1 || params.canvasScaleY !== 1) {\r\n                width = Math.round(width * params.canvasScaleX);\r\n                height = Math.round(height * params.canvasScaleY);\r\n                buffer = await sharp(buffer)\r\n                    .resize({width, height, background: \"#00000000\", fit: \"contain\"})\r\n                    .toBuffer();\r\n            }\r\n            // Resize image\r\n            if (params.scaleX !== 1 || params.scaleY !== 1) {\r\n                width = Math.round(width * params.scaleX);\r\n                height = Math.round(height * params.scaleY);\r\n                buffer = await sharp(buffer)\r\n                    .resize({width, height, background: \"#00000000\", fit: \"fill\"})\r\n                    .toBuffer();\r\n            }\r\n            // Crop after resize\r\n            if (cropAfter) {\r\n                buffer = await sharp(buffer)\r\n                    .extract(cropAfter)\r\n                    .toBuffer();\r\n            }\r\n            // Rotate\r\n            if (params.rotation !== 0) {\r\n                buffer = await sharp(buffer).rotate(params.rotation).toBuffer();\r\n            }\r\n            return bufferToStream(buffer);\r\n        } catch (e) {\r\n            console.log(\"Asset image conversion error\", e);\r\n            return bufferToStream(buffer);\r\n        }\r\n    }\r\n\r\n    get filename(): string {\r\n        return this.data.filename;\r\n    }\r\n\r\n    get contentType(): string {\r\n        return this.data.contentType;\r\n    }\r\n\r\n    get metadata(): IAssetMeta {\r\n        return this.data.metadata;\r\n    }\r\n\r\n    get stream(): Readable {\r\n        return this.bucket.openDownloadStream(this.mId);\r\n    }\r\n\r\n    constructor(id: ObjectId,\r\n                data: Partial<IAsset>,\r\n                collection: Collection,\r\n                protected bucket: GridFSBucket) {\r\n        super(id, data, collection);\r\n    }\r\n\r\n    async unlink(): Promise<string> {\r\n        return deleteFromBucket(this.bucket, this.mId);\r\n    }\r\n\r\n    getBuffer(): Promise<Buffer> {\r\n        return streamToBuffer(this.stream);\r\n    }\r\n\r\n    async download(metadata?: IAssetMeta): Promise<Readable> {\r\n        metadata = Object.assign(this.metadata, metadata || {});\r\n        metadata.downloadCount = isNaN(metadata.downloadCount) || !metadata.firstDownload\r\n            ? 1\r\n            : metadata.downloadCount + 1;\r\n        metadata.firstDownload = metadata.firstDownload || new Date();\r\n        metadata.lastDownload = new Date();\r\n        await this.collection.updateOne({_id: this.mId}, {$set: {metadata}});\r\n        return this.stream;\r\n    }\r\n\r\n    async getImage(params: IAssetImageParams = null): Promise<Readable> {\r\n        return Asset.toImage(this.stream, this.metadata, params);\r\n    }\r\n\r\n    async downloadImage(params?: IAssetImageParams, metadata?: IAssetMeta): Promise<Readable> {\r\n        return Asset.toImage(await this.download(metadata), this.metadata, params);\r\n    }\r\n}\r\n"]}
154
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset.js","sourceRoot":"","sources":["../../../../src/services/entities/asset.ts"],"names":[],"mappings":"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","sourcesContent":["import sharp_, {Region} from \"sharp\";\r\nimport {Readable} from \"stream\";\r\nimport {Collection, GridFSBucket} from \"mongodb\";\r\nimport {ObjectId} from \"bson\";\r\n\r\nimport {IAsset, IAssetCropInfo, IAssetImageParams, IAssetMeta} from \"../../common-types\";\r\nimport {bufferToStream, deleteFromBucket, isBoolean, isInterface, isString, streamToBuffer} from \"../../utils\";\r\nimport {BaseEntity} from \"./base-entity\";\r\n\r\nconst sharp = sharp_;\r\nconst cropInterface = {\r\n    x: \"number\",\r\n    y: \"number\",\r\n    w: \"number\",\r\n    h: \"number\"\r\n};\r\n\r\nexport class Asset extends BaseEntity<IAsset> implements IAsset {\r\n\r\n    static toCropRegion(cropInfo: string | IAssetCropInfo): Region {\r\n        let crop = cropInfo as IAssetCropInfo;\r\n        if (isString(cropInfo)) {\r\n            try {\r\n                crop = JSON.parse(cropInfo as string);\r\n            } catch (e) {\r\n                return null;\r\n            }\r\n        }\r\n        if (!isInterface(crop, cropInterface)) return null;\r\n        return {\r\n            width: Math.round(crop.w),\r\n            height: Math.round(crop.h),\r\n            top: Math.round(crop.y),\r\n            left: Math.round(crop.x)\r\n        };\r\n    }\r\n\r\n    static async toImage(stream: Readable, meta?: IAssetMeta, params?: IAssetImageParams): Promise<Readable> {\r\n        params = params || {};\r\n\r\n        // Get default crop info\r\n        const crop = Asset.toCropRegion(meta.crop);\r\n\r\n        // Return the stream if there is no params and no default crop exists\r\n        if (meta?.extension === \"svg\" || (Object.keys(params).length == 0 && !crop)) {\r\n            return stream;\r\n        }\r\n\r\n        // Parse params\r\n        params.rotation = isNaN(params.rotation) ? 0 : Math.round(params.rotation / 90) * 90;\r\n        params.canvasScaleX = isNaN(params.canvasScaleX) ? 1 : Number(params.canvasScaleX);\r\n        params.canvasScaleY = isNaN(params.canvasScaleY) ? 1 : Number(params.canvasScaleY);\r\n        params.scaleX = isNaN(params.scaleX) ? 1 : Number(params.scaleX);\r\n        params.scaleY = isNaN(params.scaleY) ? 1 : Number(params.scaleY);\r\n        params.crop = isBoolean(params.crop) ? params.crop : params.crop == \"true\";\r\n\r\n        // Try to modify image\r\n        let buffer = await streamToBuffer(stream);\r\n        try {\r\n            // Get crop info\r\n            const cropBefore = Asset.toCropRegion(params.cropBefore || (params.crop ? meta.cropBefore : null));\r\n            const cropAfter = Asset.toCropRegion(params.cropAfter || (params.crop ? meta.cropAfter : null));\r\n            // Get metadata\r\n            let img = sharp(buffer);\r\n            let {width, height} = await img.metadata();\r\n            // Crop before resize\r\n            if (cropBefore) {\r\n                width = cropBefore.width;\r\n                height = cropBefore.height;\r\n                img = img.extract(cropBefore);\r\n            } else if (crop) {\r\n                width = crop.width;\r\n                height = crop.height;\r\n                img = img.extract(crop);\r\n            }\r\n            // Resize canvas\r\n            const canvasScaleX = meta?.canvasScaleX || 1;\r\n            const canvasScaleY = meta?.canvasScaleY || 1;\r\n            if (params.canvasScaleX !== canvasScaleX || params.canvasScaleY !== canvasScaleY) {\r\n                width = Math.round(width * params.canvasScaleX);\r\n                height = Math.round(height * params.canvasScaleY);\r\n                img = img.resize({width, height, background: \"#00000000\", fit: \"contain\"});\r\n            }\r\n            // Resize image\r\n            if (params.scaleX !== 1 || params.scaleY !== 1) {\r\n                width = Math.round(width * params.scaleX);\r\n                height = Math.round(height * params.scaleY);\r\n                img = img.resize({width, height, background: \"#00000000\", fit: \"fill\"});\r\n            }\r\n            // Crop after resize\r\n            if (cropAfter) {\r\n                img = img.extract(cropAfter);\r\n            }\r\n            // Rotate\r\n            if (params.rotation !== 0) {\r\n                buffer = await img.toBuffer();\r\n                img = sharp(buffer).rotate(params.rotation);\r\n            }\r\n            return bufferToStream(await img.toBuffer());\r\n        } catch (e) {\r\n            console.log(\"Asset image conversion error\", e);\r\n            return bufferToStream(buffer);\r\n        }\r\n    }\r\n\r\n    get filename(): string {\r\n        return this.data.filename;\r\n    }\r\n\r\n    get contentType(): string {\r\n        return this.data.contentType;\r\n    }\r\n\r\n    get metadata(): IAssetMeta {\r\n        return this.data.metadata;\r\n    }\r\n\r\n    get stream(): Readable {\r\n        return this.bucket.openDownloadStream(this.mId);\r\n    }\r\n\r\n    constructor(id: ObjectId,\r\n                data: Partial<IAsset>,\r\n                collection: Collection,\r\n                protected bucket: GridFSBucket) {\r\n        super(id, data, collection);\r\n    }\r\n\r\n    async unlink(): Promise<string> {\r\n        return deleteFromBucket(this.bucket, this.mId);\r\n    }\r\n\r\n    getBuffer(): Promise<Buffer> {\r\n        return streamToBuffer(this.stream);\r\n    }\r\n\r\n    async download(metadata?: IAssetMeta): Promise<Readable> {\r\n        metadata = Object.assign(this.metadata, metadata || {});\r\n        metadata.downloadCount = isNaN(metadata.downloadCount) || !metadata.firstDownload\r\n            ? 1\r\n            : metadata.downloadCount + 1;\r\n        metadata.firstDownload = metadata.firstDownload || new Date();\r\n        metadata.lastDownload = new Date();\r\n        await this.collection.updateOne({_id: this.mId}, {$set: {metadata}});\r\n        return this.stream;\r\n    }\r\n\r\n    async getImage(params: IAssetImageParams = null): Promise<Readable> {\r\n        return Asset.toImage(this.stream, this.metadata, params);\r\n    }\r\n\r\n    async downloadImage(params?: IAssetImageParams, metadata?: IAssetMeta): Promise<Readable> {\r\n        return Asset.toImage(await this.download(metadata), this.metadata, params);\r\n    }\r\n}\r\n"]}
@@ -0,0 +1,65 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import { ObjectId } from "bson";
11
+ import { bufferToStream } from "../../utils";
12
+ import { Asset } from "./asset";
13
+ export class TempAsset {
14
+ constructor(buffer, filename, contentType, metadata) {
15
+ this.buffer = buffer;
16
+ this.filename = filename;
17
+ this.contentType = contentType;
18
+ this.metadata = metadata;
19
+ this.id = new ObjectId().toHexString();
20
+ }
21
+ get stream() {
22
+ return bufferToStream(this.buffer);
23
+ }
24
+ unlink() {
25
+ return __awaiter(this, void 0, void 0, function* () {
26
+ throw new Error(`Temp asset '${this.id}' can not be removed!`);
27
+ });
28
+ }
29
+ getBuffer() {
30
+ return __awaiter(this, void 0, void 0, function* () {
31
+ return this.buffer;
32
+ });
33
+ }
34
+ download(metadata) {
35
+ return __awaiter(this, void 0, void 0, function* () {
36
+ return this.stream;
37
+ });
38
+ }
39
+ downloadImage(params, metadata) {
40
+ Object.assign(this.metadata, metadata || {});
41
+ return Asset.toImage(this.stream, this.metadata, params);
42
+ }
43
+ getImage(params) {
44
+ return this.downloadImage(params);
45
+ }
46
+ save() {
47
+ return __awaiter(this, void 0, void 0, function* () {
48
+ return this;
49
+ });
50
+ }
51
+ load() {
52
+ return __awaiter(this, void 0, void 0, function* () {
53
+ return this;
54
+ });
55
+ }
56
+ toJSON() {
57
+ return {
58
+ id: this.id,
59
+ filename: this.filename,
60
+ contentType: this.contentType,
61
+ metadata: this.metadata
62
+ };
63
+ }
64
+ }
65
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVtcC1hc3NldC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9zZXJ2aWNlcy9lbnRpdGllcy90ZW1wLWFzc2V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBLE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFHOUIsT0FBTyxFQUFDLGNBQWMsRUFBQyxNQUFNLGFBQWEsQ0FBQztBQUMzQyxPQUFPLEVBQUMsS0FBSyxFQUFDLE1BQU0sU0FBUyxDQUFDO0FBRTlCLE1BQU0sT0FBTyxTQUFTO0lBUWxCLFlBQXNCLE1BQWMsRUFBVyxRQUFnQixFQUFXLFdBQW1CLEVBQVcsUUFBb0I7UUFBdEcsV0FBTSxHQUFOLE1BQU0sQ0FBUTtRQUFXLGFBQVEsR0FBUixRQUFRLENBQVE7UUFBVyxnQkFBVyxHQUFYLFdBQVcsQ0FBUTtRQUFXLGFBQVEsR0FBUixRQUFRLENBQVk7UUFDeEgsSUFBSSxDQUFDLEVBQUUsR0FBRyxJQUFJLFFBQVEsRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQzNDLENBQUM7SUFORCxJQUFJLE1BQU07UUFDTixPQUFPLGNBQWMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQU1LLE1BQU07O1lBQ1IsTUFBTSxJQUFJLEtBQUssQ0FBQyxlQUFlLElBQUksQ0FBQyxFQUFFLHVCQUF1QixDQUFDLENBQUM7UUFDbkUsQ0FBQztLQUFBO0lBRUssU0FBUzs7WUFDWCxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUM7UUFDdkIsQ0FBQztLQUFBO0lBRUssUUFBUSxDQUFDLFFBQXFCOztZQUNoQyxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUM7UUFDdkIsQ0FBQztLQUFBO0lBRUQsYUFBYSxDQUFDLE1BQTBCLEVBQUUsUUFBcUI7UUFDM0QsTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLFFBQVEsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUM3QyxPQUFPLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQzdELENBQUM7SUFFRCxRQUFRLENBQUMsTUFBMEI7UUFDL0IsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3RDLENBQUM7SUFFSyxJQUFJOztZQUNOLE9BQU8sSUFBSSxDQUFDO1FBQ2hCLENBQUM7S0FBQTtJQUVLLElBQUk7O1lBQ04sT0FBTyxJQUFJLENBQUM7UUFDaEIsQ0FBQztLQUFBO0lBRUQsTUFBTTtRQUNGLE9BQU87WUFDSCxFQUFFLEVBQUUsSUFBSSxDQUFDLEVBQUU7WUFDWCxRQUFRLEVBQUUsSUFBSSxDQUFDLFFBQVE7WUFDdkIsV0FBVyxFQUFFLElBQUksQ0FBQyxXQUFXO1lBQzdCLFFBQVEsRUFBRSxJQUFJLENBQUMsUUFBUTtTQUMxQixDQUFDO0lBQ04sQ0FBQztDQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtSZWFkYWJsZX0gZnJvbSBcInN0cmVhbVwiO1xyXG5pbXBvcnQge09iamVjdElkfSBmcm9tIFwiYnNvblwiO1xyXG5pbXBvcnQgQnVmZmVyIGZyb20gXCJidWZmZXJcIjtcclxuaW1wb3J0IHtJQXNzZXQsIElBc3NldEltYWdlUGFyYW1zLCBJQXNzZXRNZXRhfSBmcm9tIFwiLi4vLi4vY29tbW9uLXR5cGVzXCI7XHJcbmltcG9ydCB7YnVmZmVyVG9TdHJlYW19IGZyb20gXCIuLi8uLi91dGlsc1wiO1xyXG5pbXBvcnQge0Fzc2V0fSBmcm9tIFwiLi9hc3NldFwiO1xyXG5cclxuZXhwb3J0IGNsYXNzIFRlbXBBc3NldCBpbXBsZW1lbnRzIElBc3NldCB7XHJcblxyXG4gICAgcmVhZG9ubHkgaWQ6IHN0cmluZztcclxuXHJcbiAgICBnZXQgc3RyZWFtKCk6IFJlYWRhYmxlIHtcclxuICAgICAgICByZXR1cm4gYnVmZmVyVG9TdHJlYW0odGhpcy5idWZmZXIpO1xyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0cnVjdG9yKHByb3RlY3RlZCBidWZmZXI6IEJ1ZmZlciwgcmVhZG9ubHkgZmlsZW5hbWU6IHN0cmluZywgcmVhZG9ubHkgY29udGVudFR5cGU6IHN0cmluZywgcmVhZG9ubHkgbWV0YWRhdGE6IElBc3NldE1ldGEpIHtcclxuICAgICAgICB0aGlzLmlkID0gbmV3IE9iamVjdElkKCkudG9IZXhTdHJpbmcoKTtcclxuICAgIH1cclxuXHJcbiAgICBhc3luYyB1bmxpbmsoKTogUHJvbWlzZTxzdHJpbmc+IHtcclxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFRlbXAgYXNzZXQgJyR7dGhpcy5pZH0nIGNhbiBub3QgYmUgcmVtb3ZlZCFgKTtcclxuICAgIH1cclxuXHJcbiAgICBhc3luYyBnZXRCdWZmZXIoKTogUHJvbWlzZTxCdWZmZXI+IHtcclxuICAgICAgICByZXR1cm4gdGhpcy5idWZmZXI7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgZG93bmxvYWQobWV0YWRhdGE/OiBJQXNzZXRNZXRhKTogUHJvbWlzZTxSZWFkYWJsZT4ge1xyXG4gICAgICAgIHJldHVybiB0aGlzLnN0cmVhbTtcclxuICAgIH1cclxuXHJcbiAgICBkb3dubG9hZEltYWdlKHBhcmFtcz86IElBc3NldEltYWdlUGFyYW1zLCBtZXRhZGF0YT86IElBc3NldE1ldGEpOiBQcm9taXNlPFJlYWRhYmxlPiB7XHJcbiAgICAgICAgT2JqZWN0LmFzc2lnbih0aGlzLm1ldGFkYXRhLCBtZXRhZGF0YSB8fCB7fSk7XHJcbiAgICAgICAgcmV0dXJuIEFzc2V0LnRvSW1hZ2UodGhpcy5zdHJlYW0sIHRoaXMubWV0YWRhdGEsIHBhcmFtcyk7XHJcbiAgICB9XHJcblxyXG4gICAgZ2V0SW1hZ2UocGFyYW1zPzogSUFzc2V0SW1hZ2VQYXJhbXMpOiBQcm9taXNlPFJlYWRhYmxlPiB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZG93bmxvYWRJbWFnZShwYXJhbXMpO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHNhdmUoKTogUHJvbWlzZTxhbnk+IHtcclxuICAgICAgICByZXR1cm4gdGhpcztcclxuICAgIH1cclxuXHJcbiAgICBhc3luYyBsb2FkKCk6IFByb21pc2U8dGhpcz4ge1xyXG4gICAgICAgIHJldHVybiB0aGlzO1xyXG4gICAgfVxyXG5cclxuICAgIHRvSlNPTigpOiBhbnkge1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGlkOiB0aGlzLmlkLFxyXG4gICAgICAgICAgICBmaWxlbmFtZTogdGhpcy5maWxlbmFtZSxcclxuICAgICAgICAgICAgY29udGVudFR5cGU6IHRoaXMuY29udGVudFR5cGUsXHJcbiAgICAgICAgICAgIG1ldGFkYXRhOiB0aGlzLm1ldGFkYXRhXHJcbiAgICAgICAgfTtcclxuICAgIH1cclxufVxyXG4iXX0=
@@ -20,21 +20,21 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
20
20
  });
21
21
  };
22
22
  import { inject, injectable, injectAll, Lifecycle, scoped } from "tsyringe";
23
- import { Queue, Scheduler, Worker } from "node-resque";
24
23
  import { schedule, validate } from "node-cron";
25
- import ioredis from "ioredis";
24
+ import { socket } from "zeromq";
25
+ import { ObjectId } from "bson";
26
26
  import { DI_CONTAINER, JOB } from "../common-types";
27
- import { getConstructorName, isArray, isObject } from "../utils";
27
+ import { getConstructorName, isArray, isObject, jsonHighlight, promiseTimeout } from "../utils";
28
28
  import { Configuration } from "./configuration";
29
- const IORedis = ioredis;
30
29
  let JobManager = class JobManager {
31
30
  constructor(config, container, jobTypes) {
32
31
  this.config = config;
33
32
  this.container = container;
34
33
  this.jobTypes = jobTypes || [];
35
34
  this.jobs = this.jobTypes.reduce((res, jobType) => {
36
- res[getConstructorName(jobType)] = {
37
- perform: this.toPerformFunction(jobType)
35
+ res[getConstructorName(jobType)] = (jobParams) => {
36
+ const job = this.resolveJobInstance(jobType, jobParams);
37
+ return job.process();
38
38
  };
39
39
  return res;
40
40
  }, {});
@@ -52,31 +52,25 @@ let JobManager = class JobManager {
52
52
  return instance.process();
53
53
  });
54
54
  }
55
- enqueueWithName(name, params = {}, que = "main") {
55
+ enqueueWithName(name, params = {}) {
56
56
  return __awaiter(this, void 0, void 0, function* () {
57
57
  const jobName = yield this.tryResolveFromName(name, params);
58
- yield this.queue.enqueue(que, jobName, [params]);
58
+ return this.sendToWorkers(jobName, params);
59
59
  });
60
60
  }
61
- enqueue(jobType, params = {}, que = "main") {
61
+ enqueue(jobType, params = {}) {
62
62
  return __awaiter(this, void 0, void 0, function* () {
63
63
  const jobName = yield this.tryResolveAndConnect(jobType, params);
64
- yield this.queue.enqueue(que, jobName, [params]);
64
+ return this.sendToWorkers(jobName, params);
65
65
  });
66
66
  }
67
- enqueueAt(timestamp, jobType, params = {}, que = "main") {
67
+ sendToWorkers(jobName, params) {
68
68
  return __awaiter(this, void 0, void 0, function* () {
69
- const jobName = yield this.tryResolveAndConnect(jobType, params);
70
- yield this.queue.enqueueAt(timestamp, que, jobName, [params]);
71
- });
72
- }
73
- enqueueIn(time, jobType, params = {}, que = "main") {
74
- return __awaiter(this, void 0, void 0, function* () {
75
- const jobName = yield this.tryResolveAndConnect(jobType, params);
76
- yield this.queue.enqueueIn(time, que, jobName, [params]);
69
+ const publisher = yield this.scheduler;
70
+ yield publisher.send([jobName, JSON.stringify(params), new ObjectId().toHexString()]);
77
71
  });
78
72
  }
79
- schedule(minute, hour, dayOfMonth, month, dayOfWeek, jobType, params = {}, que = "main") {
73
+ schedule(minute, hour, dayOfMonth, month, dayOfWeek, jobType, params = {}) {
80
74
  const expression = [minute, hour, dayOfMonth, month, dayOfWeek].map(t => {
81
75
  if (isObject(t)) {
82
76
  const range = t;
@@ -93,19 +87,38 @@ let JobManager = class JobManager {
93
87
  return null;
94
88
  }
95
89
  return schedule(expression, () => {
96
- this.enqueue(jobType, params, que).catch(e => {
90
+ this.enqueue(jobType, params).catch(e => {
97
91
  console.log(`Can't enqueue job: '${jobName}' because: ${e}`);
98
92
  });
99
93
  });
100
94
  }
101
95
  startProcessing() {
102
- return __awaiter(this, void 0, void 0, function* () {
103
- this.initialize();
104
- yield this.worker.connect();
105
- yield this.worker.start();
106
- yield this.scheduler.connect();
107
- yield this.scheduler.start();
108
- });
96
+ const host = this.config.resolve("zmqRemoteHost");
97
+ this.worker = socket("pull");
98
+ this.worker.connect(host);
99
+ this.worker.on("message", (name, args, uniqueId) => __awaiter(this, void 0, void 0, function* () {
100
+ try {
101
+ const jobName = name.toString("utf8");
102
+ const jobParams = JSON.parse(args.toString("utf8"));
103
+ const timerId = uniqueId === null || uniqueId === void 0 ? void 0 : uniqueId.toString("utf8");
104
+ const jobNameLog = `\x1b[36m"${jobName}"\x1b[0m`;
105
+ const jobArgsLog = `\n${jsonHighlight(jobParams)}\n`;
106
+ console.time(timerId);
107
+ console.timeLog(timerId, `Started working on background job: ${jobNameLog} with args: ${jobArgsLog}`);
108
+ try {
109
+ yield Promise.race([this.jobs[jobName](jobParams), promiseTimeout(15000, true)]);
110
+ console.timeLog(timerId, `Finished working on background job: ${jobNameLog} with args: ${jobArgsLog}`);
111
+ }
112
+ catch (e) {
113
+ console.timeLog(timerId, `Background job failed: ${jobNameLog} with args: ${jobArgsLog}${e.message}\n\n`);
114
+ }
115
+ console.timeEnd(timerId);
116
+ }
117
+ catch (e) {
118
+ console.log(`Failed to start job: ${e.message}`);
119
+ }
120
+ }));
121
+ console.log(`Waiting for jobs at: ${host}`);
109
122
  }
110
123
  tryResolve(jobType, params) {
111
124
  const jobName = getConstructorName(jobType);
@@ -120,47 +133,6 @@ let JobManager = class JobManager {
120
133
  }
121
134
  return jobName;
122
135
  }
123
- initialize() {
124
- if (this.queue)
125
- return;
126
- const config = this.config;
127
- const options = { password: config.resolve("redisPassword") };
128
- const sentinels = config.resolve("redisSentinels");
129
- const redis = !sentinels
130
- ? null
131
- : new IORedis({
132
- sentinels,
133
- name: config.resolve("redisCluster"),
134
- });
135
- const connection = {
136
- pkg: "ioredis",
137
- host: config.resolve("redisHost"),
138
- password: options.password,
139
- port: config.resolve("redisPort"),
140
- namespace: config.resolve("redisNamespace"),
141
- redis,
142
- options
143
- };
144
- const queues = config.resolve("workQueues");
145
- this.queue = new Queue({ connection }, this.jobs);
146
- this.worker = new Worker({ connection, queues }, this.jobs);
147
- this.worker.on("job", (queue, job) => {
148
- console.log(`working job ${queue} ${JSON.stringify(job)}`);
149
- });
150
- this.worker.on("reEnqueue", (queue, job, plugin) => {
151
- console.log(`reEnqueue job (${plugin}) ${queue} ${JSON.stringify(job)}`);
152
- });
153
- this.worker.on("success", (queue, job, result, duration) => {
154
- console.log(`job success ${queue} ${JSON.stringify(job)} >> ${result} (${duration}ms)`);
155
- });
156
- this.worker.on("failure", (queue, job, failure, duration) => {
157
- console.log(`job failure ${queue} ${JSON.stringify(job)} >> ${failure} (${duration}ms)`);
158
- });
159
- this.worker.on("error", (error, queue, job) => {
160
- console.log(`error ${queue} ${JSON.stringify(job)} >> ${error}`);
161
- });
162
- this.scheduler = new Scheduler({ connection }, this.jobs);
163
- }
164
136
  tryResolveFromName(jobName, params) {
165
137
  const jobType = this.jobTypes.find(type => {
166
138
  return getConstructorName(type) == jobName;
@@ -172,10 +144,14 @@ let JobManager = class JobManager {
172
144
  }
173
145
  tryResolveAndConnect(jobType, params) {
174
146
  return __awaiter(this, void 0, void 0, function* () {
175
- this.initialize();
176
- const jobName = this.tryResolve(jobType, params);
177
- yield this.queue.connect();
178
- return jobName;
147
+ this.scheduler = this.scheduler || new Promise((resolve) => __awaiter(this, void 0, void 0, function* () {
148
+ const port = this.config.resolve("zmqPort");
149
+ const publisher = socket("push");
150
+ yield publisher.bind(`tcp://0.0.0.0:${port}`);
151
+ console.log(`Publisher bound to port: ${port}`);
152
+ resolve(publisher);
153
+ }));
154
+ return this.tryResolve(jobType, params);
179
155
  });
180
156
  }
181
157
  resolveJobInstance(jobType, params) {
@@ -186,12 +162,6 @@ let JobManager = class JobManager {
186
162
  container.register(jobType, jobType);
187
163
  return container.resolve(jobType);
188
164
  }
189
- toPerformFunction(jobType) {
190
- return (jobParams) => {
191
- const job = this.resolveJobInstance(jobType, jobParams);
192
- return job.process();
193
- };
194
- }
195
165
  };
196
166
  JobManager = __decorate([
197
167
  injectable(),
@@ -200,4 +170,4 @@ JobManager = __decorate([
200
170
  __metadata("design:paramtypes", [Configuration, Object, Array])
201
171
  ], JobManager);
202
172
  export { JobManager };
203
- //# 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 {Queue, Scheduler, Worker} from \"node-resque\";\r\nimport {schedule, validate} from \"node-cron\";\r\nimport ioredis from \"ioredis\";\r\nimport {DI_CONTAINER, IJob, IJobTask, JOB, JobParams, JobScheduleRange, JobScheduleTime, Type} from \"../common-types\";\r\nimport {getConstructorName, isArray, isObject} from \"../utils\";\r\nimport {Configuration} from \"./configuration\";\r\n\r\nconst IORedis = ioredis;\r\n\r\n@injectable()\r\n@scoped(Lifecycle.ContainerScoped)\r\nexport class JobManager {\r\n\r\n    protected jobs: any;\r\n    protected queue: Queue;\r\n    protected worker: Worker;\r\n    protected scheduler: Scheduler;\r\n    protected jobTypes: Type<IJob>[];\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)] = {\r\n                perform: this.toPerformFunction(jobType)\r\n            };\r\n            return res;\r\n        }, {});\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 = {}, que: string = \"main\"): Promise<any> {\r\n        const jobName = await this.tryResolveFromName(name, params);\r\n        await this.queue.enqueue(que, jobName, [params]);\r\n    }\r\n\r\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}, que: string = \"main\"): Promise<any> {\r\n        const jobName = await this.tryResolveAndConnect(jobType, params);\r\n        await this.queue.enqueue(que, jobName, [params]);\r\n    }\r\n\r\n    async enqueueAt(timestamp: number, jobType: Type<IJob>, params: JobParams = {}, que: string = \"main\"): Promise<any> {\r\n        const jobName = await this.tryResolveAndConnect(jobType, params);\r\n        await this.queue.enqueueAt(timestamp, que, jobName, [params]);\r\n    }\r\n\r\n    async enqueueIn(time: number, jobType: Type<IJob>, params: JobParams = {}, que: string = \"main\"): Promise<any> {\r\n        const jobName = await this.tryResolveAndConnect(jobType, params);\r\n        await this.queue.enqueueIn(time, que, jobName, [params]);\r\n    }\r\n\r\n    schedule(minute: JobScheduleTime, hour: JobScheduleTime, dayOfMonth: JobScheduleTime, month: JobScheduleTime, dayOfWeek: JobScheduleTime, jobType: Type<IJob>, params: JobParams = {}, que: string = \"main\"): 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, que).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        this.initialize();\r\n        await this.worker.connect();\r\n        await this.worker.start();\r\n        await this.scheduler.connect();\r\n        await this.scheduler.start();\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 initialize(): void {\r\n        if (this.queue) return;\r\n        const config = this.config;\r\n        const options = {password: config.resolve(\"redisPassword\")};\r\n        const sentinels: Array<{host: string, port: number}> = config.resolve(\"redisSentinels\");\r\n        const redis = !sentinels\r\n            ? null\r\n            : new IORedis({\r\n                sentinels,\r\n                name: config.resolve(\"redisCluster\"),\r\n            });\r\n        const connection = {\r\n            pkg: \"ioredis\",\r\n            host: config.resolve(\"redisHost\"),\r\n            password: options.password,\r\n            port: config.resolve(\"redisPort\"),\r\n            namespace: config.resolve(\"redisNamespace\"),\r\n            redis,\r\n            options\r\n        };\r\n        const queues = config.resolve(\"workQueues\");\r\n        this.queue = new Queue({connection}, this.jobs);\r\n        this.worker = new Worker({connection, queues}, this.jobs);\r\n        this.worker.on(\"job\", (queue, job) => {\r\n            console.log(`working job ${queue} ${JSON.stringify(job)}`);\r\n        });\r\n        this.worker.on(\"reEnqueue\", (queue, job, plugin) => {\r\n            console.log(`reEnqueue job (${plugin}) ${queue} ${JSON.stringify(job)}`);\r\n        });\r\n        this.worker.on(\"success\", (queue, job, result, duration) => {\r\n            console.log(\r\n                `job success ${queue} ${JSON.stringify(job)} >> ${result} (${duration}ms)`\r\n            );\r\n        });\r\n        this.worker.on(\"failure\", (queue, job, failure, duration) => {\r\n            console.log(\r\n                `job failure ${queue} ${JSON.stringify(\r\n                    job\r\n                )} >> ${failure} (${duration}ms)`\r\n            );\r\n        });\r\n        this.worker.on(\"error\", (error, queue, job) => {\r\n            console.log(`error ${queue} ${JSON.stringify(job)}  >> ${error}`);\r\n        });\r\n        this.scheduler = new Scheduler({connection}, this.jobs);\r\n    }\r\n\r\n    protected tryResolveFromName(jobName: string, params: JobParams): Promise<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.tryResolveAndConnect(jobType, params);\r\n    }\r\n\r\n    protected async tryResolveAndConnect(jobType: Type<IJob>, params: JobParams): Promise<string> {\r\n        this.initialize();\r\n        const jobName = this.tryResolve(jobType, params);\r\n        await this.queue.connect();\r\n        return jobName;\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\r\n        return container.resolve(jobType) as IJob;\r\n    }\r\n\r\n    protected toPerformFunction(jobType: Type<IJob>): Function {\r\n        return (jobParams: JobParams) => {\r\n            const job = this.resolveJobInstance(jobType, jobParams);\r\n            return job.process();\r\n        }\r\n    }\r\n}\r\n"]}
173
+ //# 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 {ObjectId} from \"bson\";\r\nimport {DI_CONTAINER, IJob, IJobTask, JOB, JobParams, JobScheduleRange, JobScheduleTime, Type} from \"../common-types\";\r\nimport {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 jobs: {[name: string]: (jobParams: JobParams) => Promise<any>};\r\n    protected scheduler: Promise<Socket>;\r\n    protected worker: Socket;\r\n    protected jobTypes: Type<IJob>[];\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();\r\n            }\r\n            return res;\r\n        }, {});\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        const jobName = await this.tryResolveFromName(name, params);\r\n        return this.sendToWorkers(jobName, params);\r\n    }\r\n\r\n    async enqueue(jobType: Type<IJob>, params: JobParams = {}): Promise<any> {\r\n        const jobName = await this.tryResolveAndConnect(jobType, params);\r\n        return this.sendToWorkers(jobName, params);\r\n    }\r\n\r\n    protected async sendToWorkers(jobName: string, params: JobParams): Promise<any> {\r\n        const publisher = await this.scheduler;\r\n        await publisher.send([jobName, JSON.stringify(params), new ObjectId().toHexString()]);\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    startProcessing(): void {\r\n        const host = this.config.resolve(\"zmqRemoteHost\");\r\n        this.worker = socket(\"pull\");\r\n        this.worker.connect(host);\r\n        this.worker.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                const jobNameLog = `\\x1b[36m\"${jobName}\"\\x1b[0m`;\r\n                const jobArgsLog = `\\n${jsonHighlight(jobParams)}\\n`;\r\n\r\n                console.time(timerId);\r\n                console.timeLog(timerId, `Started working on background job: ${jobNameLog} with args: ${jobArgsLog}`);\r\n                try {\r\n                    await Promise.race([this.jobs[jobName](jobParams), promiseTimeout(15000, true)]);\r\n                    console.timeLog(timerId, `Finished working on background job: ${jobNameLog} with args: ${jobArgsLog}`);\r\n                } catch (e) {\r\n                    console.timeLog(timerId, `Background job failed: ${jobNameLog} with args: ${jobArgsLog}${e.message}\\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        console.log(`Waiting for jobs at: ${host}`);\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): Promise<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.tryResolveAndConnect(jobType, params);\r\n    }\r\n\r\n    protected async tryResolveAndConnect(jobType: Type<IJob>, params: JobParams): Promise<string> {\r\n        this.scheduler = this.scheduler || new Promise<any>(async resolve => {\r\n            const port = this.config.resolve(\"zmqPort\");\r\n            const publisher = socket(\"push\");\r\n            await publisher.bind(`tcp://0.0.0.0:${port}`)\r\n            console.log(`Publisher bound to port: ${port}`);\r\n            resolve(publisher);\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"]}