@stemy/backend 2.9.7 → 3.0.1

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":";;;;;;;;;AAAA,OAAO,MAAgB,MAAM,OAAO,CAAC;AAMrC,OAAO,EAAC,cAAc,EAAE,gBAAgB,EAAE,SAAS,EAAE,WAAW,EAAE,QAAQ,EAAE,cAAc,EAAC,MAAM,aAAa,CAAC;AAC/G,OAAO,EAAC,UAAU,EAAC,MAAM,eAAe,CAAC;AAEzC,MAAM,KAAK,GAAG,MAAM,CAAC;AACrB,MAAM,aAAa,GAAG;IAClB,CAAC,EAAE,QAAQ;IACX,CAAC,EAAE,QAAQ;IACX,CAAC,EAAE,QAAQ;IACX,CAAC,EAAE,QAAQ;CACd,CAAC;AAEF,MAAM,OAAO,KAAM,SAAQ,UAAkB;IAwGzC,YAAY,EAAY,EACZ,IAAqB,EACrB,UAAsB,EACZ,MAAoB;QACtC,KAAK,CAAC,EAAE,EAAE,IAAI,EAAE,UAAU,CAAC,CAAC;QADV,WAAM,GAAN,MAAM,CAAc;IAE1C,CAAC;IA3GD,MAAM,CAAC,YAAY,CAAC,QAAiC;QACjD,IAAI,IAAI,GAAG,QAA0B,CAAC;QACtC,IAAI,QAAQ,CAAC,QAAQ,CAAC,EAAE;YACpB,IAAI;gBACA,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,QAAkB,CAAC,CAAC;aACzC;YAAC,OAAO,CAAC,EAAE;gBACR,OAAO,IAAI,CAAC;aACf;SACJ;QACD,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,aAAa,CAAC;YAAE,OAAO,IAAI,CAAC;QACnD,OAAO;YACH,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;YACzB,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;YAC1B,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;YACvB,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;SAC3B,CAAC;IACN,CAAC;IAED,MAAM,CAAO,OAAO,CAAC,MAAgB,EAAE,IAAiB,EAAE,MAA0B;;YAChF,MAAM,GAAG,MAAM,IAAI,EAAE,CAAC;YAEtB,wBAAwB;YACxB,MAAM,IAAI,GAAG,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAE3C,qEAAqE;YACrE,IAAI,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,SAAS,MAAK,KAAK,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;gBACzE,OAAO,MAAM,CAAC;aACjB;YAED,eAAe;YACf,MAAM,CAAC,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;YACrF,MAAM,CAAC,YAAY,GAAG,KAAK,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;YACnF,MAAM,CAAC,YAAY,GAAG,KAAK,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;YACnF,MAAM,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACjE,MAAM,CAAC,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACjE,MAAM,CAAC,IAAI,GAAG,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC;YAE3E,sBAAsB;YACtB,IAAI,MAAM,GAAG,MAAM,cAAc,CAAC,MAAM,CAAC,CAAC;YAC1C,IAAI;gBACA,gBAAgB;gBAChB,MAAM,UAAU,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,UAAU,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;gBACnG,MAAM,SAAS,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,SAAS,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;gBAChG,eAAe;gBACf,IAAI,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC;gBACxB,IAAI,EAAC,KAAK,EAAE,MAAM,EAAC,GAAG,MAAM,GAAG,CAAC,QAAQ,EAAE,CAAC;gBAC3C,qBAAqB;gBACrB,IAAI,UAAU,EAAE;oBACZ,KAAK,GAAG,UAAU,CAAC,KAAK,CAAC;oBACzB,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC;oBAC3B,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;iBACjC;qBAAM,IAAI,IAAI,EAAE;oBACb,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;oBACnB,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;oBACrB,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;iBAC3B;gBACD,gBAAgB;gBAChB,MAAM,YAAY,GAAG,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,YAAY,KAAI,CAAC,CAAC;gBAC7C,MAAM,YAAY,GAAG,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,YAAY,KAAI,CAAC,CAAC;gBAC7C,IAAI,MAAM,CAAC,YAAY,KAAK,YAAY,IAAI,MAAM,CAAC,YAAY,KAAK,YAAY,EAAE;oBAC9E,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;oBAChD,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;oBAClD,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,EAAC,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,WAAW,EAAE,GAAG,EAAE,SAAS,EAAC,CAAC,CAAC;iBAC9E;gBACD,eAAe;gBACf,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE;oBAC5C,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;oBAC1C,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;oBAC5C,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,EAAC,KAAK,EAAE,MAAM,EAAE,UAAU,EAAE,WAAW,EAAE,GAAG,EAAE,MAAM,EAAC,CAAC,CAAC;iBAC3E;gBACD,oBAAoB;gBACpB,IAAI,SAAS,EAAE;oBACX,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;iBAChC;gBACD,SAAS;gBACT,IAAI,MAAM,CAAC,QAAQ,KAAK,CAAC,EAAE;oBACvB,MAAM,GAAG,MAAM,GAAG,CAAC,QAAQ,EAAE,CAAC;oBAC9B,GAAG,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;iBAC/C;gBACD,OAAO,cAAc,CAAC,MAAM,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;aAC/C;YAAC,OAAO,CAAC,EAAE;gBACR,OAAO,CAAC,GAAG,CAAC,8BAA8B,EAAE,CAAC,CAAC,CAAC;gBAC/C,OAAO,cAAc,CAAC,MAAM,CAAC,CAAC;aACjC;QACL,CAAC;KAAA;IAED,IAAI,QAAQ;QACR,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC;IAC9B,CAAC;IAED,IAAI,WAAW;QACX,OAAO,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;IACjC,CAAC;IAED,IAAI,QAAQ;QACR,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC;IAC9B,CAAC;IAED,IAAI,MAAM;QACN,OAAO,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACpD,CAAC;IASK,MAAM;;YACR,OAAO,gBAAgB,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QACnD,CAAC;KAAA;IAED,SAAS;QACL,OAAO,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACvC,CAAC;IAEK,QAAQ,CAAC,QAAqB;;YAChC,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,QAAQ,IAAI,EAAE,CAAC,CAAC;YACxD,QAAQ,CAAC,aAAa,GAAG,KAAK,CAAC,QAAQ,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa;gBAC7E,CAAC,CAAC,CAAC;gBACH,CAAC,CAAC,QAAQ,CAAC,aAAa,GAAG,CAAC,CAAC;YACjC,QAAQ,CAAC,aAAa,GAAG,QAAQ,CAAC,aAAa,IAAI,IAAI,IAAI,EAAE,CAAC;YAC9D,QAAQ,CAAC,YAAY,GAAG,IAAI,IAAI,EAAE,CAAC;YACnC,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,EAAC,GAAG,EAAE,IAAI,CAAC,GAAG,EAAC,EAAE,EAAC,IAAI,EAAE,EAAC,QAAQ,EAAC,EAAC,CAAC,CAAC;YACrE,OAAO,IAAI,CAAC,MAAM,CAAC;QACvB,CAAC;KAAA;IAEK,QAAQ,CAAC,SAA4B,IAAI;;YAC3C,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC7D,CAAC;KAAA;IAEK,aAAa,CAAC,MAA0B,EAAE,QAAqB;;YACjE,OAAO,KAAK,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC/E,CAAC;KAAA;CACJ","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"]}
@@ -48,7 +48,7 @@ export class LazyAsset extends BaseEntity {
48
48
  this.progresses.get(this.progressId).then(p => {
49
49
  p === null || p === void 0 ? void 0 : p.cancel();
50
50
  });
51
- this.startWorkingOnAsset().then(() => {
51
+ this.startWorkingOnAsset(false).then(() => {
52
52
  console.log(`Started working on lazy asset: ${this.id}`);
53
53
  }).catch(reason => {
54
54
  console.log(`Can't start working on lazy asset: ${this.id}\nReason: ${reason}`);
@@ -67,7 +67,7 @@ export class LazyAsset extends BaseEntity {
67
67
  yield this.progresses.waitToFinish(this.progressId);
68
68
  return this.loadAsset();
69
69
  }
70
- yield this.startWorkingOnAsset();
70
+ yield this.startWorkingOnAsset(true);
71
71
  return this.loadAsset();
72
72
  });
73
73
  }
@@ -78,13 +78,13 @@ export class LazyAsset extends BaseEntity {
78
78
  return asset;
79
79
  });
80
80
  }
81
- startWorkingOnAsset() {
81
+ startWorkingOnAsset(fromLoad) {
82
82
  return __awaiter(this, void 0, void 0, function* () {
83
83
  this.data.progressId = (yield this.progresses.create()).id;
84
84
  this.data.assetId = null;
85
85
  yield this.save();
86
- yield this.jobMan.enqueueWithName(this.data.jobName, Object.assign(Object.assign({}, this.data.jobParams), { lazyId: this.id }));
86
+ yield this.jobMan.enqueueWithName(this.data.jobName, Object.assign(Object.assign({}, this.data.jobParams), { lazyId: this.id, fromLoad }));
87
87
  });
88
88
  }
89
89
  }
90
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGF6eS1hc3NldC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9zZXJ2aWNlcy9lbnRpdGllcy9sYXp5LWFzc2V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBLE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFHOUIsT0FBTyxFQUFDLGdCQUFnQixFQUFDLE1BQU0sYUFBYSxDQUFDO0FBSTdDLE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFFekMsTUFBTSxPQUFPLFNBQVUsU0FBUSxVQUFzQjtJQXNCakQsWUFBWSxFQUFZLEVBQ1osSUFBeUIsRUFDekIsVUFBc0IsRUFDWixNQUFjLEVBQ2QsVUFBc0IsRUFDdEIsTUFBa0I7UUFDcEMsS0FBSyxDQUFDLEVBQUUsRUFBRSxJQUFJLEVBQUUsVUFBVSxDQUFDLENBQUM7UUFIVixXQUFNLEdBQU4sTUFBTSxDQUFRO1FBQ2QsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUN0QixXQUFNLEdBQU4sTUFBTSxDQUFZO0lBRXhDLENBQUM7SUEzQkQsSUFBSSxPQUFPO1FBQ1AsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztJQUM3QixDQUFDO0lBRUQsSUFBSSxTQUFTO1FBQ1QsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQztJQUMvQixDQUFDO0lBRUQsSUFBSSxNQUFNO1FBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUM1QixDQUFDO0lBRUQsSUFBSSxVQUFVO1FBQ1YsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQztJQUNoQyxDQUFDO0lBRUQsSUFBSSxPQUFPO1FBQ1AsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztJQUM3QixDQUFDO0lBV0ssTUFBTTs7WUFDUixNQUFNLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNsQixJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsRUFBRTtnQkFDbEIsTUFBTSxJQUFJLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxFQUFDLEdBQUcsRUFBRSxJQUFJLENBQUMsR0FBRyxFQUFDLENBQUMsQ0FBQzthQUNwRDtZQUNELE9BQU8sZ0JBQWdCLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsSUFBSSxRQUFRLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUM7UUFDNUUsQ0FBQztLQUFBO0lBRUQsWUFBWTtRQUNSLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFO1lBQ2xCLElBQUksSUFBSSxDQUFDLE9BQU87Z0JBQUUsT0FBTztZQUN6QixJQUFJLENBQUMsVUFBVSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFO2dCQUMxQyxDQUFDLGFBQUQsQ0FBQyx1QkFBRCxDQUFDLENBQUUsTUFBTSxHQUFHO1lBQ2hCLENBQUMsQ0FBQyxDQUFDO1lBQ0gsSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRTtnQkFDakMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxrQ0FBa0MsSUFBSSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDN0QsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxFQUFFO2dCQUNkLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0NBQXNDLElBQUksQ0FBQyxFQUFFLGFBQWEsTUFBTSxFQUFFLENBQUMsQ0FBQztZQUNwRixDQUFDLENBQUMsQ0FBQztRQUNQLENBQUMsQ0FBQyxDQUFDO0lBQ1AsQ0FBQztJQUVLLFNBQVM7O1lBQ1gsTUFBTSxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDbEIsSUFBSSxJQUFJLENBQUMsT0FBTztnQkFBRSxPQUFPLElBQUksQ0FBQztZQUM5QixJQUFJLElBQUksQ0FBQyxPQUFPLEVBQUU7Z0JBQ2QsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7YUFDekM7WUFDRCxJQUFJLElBQUksQ0FBQyxVQUFVLEVBQUU7Z0JBQ2pCLE1BQU0sSUFBSSxDQUFDLFVBQVUsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO2dCQUNwRCxPQUFPLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQzthQUMzQjtZQUNELE1BQU0sSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUM7WUFDakMsT0FBTyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDNUIsQ0FBQztLQUFBO0lBRUssVUFBVSxDQUFDLEtBQWE7O1lBQzFCLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxHQUFHLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDN0IsTUFBTSxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDbEIsT0FBTyxLQUFLLENBQUM7UUFDakIsQ0FBQztLQUFBO0lBRWUsbUJBQW1COztZQUMvQixJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsR0FBRyxDQUFDLE1BQU0sSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQztZQUMzRCxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sR0FBRyxJQUFJLENBQUM7WUFDekIsTUFBTSxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDbEIsTUFBTSxJQUFJLENBQUMsTUFBTSxDQUFDLGVBQWUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sa0NBQU0sSUFBSSxDQUFDLElBQUksQ0FBQyxTQUFTLEtBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxFQUFFLElBQUUsQ0FBQztRQUNwRyxDQUFDO0tBQUE7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Q29sbGVjdGlvbn0gZnJvbSBcIm1vbmdvZGJcIjtcclxuaW1wb3J0IHtPYmplY3RJZH0gZnJvbSBcImJzb25cIjtcclxuXHJcbmltcG9ydCB7SUFzc2V0LCBJTGF6eUFzc2V0fSBmcm9tIFwiLi4vLi4vY29tbW9uLXR5cGVzXCI7XHJcbmltcG9ydCB7ZGVsZXRlRnJvbUJ1Y2tldH0gZnJvbSBcIi4uLy4uL3V0aWxzXCI7XHJcbmltcG9ydCB7QXNzZXRzfSBmcm9tIFwiLi4vYXNzZXRzXCI7XHJcbmltcG9ydCB7Sm9iTWFuYWdlcn0gZnJvbSBcIi4uL2pvYi1tYW5hZ2VyXCI7XHJcbmltcG9ydCB7UHJvZ3Jlc3Nlc30gZnJvbSBcIi4uL3Byb2dyZXNzZXNcIjtcclxuaW1wb3J0IHtCYXNlRW50aXR5fSBmcm9tIFwiLi9iYXNlLWVudGl0eVwiO1xyXG5cclxuZXhwb3J0IGNsYXNzIExhenlBc3NldCBleHRlbmRzIEJhc2VFbnRpdHk8SUxhenlBc3NldD4gaW1wbGVtZW50cyBJTGF6eUFzc2V0IHtcclxuXHJcbiAgICBnZXQgam9iTmFtZSgpOiBzdHJpbmcge1xyXG4gICAgICAgIHJldHVybiB0aGlzLmRhdGEuam9iTmFtZTtcclxuICAgIH1cclxuXHJcbiAgICBnZXQgam9iUGFyYW1zKCk6IGFueSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZGF0YS5qb2JQYXJhbXM7XHJcbiAgICB9XHJcblxyXG4gICAgZ2V0IGpvYlF1ZSgpOiBzdHJpbmcge1xyXG4gICAgICAgIHJldHVybiB0aGlzLmRhdGEuam9iUXVlO1xyXG4gICAgfVxyXG5cclxuICAgIGdldCBwcm9ncmVzc0lkKCk6IHN0cmluZyB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZGF0YS5wcm9ncmVzc0lkO1xyXG4gICAgfVxyXG5cclxuICAgIGdldCBhc3NldElkKCk6IHN0cmluZyB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZGF0YS5hc3NldElkO1xyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0cnVjdG9yKGlkOiBPYmplY3RJZCxcclxuICAgICAgICAgICAgICAgIGRhdGE6IFBhcnRpYWw8SUxhenlBc3NldD4sXHJcbiAgICAgICAgICAgICAgICBjb2xsZWN0aW9uOiBDb2xsZWN0aW9uLFxyXG4gICAgICAgICAgICAgICAgcHJvdGVjdGVkIGFzc2V0czogQXNzZXRzLFxyXG4gICAgICAgICAgICAgICAgcHJvdGVjdGVkIHByb2dyZXNzZXM6IFByb2dyZXNzZXMsXHJcbiAgICAgICAgICAgICAgICBwcm90ZWN0ZWQgam9iTWFuOiBKb2JNYW5hZ2VyKSB7XHJcbiAgICAgICAgc3VwZXIoaWQsIGRhdGEsIGNvbGxlY3Rpb24pO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHVubGluaygpOiBQcm9taXNlPHN0cmluZz4ge1xyXG4gICAgICAgIGF3YWl0IHRoaXMubG9hZCgpO1xyXG4gICAgICAgIGlmICghdGhpcy5wcm9ncmVzc0lkKSB7XHJcbiAgICAgICAgICAgIGF3YWl0IHRoaXMuY29sbGVjdGlvbi5kZWxldGVPbmUoe19pZDogdGhpcy5tSWR9KTtcclxuICAgICAgICB9XHJcbiAgICAgICAgcmV0dXJuIGRlbGV0ZUZyb21CdWNrZXQodGhpcy5hc3NldHMuYnVja2V0LCBuZXcgT2JqZWN0SWQodGhpcy5hc3NldElkKSk7XHJcbiAgICB9XHJcblxyXG4gICAgc3RhcnRXb3JraW5nKCk6IHZvaWQge1xyXG4gICAgICAgIHRoaXMubG9hZCgpLnRoZW4oKCkgPT4ge1xyXG4gICAgICAgICAgICBpZiAodGhpcy5kZWxldGVkKSByZXR1cm47XHJcbiAgICAgICAgICAgIHRoaXMucHJvZ3Jlc3Nlcy5nZXQodGhpcy5wcm9ncmVzc0lkKS50aGVuKHAgPT4ge1xyXG4gICAgICAgICAgICAgICAgcD8uY2FuY2VsKCk7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgICAgICB0aGlzLnN0YXJ0V29ya2luZ09uQXNzZXQoKS50aGVuKCgpID0+IHtcclxuICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKGBTdGFydGVkIHdvcmtpbmcgb24gbGF6eSBhc3NldDogJHt0aGlzLmlkfWApO1xyXG4gICAgICAgICAgICB9KS5jYXRjaChyZWFzb24gPT4ge1xyXG4gICAgICAgICAgICAgICAgY29uc29sZS5sb2coYENhbid0IHN0YXJ0IHdvcmtpbmcgb24gbGF6eSBhc3NldDogJHt0aGlzLmlkfVxcblJlYXNvbjogJHtyZWFzb259YCk7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIGxvYWRBc3NldCgpOiBQcm9taXNlPElBc3NldD4ge1xyXG4gICAgICAgIGF3YWl0IHRoaXMubG9hZCgpO1xyXG4gICAgICAgIGlmICh0aGlzLmRlbGV0ZWQpIHJldHVybiBudWxsO1xyXG4gICAgICAgIGlmICh0aGlzLmFzc2V0SWQpIHtcclxuICAgICAgICAgICAgcmV0dXJuIHRoaXMuYXNzZXRzLnJlYWQodGhpcy5hc3NldElkKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgaWYgKHRoaXMucHJvZ3Jlc3NJZCkge1xyXG4gICAgICAgICAgICBhd2FpdCB0aGlzLnByb2dyZXNzZXMud2FpdFRvRmluaXNoKHRoaXMucHJvZ3Jlc3NJZCk7XHJcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmxvYWRBc3NldCgpO1xyXG4gICAgICAgIH1cclxuICAgICAgICBhd2FpdCB0aGlzLnN0YXJ0V29ya2luZ09uQXNzZXQoKTtcclxuICAgICAgICByZXR1cm4gdGhpcy5sb2FkQXNzZXQoKTtcclxuICAgIH1cclxuXHJcbiAgICBhc3luYyB3cml0ZUFzc2V0KGFzc2V0OiBJQXNzZXQpOiBQcm9taXNlPElBc3NldD4ge1xyXG4gICAgICAgIHRoaXMuZGF0YS5hc3NldElkID0gYXNzZXQuaWQ7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5zYXZlKCk7XHJcbiAgICAgICAgcmV0dXJuIGFzc2V0O1xyXG4gICAgfVxyXG5cclxuICAgIHByb3RlY3RlZCBhc3luYyBzdGFydFdvcmtpbmdPbkFzc2V0KCk6IFByb21pc2U8YW55PiB7XHJcbiAgICAgICAgdGhpcy5kYXRhLnByb2dyZXNzSWQgPSAoYXdhaXQgdGhpcy5wcm9ncmVzc2VzLmNyZWF0ZSgpKS5pZDtcclxuICAgICAgICB0aGlzLmRhdGEuYXNzZXRJZCA9IG51bGw7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5zYXZlKCk7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5qb2JNYW4uZW5xdWV1ZVdpdGhOYW1lKHRoaXMuZGF0YS5qb2JOYW1lLCB7Li4udGhpcy5kYXRhLmpvYlBhcmFtcywgbGF6eUlkOiB0aGlzLmlkfSk7XHJcbiAgICB9XHJcbn1cclxuIl19
90
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGF6eS1hc3NldC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9zZXJ2aWNlcy9lbnRpdGllcy9sYXp5LWFzc2V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUNBLE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFHOUIsT0FBTyxFQUFDLGdCQUFnQixFQUFDLE1BQU0sYUFBYSxDQUFDO0FBSTdDLE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFFekMsTUFBTSxPQUFPLFNBQVUsU0FBUSxVQUFzQjtJQXNCakQsWUFBWSxFQUFZLEVBQ1osSUFBeUIsRUFDekIsVUFBc0IsRUFDWixNQUFjLEVBQ2QsVUFBc0IsRUFDdEIsTUFBa0I7UUFDcEMsS0FBSyxDQUFDLEVBQUUsRUFBRSxJQUFJLEVBQUUsVUFBVSxDQUFDLENBQUM7UUFIVixXQUFNLEdBQU4sTUFBTSxDQUFRO1FBQ2QsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUN0QixXQUFNLEdBQU4sTUFBTSxDQUFZO0lBRXhDLENBQUM7SUEzQkQsSUFBSSxPQUFPO1FBQ1AsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztJQUM3QixDQUFDO0lBRUQsSUFBSSxTQUFTO1FBQ1QsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQztJQUMvQixDQUFDO0lBRUQsSUFBSSxNQUFNO1FBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUM1QixDQUFDO0lBRUQsSUFBSSxVQUFVO1FBQ1YsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQztJQUNoQyxDQUFDO0lBRUQsSUFBSSxPQUFPO1FBQ1AsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztJQUM3QixDQUFDO0lBV0ssTUFBTTs7WUFDUixNQUFNLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztZQUNsQixJQUFJ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cnQge09iamVjdElkfSBmcm9tIFwiYnNvblwiO1xyXG5cclxuaW1wb3J0IHtJQXNzZXQsIElMYXp5QXNzZXR9IGZyb20gXCIuLi8uLi9jb21tb24tdHlwZXNcIjtcclxuaW1wb3J0IHtkZWxldGVGcm9tQnVja2V0fSBmcm9tIFwiLi4vLi4vdXRpbHNcIjtcclxuaW1wb3J0IHtBc3NldHN9IGZyb20gXCIuLi9hc3NldHNcIjtcclxuaW1wb3J0IHtKb2JNYW5hZ2VyfSBmcm9tIFwiLi4vam9iLW1hbmFnZXJcIjtcclxuaW1wb3J0IHtQcm9ncmVzc2VzfSBmcm9tIFwiLi4vcHJvZ3Jlc3Nlc1wiO1xyXG5pbXBvcnQge0Jhc2VFbnRpdHl9IGZyb20gXCIuL2Jhc2UtZW50aXR5XCI7XHJcblxyXG5leHBvcnQgY2xhc3MgTGF6eUFzc2V0IGV4dGVuZHMgQmFzZUVudGl0eTxJTGF6eUFzc2V0PiBpbXBsZW1lbnRzIElMYXp5QXNzZXQge1xyXG5cclxuICAgIGdldCBqb2JOYW1lKCk6IHN0cmluZyB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZGF0YS5qb2JOYW1lO1xyXG4gICAgfVxyXG5cclxuICAgIGdldCBqb2JQYXJhbXMoKTogYW55IHtcclxuICAgICAgICByZXR1cm4gdGhpcy5kYXRhLmpvYlBhcmFtcztcclxuICAgIH1cclxuXHJcbiAgICBnZXQgam9iUXVlKCk6IHN0cmluZyB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuZGF0YS5qb2JRdWU7XHJcbiAgICB9XHJcblxyXG4gICAgZ2V0IHByb2dyZXNzSWQoKTogc3RyaW5nIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5kYXRhLnByb2dyZXNzSWQ7XHJcbiAgICB9XHJcblxyXG4gICAgZ2V0IGFzc2V0SWQoKTogc3RyaW5nIHtcclxuICAgICAgICByZXR1cm4gdGhpcy5kYXRhLmFzc2V0SWQ7XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3RydWN0b3IoaWQ6IE9iamVjdElkLFxyXG4gICAgICAgICAgICAgICAgZGF0YTogUGFydGlhbDxJTGF6eUFzc2V0PixcclxuICAgICAgICAgICAgICAgIGNvbGxlY3Rpb246IENvbGxlY3Rpb24sXHJcbiAgICAgICAgICAgICAgICBwcm90ZWN0ZWQgYXNzZXRzOiBBc3NldHMsXHJcbiAgICAgICAgICAgICAgICBwcm90ZWN0ZWQgcHJvZ3Jlc3NlczogUHJvZ3Jlc3NlcyxcclxuICAgICAgICAgICAgICAgIHByb3RlY3RlZCBqb2JNYW46IEpvYk1hbmFnZXIpIHtcclxuICAgICAgICBzdXBlcihpZCwgZGF0YSwgY29sbGVjdGlvbik7XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgdW5saW5rKCk6IFByb21pc2U8c3RyaW5nPiB7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5sb2FkKCk7XHJcbiAgICAgICAgaWYgKCF0aGlzLnByb2dyZXNzSWQpIHtcclxuICAgICAgICAgICAgYXdhaXQgdGhpcy5jb2xsZWN0aW9uLmRlbGV0ZU9uZSh7X2lkOiB0aGlzLm1JZH0pO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gZGVsZXRlRnJvbUJ1Y2tldCh0aGlzLmFzc2V0cy5idWNrZXQsIG5ldyBPYmplY3RJZCh0aGlzLmFzc2V0SWQpKTtcclxuICAgIH1cclxuXHJcbiAgICBzdGFydFdvcmtpbmcoKTogdm9pZCB7XHJcbiAgICAgICAgdGhpcy5sb2FkKCkudGhlbigoKSA9PiB7XHJcbiAgICAgICAgICAgIGlmICh0aGlzLmRlbGV0ZWQpIHJldHVybjtcclxuICAgICAgICAgICAgdGhpcy5wcm9ncmVzc2VzLmdldCh0aGlzLnByb2dyZXNzSWQpLnRoZW4ocCA9PiB7XHJcbiAgICAgICAgICAgICAgICBwPy5jYW5jZWwoKTtcclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIHRoaXMuc3RhcnRXb3JraW5nT25Bc3NldChmYWxzZSkudGhlbigoKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhgU3RhcnRlZCB3b3JraW5nIG9uIGxhenkgYXNzZXQ6ICR7dGhpcy5pZH1gKTtcclxuICAgICAgICAgICAgfSkuY2F0Y2gocmVhc29uID0+IHtcclxuICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKGBDYW4ndCBzdGFydCB3b3JraW5nIG9uIGxhenkgYXNzZXQ6ICR7dGhpcy5pZH1cXG5SZWFzb246ICR7cmVhc29ufWApO1xyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICBhc3luYyBsb2FkQXNzZXQoKTogUHJvbWlzZTxJQXNzZXQ+IHtcclxuICAgICAgICBhd2FpdCB0aGlzLmxvYWQoKTtcclxuICAgICAgICBpZiAodGhpcy5kZWxldGVkKSByZXR1cm4gbnVsbDtcclxuICAgICAgICBpZiAodGhpcy5hc3NldElkKSB7XHJcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmFzc2V0cy5yZWFkKHRoaXMuYXNzZXRJZCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmICh0aGlzLnByb2dyZXNzSWQpIHtcclxuICAgICAgICAgICAgYXdhaXQgdGhpcy5wcm9ncmVzc2VzLndhaXRUb0ZpbmlzaCh0aGlzLnByb2dyZXNzSWQpO1xyXG4gICAgICAgICAgICByZXR1cm4gdGhpcy5sb2FkQXNzZXQoKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgYXdhaXQgdGhpcy5zdGFydFdvcmtpbmdPbkFzc2V0KHRydWUpO1xyXG4gICAgICAgIHJldHVybiB0aGlzLmxvYWRBc3NldCgpO1xyXG4gICAgfVxyXG5cclxuICAgIGFzeW5jIHdyaXRlQXNzZXQoYXNzZXQ6IElBc3NldCk6IFByb21pc2U8SUFzc2V0PiB7XHJcbiAgICAgICAgdGhpcy5kYXRhLmFzc2V0SWQgPSBhc3NldC5pZDtcclxuICAgICAgICBhd2FpdCB0aGlzLnNhdmUoKTtcclxuICAgICAgICByZXR1cm4gYXNzZXQ7XHJcbiAgICB9XHJcblxyXG4gICAgcHJvdGVjdGVkIGFzeW5jIHN0YXJ0V29ya2luZ09uQXNzZXQoZnJvbUxvYWQ6IGJvb2xlYW4pOiBQcm9taXNlPGFueT4ge1xyXG4gICAgICAgIHRoaXMuZGF0YS5wcm9ncmVzc0lkID0gKGF3YWl0IHRoaXMucHJvZ3Jlc3Nlcy5jcmVhdGUoKSkuaWQ7XHJcbiAgICAgICAgdGhpcy5kYXRhLmFzc2V0SWQgPSBudWxsO1xyXG4gICAgICAgIGF3YWl0IHRoaXMuc2F2ZSgpO1xyXG4gICAgICAgIGF3YWl0IHRoaXMuam9iTWFuLmVucXVldWVXaXRoTmFtZSh0aGlzLmRhdGEuam9iTmFtZSwgey4uLnRoaXMuZGF0YS5qb2JQYXJhbXMsIGxhenlJZDogdGhpcy5pZCwgZnJvbUxvYWR9KTtcclxuICAgIH1cclxufVxyXG4iXX0=
@@ -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)}`;
106
+ console.time(timerId);
107
+ console.timeLog(timerId, `Started working on background job: ${jobNameLog} with args: ${jobArgsLog}\n\n`);
108
+ try {
109
+ yield Promise.race([this.jobs[jobName](jobParams), promiseTimeout(15000, true)]);
110
+ console.timeLog(timerId, `Finished working on background job: ${jobNameLog}\n\n`);
111
+ }
112
+ catch (e) {
113
+ console.timeLog(timerId, `Background job failed: ${jobNameLog}\n${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":";;;;;;;;;;;;;;;;;;;;;AAAA,OAAO,EAAsB,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,EAAC,MAAM,UAAU,CAAC;AAC/F,OAAO,EAAC,QAAQ,EAAE,QAAQ,EAAC,MAAM,WAAW,CAAC;AAC7C,OAAO,EAAC,MAAM,EAAS,MAAM,QAAQ,CAAC;AACtC,OAAO,EAAC,QAAQ,EAAC,MAAM,MAAM,CAAC;AAC9B,OAAO,EAAC,YAAY,EAAkB,GAAG,EAAqD,MAAM,iBAAiB,CAAC;AACtH,OAAO,EAAC,kBAAkB,EAAE,OAAO,EAAE,QAAQ,EAAE,aAAa,EAAE,cAAc,EAAC,MAAM,UAAU,CAAC;AAC9F,OAAO,EAAC,aAAa,EAAC,MAAM,iBAAiB,CAAC;IAIjC,UAAU,SAAV,UAAU;IAOnB,YAAqB,MAAqB,EAAiC,SAA8B,EAAkB,QAAsB;QAA5H,WAAM,GAAN,MAAM,CAAe;QAAiC,cAAS,GAAT,SAAS,CAAqB;QACrG,IAAI,CAAC,QAAQ,GAAG,QAAQ,IAAI,EAAE,CAAC;QAC/B,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,OAAO,EAAE,EAAE;YAC9C,GAAG,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,SAAoB,EAAE,EAAE;gBACxD,MAAM,GAAG,GAAG,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;gBACxD,OAAO,GAAG,CAAC,OAAO,EAAE,CAAC;YACzB,CAAC,CAAA;YACD,OAAO,GAAG,CAAC;QACf,CAAC,EAAE,EAAE,CAAC,CAAC;IACX,CAAC;IAEK,OAAO,CAAC,OAAmB,EAAE,SAAoB,EAAE;;YACrD,IAAI,QAAQ,GAAS,IAAI,CAAC;YAC1B,IAAI;gBACA,QAAQ,GAAG,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;aACvD;YAAC,OAAO,CAAC,EAAE;gBACR,MAAM,OAAO,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;gBAC5C,MAAM,iCAAiC,OAAO,kBAAkB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;aAC1G;YACD,OAAO,QAAQ,CAAC,OAAO,EAAE,CAAC;QAC9B,CAAC;KAAA;IAEK,eAAe,CAAC,IAAY,EAAE,SAAoB,EAAE;;YACtD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;YAC5D,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC/C,CAAC;KAAA;IAEK,OAAO,CAAC,OAAmB,EAAE,SAAoB,EAAE;;YACrD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;YACjE,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC/C,CAAC;KAAA;IAEe,aAAa,CAAC,OAAe,EAAE,MAAiB;;YAC5D,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC;YACvC,MAAM,SAAS,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,IAAI,QAAQ,EAAE,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;QAC1F,CAAC;KAAA;IAED,QAAQ,CAAC,MAAuB,EAAE,IAAqB,EAAE,UAA2B,EAAE,KAAsB,EAAE,SAA0B,EAAE,OAAmB,EAAE,SAAoB,EAAE;QACjL,MAAM,UAAU,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,UAAU,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACpE,IAAI,QAAQ,CAAC,CAAC,CAAC,EAAE;gBACb,MAAM,KAAK,GAAG,CAAqB,CAAC;gBACpC,OAAO,GAAG,KAAK,CAAC,GAAG,IAAI,CAAC,IAAI,KAAK,CAAC,GAAG,IAAI,CAAC,EAAE,CAAC;aAChD;YACD,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE;gBACZ,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aACtB;YACD,OAAO,GAAG,CAAC,EAAE,CAAC;QAClB,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACb,MAAM,OAAO,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;QAC5C,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;YACvB,OAAO,CAAC,GAAG,CAAC,6BAA6B,OAAO,uCAAuC,CAAC,CAAC;YACzF,OAAO,IAAI,CAAC;SACf;QACD,OAAO,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE;YAC7B,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;gBACpC,OAAO,CAAC,GAAG,CAAC,uBAAuB,OAAO,cAAc,CAAC,EAAE,CAAC,CAAC;YACjE,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACP,CAAC;IAED,eAAe;QACX,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC;QAClD,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAC7B,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAC1B,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,CAAO,IAAY,EAAE,IAAY,EAAE,QAAgB,EAAE,EAAE;YAC7E,IAAI;gBACA,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBACtC,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAc,CAAC;gBACjE,MAAM,OAAO,GAAG,QAAQ,aAAR,QAAQ,uBAAR,QAAQ,CAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;gBAC3C,MAAM,UAAU,GAAG,YAAY,OAAO,UAAU,CAAC;gBACjD,MAAM,UAAU,GAAG,KAAK,aAAa,CAAC,SAAS,CAAC,EAAE,CAAC;gBAEnD,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACtB,OAAO,CAAC,OAAO,CAAC,OAAO,EAAE,sCAAsC,UAAU,eAAe,UAAU,MAAM,CAAC,CAAC;gBAC1G,IAAI;oBACA,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,EAAE,cAAc,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC;oBACjF,OAAO,CAAC,OAAO,CAAC,OAAO,EAAE,uCAAuC,UAAU,MAAM,CAAC,CAAC;iBACrF;gBAAC,OAAO,CAAC,EAAE;oBACR,OAAO,CAAC,OAAO,CAAC,OAAO,EAAE,0BAA0B,UAAU,KAAK,CAAC,CAAC,OAAO,MAAM,CAAC,CAAC;iBACtF;gBACD,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;aAC5B;YAAC,OAAO,CAAC,EAAE;gBACR,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;aACpD;QACL,CAAC,CAAA,CAAC,CAAC;QACH,OAAO,CAAC,GAAG,CAAC,wBAAwB,IAAI,EAAE,CAAC,CAAC;IAChD,CAAC;IAED,UAAU,CAAC,OAAmB,EAAE,MAAiB;QAC7C,MAAM,OAAO,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;QAC5C,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACrB,MAAM,6BAA6B,OAAO,2BAA2B,CAAC;SACzE;QACD,IAAI;YACA,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;SAC5C;QAAC,OAAO,CAAC,EAAE;YACR,MAAM,iCAAiC,OAAO,kBAAkB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;SAC1G;QACD,OAAO,OAAO,CAAC;IACnB,CAAC;IAES,kBAAkB,CAAC,OAAe,EAAE,MAAiB;QAC3D,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YACtC,OAAO,kBAAkB,CAAC,IAAI,CAAC,IAAI,OAAO,CAAC;QAC/C,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,OAAO,EAAE;YACV,MAAM,kCAAkC,OAAO,2BAA2B,CAAC;SAC9E;QACD,OAAO,IAAI,CAAC,oBAAoB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IACtD,CAAC;IAEe,oBAAoB,CAAC,OAAmB,EAAE,MAAiB;;YACvE,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,IAAI,IAAI,OAAO,CAAM,CAAM,OAAO,EAAC,EAAE;gBAChE,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBAC5C,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;gBACjC,MAAM,SAAS,CAAC,IAAI,CAAC,iBAAiB,IAAI,EAAE,CAAC,CAAA;gBAC7C,OAAO,CAAC,GAAG,CAAC,4BAA4B,IAAI,EAAE,CAAC,CAAC;gBAChD,OAAO,CAAC,SAAS,CAAC,CAAC;YACvB,CAAC,CAAA,CAAC,CAAC;YACH,OAAO,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC5C,CAAC;KAAA;IAES,kBAAkB,CAAC,OAAmB,EAAE,MAAiB;QAC/D,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC,oBAAoB,EAAE,CAAC;QACxD,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YAC7B,SAAS,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAC,QAAQ,EAAE,MAAM,CAAC,IAAI,CAAC,EAAC,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QACH,SAAS,CAAC,QAAQ,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QACrC,OAAO,SAAS,CAAC,OAAO,CAAC,OAAO,CAAS,CAAC;IAC9C,CAAC;CACJ,CAAA;AAzIY,UAAU;IAFtB,UAAU,EAAE;IACZ,MAAM,CAAC,SAAS,CAAC,eAAe,CAAC;IAQe,WAAA,MAAM,CAAC,YAAY,CAAC,CAAA,EAA2C,WAAA,SAAS,CAAC,GAAG,CAAC,CAAA;qCAA7F,aAAa;GAPjC,UAAU,CAyItB;SAzIY,UAAU","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)}`;\r\n\r\n                console.time(timerId);\r\n                console.timeLog(timerId, `Started working on background job: ${jobNameLog} with args: ${jobArgsLog}\\n\\n`);\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}\\n\\n`);\r\n                } catch (e) {\r\n                    console.timeLog(timerId, `Background job failed: ${jobNameLog}\\n${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"]}