@xyo-network/image-thumbnail-plugin 2.99.5 → 2.99.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,11 +1,3 @@
1
- var __defProp = Object.defineProperty;
2
- var __getProtoOf = Object.getPrototypeOf;
3
- var __reflectGet = Reflect.get;
4
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
5
- var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
6
- var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
7
- var __superGet = (cls, obj, key) => __reflectGet(__getProtoOf(cls), key, obj);
8
-
9
1
  // src/Plugin.ts
10
2
  import { ImageThumbnailDiviner } from "@xyo-network/diviner-image-thumbnail";
11
3
  import { ImageThumbnailSchema as ImageThumbnailSchema3 } from "@xyo-network/image-thumbnail-payload-plugin";
@@ -19,13 +11,13 @@ var ImageThumbnailWitnessConfigSchema = `${ImageThumbnailSchema}.witness.config`
19
11
  // src/Witness/Witness.ts
20
12
  import { Buffer as Buffer2 } from "node:buffer";
21
13
  import { promises as dnsPromises } from "node:dns";
22
- import { axios as axios2 } from "@xylabs/axios";
23
14
  import { compact } from "@xylabs/lodash";
24
15
  import { AbstractWitness } from "@xyo-network/abstract-witness";
25
16
  import { PayloadHasher } from "@xyo-network/hash";
26
17
  import { ImageThumbnailSchema as ImageThumbnailSchema2 } from "@xyo-network/image-thumbnail-payload-plugin";
27
18
  import { UrlSchema } from "@xyo-network/url-payload-plugin";
28
19
  import { Semaphore } from "async-mutex";
20
+ import axios2 from "axios";
29
21
  import FileType from "file-type";
30
22
  import graphicsMagick from "gm";
31
23
  import hasbin from "hasbin";
@@ -39,8 +31,7 @@ import { tmpdir } from "node:os";
39
31
  import { Writable } from "node:stream";
40
32
  import ffmpeg from "fluent-ffmpeg";
41
33
  import { v4 as uuid } from "uuid";
42
- var _a;
43
- var FfmpegOutputStream = (_a = class extends Writable {
34
+ var FfmpegOutputStream = class extends Writable {
44
35
  chunks = [];
45
36
  constructor(options) {
46
37
  super(options);
@@ -50,18 +41,16 @@ var FfmpegOutputStream = (_a = class extends Writable {
50
41
  callback();
51
42
  }
52
43
  /**
53
- * Collects the output from ffmpeg into a buffer.
54
- * @returns A buffer containing the concatenated
55
- * output from ffmpeg.
56
- */
57
- toBuffer = /* @__PURE__ */ __name(() => Buffer.concat(this.chunks), "toBuffer");
58
- }, __name(_a, "FfmpegOutputStream"), _a);
59
- var getVideoFrameAsImageFluent = /* @__PURE__ */ __name(async (videoBuffer) => {
44
+ * Collects the output from ffmpeg into a buffer.
45
+ * @returns A buffer containing the concatenated
46
+ * output from ffmpeg.
47
+ */
48
+ toBuffer = () => Buffer.concat(this.chunks);
49
+ };
50
+ var getVideoFrameAsImageFluent = async (videoBuffer) => {
60
51
  const tmpFile = `/${tmpdir()}/${uuid()}`;
61
52
  try {
62
- await writeFile(tmpFile, new Uint8Array(videoBuffer), {
63
- encoding: "binary"
64
- });
53
+ await writeFile(tmpFile, new Uint8Array(videoBuffer), { encoding: "binary" });
65
54
  const imageBuffer = await new Promise((resolve, reject) => {
66
55
  const ffmpegOutput = new FfmpegOutputStream();
67
56
  ffmpeg().on("error", (err) => reject(err.message)).on("end", () => resolve(ffmpegOutput.toBuffer())).input(tmpFile).takeFrames(1).withNoAudio().outputOptions("-f image2pipe").videoCodec("png").pipe(ffmpegOutput);
@@ -73,12 +62,12 @@ var getVideoFrameAsImageFluent = /* @__PURE__ */ __name(async (videoBuffer) => {
73
62
  } catch {
74
63
  }
75
64
  }
76
- }, "getVideoFrameAsImageFluent");
65
+ };
77
66
 
78
67
  // src/Witness/lib/checkIpfsUrl.ts
79
68
  import { assertEx } from "@xylabs/assert";
80
69
  var allowIpfsIoRepair = true;
81
- var checkIpfsUrl = /* @__PURE__ */ __name((urlToCheck, ipfsGateway) => {
70
+ var checkIpfsUrl = (urlToCheck, ipfsGateway) => {
82
71
  try {
83
72
  const url = new URL(urlToCheck);
84
73
  let protocol = url.protocol;
@@ -107,19 +96,19 @@ var checkIpfsUrl = /* @__PURE__ */ __name((urlToCheck, ipfsGateway) => {
107
96
  } catch {
108
97
  return urlToCheck;
109
98
  }
110
- }, "checkIpfsUrl");
99
+ };
111
100
 
112
101
  // src/Witness/lib/createDataUrl.ts
113
102
  import { fromByteArray } from "base64-js";
114
- var createDataUrl = /* @__PURE__ */ __name((data, contextType, encoding = "base64") => {
103
+ var createDataUrl = (data, contextType, encoding = "base64") => {
115
104
  return `data:${contextType};${encoding},${fromByteArray(new Uint8Array(data))}`;
116
- }, "createDataUrl");
105
+ };
117
106
 
118
107
  // src/Witness/lib/resolveDynamicSvg.ts
119
- import { axios } from "@xylabs/axios";
108
+ import axios from "axios";
120
109
  import { toByteArray } from "base64-js";
121
110
  import { Builder, parseStringPromise } from "xml2js";
122
- var resolveDynamicSvg = /* @__PURE__ */ __name(async (base64Bytes) => {
111
+ var resolveDynamicSvg = async (base64Bytes) => {
123
112
  const decoder = new TextDecoder();
124
113
  const bytes = toByteArray(base64Bytes);
125
114
  const svg = decoder.decode(bytes);
@@ -135,38 +124,24 @@ var resolveDynamicSvg = /* @__PURE__ */ __name(async (base64Bytes) => {
135
124
  ])
136
125
  );
137
126
  const image = imageResults.map(([href, response]) => {
138
- var _a2;
127
+ var _a;
139
128
  if (response.data) {
140
129
  const sourceBuffer = Buffer.from(response.data, "binary");
141
- return {
142
- $: {
143
- href: `data:${(_a2 = response.headers["content-type"]) == null ? void 0 : _a2.toString()};base64,${sourceBuffer.toString("base64")}`
144
- }
145
- };
130
+ return { $: { href: `data:${(_a = response.headers["content-type"]) == null ? void 0 : _a.toString()};base64,${sourceBuffer.toString("base64")}` } };
146
131
  } else {
147
- return {
148
- $: {
149
- href
150
- }
151
- };
132
+ return { $: { href } };
152
133
  }
153
134
  });
154
- const updatedSVG = {
155
- ...svgObj,
156
- svg: {
157
- ...svgNode,
158
- image
159
- }
160
- };
135
+ const updatedSVG = { ...svgObj, svg: { ...svgNode, image } };
161
136
  const builder = new Builder();
162
137
  return builder.buildObject(updatedSVG);
163
- }, "resolveDynamicSvg");
138
+ };
164
139
 
165
140
  // src/Witness/Witness.ts
166
- var gm = graphicsMagick.subClass({
167
- imageMagick: "7+"
168
- });
169
- var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitness {
141
+ var gm = graphicsMagick.subClass({ imageMagick: "7+" });
142
+ var ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitness {
143
+ static configSchemas = [...super.configSchemas, ImageThumbnailWitnessConfigSchema];
144
+ static defaultConfigSchema = ImageThumbnailWitnessConfigSchema;
170
145
  _semaphore = new Semaphore(this.maxAsyncProcesses);
171
146
  get encoding() {
172
147
  return this.config.encoding ?? "PNG";
@@ -218,42 +193,50 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
218
193
  throw new Error("ImageMagick is required for this witness");
219
194
  }
220
195
  const urlPayloads = payloads.filter((payload) => payload.schema === UrlSchema);
221
- const process = /* @__PURE__ */ __name(async () => {
222
- return compact(await Promise.all(urlPayloads.map(async ({ url }) => {
223
- let result;
224
- const dataBuffer = _ImageThumbnailWitness.bufferFromDataUrl(url);
225
- if (dataBuffer) {
226
- if (this.config.dataUrlPassthrough) {
227
- result = {
228
- schema: ImageThumbnailSchema2,
229
- sourceHash: await _ImageThumbnailWitness.binaryToSha256(dataBuffer),
230
- sourceUrl: url,
231
- url
232
- };
233
- } else {
234
- let cookedDataBuffer = dataBuffer;
235
- const urlParts = url.split(";");
236
- const [, contentType] = urlParts[0].split(":");
237
- if (contentType.startsWith("image/svg")) {
238
- const [encoding, byteString] = urlParts[1].split(",");
239
- if (encoding === "base64") {
240
- const newSvg = await resolveDynamicSvg(byteString);
241
- const newSvgDataUrl = createDataUrl(Buffer2.from(newSvg), contentType);
242
- cookedDataBuffer = _ImageThumbnailWitness.bufferFromDataUrl(newSvgDataUrl) ?? dataBuffer;
196
+ const process = async () => {
197
+ return compact(
198
+ await Promise.all(
199
+ urlPayloads.map(async ({ url }) => {
200
+ let result;
201
+ const dataBuffer = _ImageThumbnailWitness.bufferFromDataUrl(url);
202
+ if (dataBuffer) {
203
+ if (this.config.dataUrlPassthrough) {
204
+ result = {
205
+ schema: ImageThumbnailSchema2,
206
+ sourceHash: await _ImageThumbnailWitness.binaryToSha256(dataBuffer),
207
+ sourceUrl: url,
208
+ url
209
+ };
210
+ } else {
211
+ let cookedDataBuffer = dataBuffer;
212
+ const urlParts = url.split(";");
213
+ const [, contentType] = urlParts[0].split(":");
214
+ if (contentType.startsWith("image/svg")) {
215
+ const [encoding, byteString] = urlParts[1].split(",");
216
+ if (encoding === "base64") {
217
+ const newSvg = await resolveDynamicSvg(byteString);
218
+ const newSvgDataUrl = createDataUrl(Buffer2.from(newSvg), contentType);
219
+ cookedDataBuffer = _ImageThumbnailWitness.bufferFromDataUrl(newSvgDataUrl) ?? dataBuffer;
220
+ }
221
+ }
222
+ result = await this.processMedia(
223
+ cookedDataBuffer,
224
+ {
225
+ schema: ImageThumbnailSchema2,
226
+ sourceUrl: url
227
+ },
228
+ contentType
229
+ );
243
230
  }
231
+ } else {
232
+ const mutatedUrl = checkIpfsUrl(url, this.ipfsGateway);
233
+ result = await this.fromHttp(mutatedUrl, url);
244
234
  }
245
- result = await this.processMedia(cookedDataBuffer, {
246
- schema: ImageThumbnailSchema2,
247
- sourceUrl: url
248
- }, contentType);
249
- }
250
- } else {
251
- const mutatedUrl = checkIpfsUrl(url, this.ipfsGateway);
252
- result = await this.fromHttp(mutatedUrl, url);
253
- }
254
- return result;
255
- })));
256
- }, "process");
235
+ return result;
236
+ })
237
+ )
238
+ );
239
+ };
257
240
  return this.config.runExclusive ? await this._semaphore.runExclusive(() => process()) : process();
258
241
  }
259
242
  async createThumbnailDataUrl(sourceBuffer, encoding) {
@@ -269,17 +252,17 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
269
252
  return createDataUrl(thumb, "image/png");
270
253
  }
271
254
  /**
272
- * Creates an image thumbnail from a video.
273
- * @param videoBuffer The input video buffer.
274
- * @returns An buffer containing an image thumbnail for the video.
275
- */
255
+ * Creates an image thumbnail from a video.
256
+ * @param videoBuffer The input video buffer.
257
+ * @returns An buffer containing an image thumbnail for the video.
258
+ */
276
259
  async createThumbnailFromVideo(videoBuffer) {
277
260
  const imageBuffer = await getVideoFrameAsImageFluent(videoBuffer);
278
261
  return this.createThumbnailDataUrl(imageBuffer);
279
262
  }
280
263
  // eslint-disable-next-line complexity
281
264
  async fromHttp(url, sourceUrl) {
282
- var _a2, _b, _c;
265
+ var _a, _b, _c;
283
266
  let response;
284
267
  let dnsResult;
285
268
  try {
@@ -310,7 +293,7 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
310
293
  schema: ImageThumbnailSchema2,
311
294
  sourceUrl: sourceUrl ?? url
312
295
  };
313
- if (((_a2 = axiosError == null ? void 0 : axiosError.response) == null ? void 0 : _a2.status) !== void 0) {
296
+ if (((_a = axiosError == null ? void 0 : axiosError.response) == null ? void 0 : _a.status) !== void 0) {
314
297
  result2.http = result2.http ?? {};
315
298
  result2.http.status = (_b = axiosError == null ? void 0 : axiosError.response) == null ? void 0 : _b.status;
316
299
  }
@@ -338,24 +321,21 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
338
321
  return result;
339
322
  }
340
323
  async processMedia(sourceBuffer, imageThumbnail, contentType) {
341
- var _a2, _b, _c, _d, _e;
342
- const [mediaType, fileType] = (contentType == null ? void 0 : contentType.split("/")) ?? [
343
- "",
344
- ""
345
- ];
324
+ var _a, _b, _c, _d, _e;
325
+ const [mediaType, fileType] = (contentType == null ? void 0 : contentType.split("/")) ?? ["", ""];
346
326
  imageThumbnail.mime = imageThumbnail.mime ?? {};
347
327
  imageThumbnail.mime.returned = mediaType;
348
328
  try {
349
329
  imageThumbnail.mime.detected = await FileType.fromBuffer(sourceBuffer);
350
330
  } catch (ex) {
351
331
  const error = ex;
352
- (_a2 = this.logger) == null ? void 0 : _a2.error(`FileType error: ${error.message}`);
332
+ (_a = this.logger) == null ? void 0 : _a.error(`FileType error: ${error.message}`);
353
333
  }
354
- const processImage = /* @__PURE__ */ __name(async (encoding2) => {
334
+ const processImage = async (encoding2) => {
355
335
  imageThumbnail.sourceHash = await _ImageThumbnailWitness.binaryToSha256(sourceBuffer);
356
336
  imageThumbnail.url = await this.createThumbnailDataUrl(sourceBuffer, encoding2);
357
- }, "processImage");
358
- const processVideo = /* @__PURE__ */ __name(async () => {
337
+ };
338
+ const processVideo = async () => {
359
339
  if (hasbin.sync("ffmpeg")) {
360
340
  imageThumbnail.sourceHash = await _ImageThumbnailWitness.binaryToSha256(sourceBuffer);
361
341
  imageThumbnail.url = await this.createThumbnailFromVideo(sourceBuffer);
@@ -363,7 +343,7 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
363
343
  imageThumbnail.mime = imageThumbnail.mime ?? {};
364
344
  imageThumbnail.mime.invalid = true;
365
345
  }
366
- }, "processVideo");
346
+ };
367
347
  let encoding = "PNG";
368
348
  switch (fileType.toUpperCase()) {
369
349
  case "GIF": {
@@ -388,10 +368,7 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
388
368
  break;
389
369
  }
390
370
  default: {
391
- const [detectedMediaType] = ((_c = (_b = imageThumbnail.mime.detected) == null ? void 0 : _b.mime) == null ? void 0 : _c.split("/")) ?? [
392
- "",
393
- ""
394
- ];
371
+ const [detectedMediaType] = ((_c = (_b = imageThumbnail.mime.detected) == null ? void 0 : _b.mime) == null ? void 0 : _c.split("/")) ?? ["", ""];
395
372
  switch (detectedMediaType) {
396
373
  case "image": {
397
374
  await processImage();
@@ -414,30 +391,21 @@ var _ImageThumbnailWitness = class _ImageThumbnailWitness extends AbstractWitnes
414
391
  return imageThumbnail;
415
392
  }
416
393
  };
417
- __name(_ImageThumbnailWitness, "ImageThumbnailWitness");
418
- __publicField(_ImageThumbnailWitness, "configSchemas", [
419
- ...__superGet(_ImageThumbnailWitness, _ImageThumbnailWitness, "configSchemas"),
420
- ImageThumbnailWitnessConfigSchema
421
- ]);
422
- __publicField(_ImageThumbnailWitness, "defaultConfigSchema", ImageThumbnailWitnessConfigSchema);
423
- var ImageThumbnailWitness = _ImageThumbnailWitness;
424
394
 
425
395
  // src/Plugin.ts
426
- var ImageThumbnailPlugin = /* @__PURE__ */ __name(() => createPayloadSetDualPlugin({
427
- required: {
428
- [ImageThumbnailSchema3]: 1
429
- },
430
- schema: PayloadSetSchema
431
- }, {
432
- diviner: /* @__PURE__ */ __name(async (params) => {
433
- const result = await ImageThumbnailDiviner.create(params);
434
- return result;
435
- }, "diviner"),
436
- witness: /* @__PURE__ */ __name(async (params) => {
437
- const result = await ImageThumbnailWitness.create(params);
438
- return result;
439
- }, "witness")
440
- }), "ImageThumbnailPlugin");
396
+ var ImageThumbnailPlugin = () => createPayloadSetDualPlugin(
397
+ { required: { [ImageThumbnailSchema3]: 1 }, schema: PayloadSetSchema },
398
+ {
399
+ diviner: async (params) => {
400
+ const result = await ImageThumbnailDiviner.create(params);
401
+ return result;
402
+ },
403
+ witness: async (params) => {
404
+ const result = await ImageThumbnailWitness.create(params);
405
+ return result;
406
+ }
407
+ }
408
+ );
441
409
 
442
410
  // src/index.ts
443
411
  export * from "@xyo-network/diviner-image-thumbnail";
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/Plugin.ts","../../src/Witness/Config.ts","../../src/Witness/Witness.ts","../../src/Witness/ffmpeg/fluent/getVideoFrameAsImageFluent.ts","../../src/Witness/lib/checkIpfsUrl.ts","../../src/Witness/lib/createDataUrl.ts","../../src/Witness/lib/resolveDynamicSvg.ts","../../src/index.ts"],"sourcesContent":["import { ImageThumbnailDiviner } from '@xyo-network/diviner-image-thumbnail'\nimport { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDualPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { ImageThumbnailWitness } from './Witness/index.ts'\n\nexport const ImageThumbnailPlugin = () =>\n createPayloadSetDualPlugin<ImageThumbnailWitness, ImageThumbnailDiviner>(\n { required: { [ImageThumbnailSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await ImageThumbnailDiviner.create(params)\n return result\n },\n witness: async (params) => {\n const result = await ImageThumbnailWitness.create(params)\n return result\n },\n },\n )\n","import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { WitnessConfig } from '@xyo-network/witness-model'\n\nexport const ImageThumbnailWitnessConfigSchema = `${ImageThumbnailSchema}.witness.config` as const\nexport type ImageThumbnailWitnessConfigSchema = typeof ImageThumbnailWitnessConfigSchema\n\nexport type ImageThumbnailEncoding = 'PNG' | 'JPG' | 'GIF'\n\nexport type ImageThumbnailWitnessConfig = WitnessConfig<{\n dataUrlPassthrough?: boolean\n encoding?: ImageThumbnailEncoding\n height?: number\n ipfsGateway?: string\n maxAsyncProcesses?: number\n maxCacheBytes?: number\n maxCacheEntries?: number\n quality?: number\n runExclusive?: boolean\n schema: ImageThumbnailWitnessConfigSchema\n width?: number\n}>\n","/* eslint-disable max-statements */\nimport { Buffer } from 'node:buffer'\nimport { promises as dnsPromises } from 'node:dns'\n\nimport { axios, AxiosError, AxiosResponse } from '@xylabs/axios'\nimport { compact } from '@xylabs/lodash'\nimport { AbstractWitness } from '@xyo-network/abstract-witness'\nimport { PayloadHasher } from '@xyo-network/hash'\nimport { ImageThumbnail, ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { Schema } from '@xyo-network/payload-model'\nimport { UrlPayload, UrlSchema } from '@xyo-network/url-payload-plugin'\nimport { Semaphore } from 'async-mutex'\nimport FileType from 'file-type'\nimport graphicsMagick from 'gm'\nimport hasbin from 'hasbin'\nimport { sha256 } from 'hash-wasm'\nimport shajs from 'sha.js'\nimport Url from 'url-parse'\n\nimport { ImageThumbnailEncoding, ImageThumbnailWitnessConfigSchema } from './Config.ts'\nimport { getVideoFrameAsImageFluent } from './ffmpeg/index.ts'\nimport { checkIpfsUrl, createDataUrl, resolveDynamicSvg } from './lib/index.ts'\nimport { ImageThumbnailWitnessParams } from './Params.ts'\n\n// TODO: Break this into two Witnesses?\n\n// setFfmpegPath(ffmpegPath)\n\nconst gm = graphicsMagick.subClass({ imageMagick: '7+' })\n\nexport interface ImageThumbnailWitnessError extends Error {\n name: 'ImageThumbnailWitnessError'\n url: string\n}\n\nexport interface DnsError extends Error {\n code: string\n}\n\nexport class ImageThumbnailWitness<TParams extends ImageThumbnailWitnessParams = ImageThumbnailWitnessParams> extends AbstractWitness<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, ImageThumbnailWitnessConfigSchema]\n static override readonly defaultConfigSchema: Schema = ImageThumbnailWitnessConfigSchema\n\n private _semaphore = new Semaphore(this.maxAsyncProcesses)\n\n get encoding() {\n return this.config.encoding ?? 'PNG'\n }\n\n get height() {\n return this.config.height ?? 128\n }\n\n get ipfsGateway() {\n return this.config.ipfsGateway ?? '5d7b6582.beta.decentralnetworkservices.com'\n }\n\n get maxAsyncProcesses() {\n return this.config.maxAsyncProcesses ?? 4\n }\n\n get quality() {\n return this.config.quality ?? 50\n }\n\n get width() {\n return this.config.width ?? 128\n }\n\n private static async binaryToSha256(data: ArrayBuffer) {\n const viewData = new Uint8Array(data)\n await PayloadHasher.wasmInitialized\n if (PayloadHasher.wasmSupport.canUseWasm) {\n try {\n return await sha256(viewData)\n } catch {\n PayloadHasher.wasmSupport.allowWasm = false\n }\n }\n\n return shajs('sha256').update(viewData).digest().toString()\n }\n\n private static bufferFromDataUrl(url: string): ArrayBuffer | undefined {\n if (url.startsWith('data:image')) {\n const data = url.split(',')[1]\n if (data) {\n return Uint8Array.from(atob(data), c => c.codePointAt(0) ?? 0)\n } else {\n const error: ImageThumbnailWitnessError = {\n message: 'Invalid data Url',\n name: 'ImageThumbnailWitnessError',\n url,\n }\n throw error\n }\n }\n }\n\n protected override async observeHandler(payloads: UrlPayload[] = []): Promise<ImageThumbnail[]> {\n if (!hasbin.sync('magick')) {\n throw new Error('ImageMagick is required for this witness')\n }\n const urlPayloads = payloads.filter(payload => payload.schema === UrlSchema)\n const process = async () => {\n return compact(\n await Promise.all(\n urlPayloads.map<Promise<ImageThumbnail>>(async ({ url }) => {\n let result: ImageThumbnail\n\n // if it is a data URL, return a Buffer\n const dataBuffer = ImageThumbnailWitness.bufferFromDataUrl(url)\n\n if (dataBuffer) {\n if (this.config.dataUrlPassthrough) {\n result = {\n schema: ImageThumbnailSchema,\n sourceHash: await ImageThumbnailWitness.binaryToSha256(dataBuffer),\n sourceUrl: url,\n url,\n }\n } else {\n let cookedDataBuffer = dataBuffer\n const urlParts = url.split(';')\n const [, contentType] = urlParts[0].split(':')\n if (contentType.startsWith('image/svg')) {\n const [encoding, byteString] = urlParts[1].split(',')\n if (encoding === 'base64') {\n const newSvg = await resolveDynamicSvg(byteString)\n const newSvgDataUrl = createDataUrl(Buffer.from(newSvg), contentType)\n cookedDataBuffer = ImageThumbnailWitness.bufferFromDataUrl(newSvgDataUrl) ?? dataBuffer\n }\n }\n result = await this.processMedia(\n cookedDataBuffer,\n {\n schema: ImageThumbnailSchema,\n sourceUrl: url,\n },\n contentType,\n )\n }\n } else {\n // if it is ipfs, go through cloud flair\n const mutatedUrl = checkIpfsUrl(url, this.ipfsGateway)\n result = await this.fromHttp(mutatedUrl, url)\n }\n return result\n }),\n ),\n )\n }\n return this.config.runExclusive ? await this._semaphore.runExclusive(() => process()) : process()\n }\n\n private async createThumbnailDataUrl(sourceBuffer: ArrayBuffer, encoding?: ImageThumbnailEncoding) {\n const thumb = await new Promise<Buffer>((resolve, reject) => {\n gm(Buffer.from(sourceBuffer))\n .quality(this.quality)\n .resize(this.width, this.height)\n .flatten()\n .toBuffer(encoding ?? this.encoding, (error, buffer) => {\n if (error) {\n reject(error)\n } else {\n resolve(buffer)\n }\n })\n })\n return createDataUrl(thumb, 'image/png')\n }\n\n /**\n * Creates an image thumbnail from a video.\n * @param videoBuffer The input video buffer.\n * @returns An buffer containing an image thumbnail for the video.\n */\n private async createThumbnailFromVideo(videoBuffer: ArrayBuffer) {\n const imageBuffer = await getVideoFrameAsImageFluent(videoBuffer)\n return this.createThumbnailDataUrl(imageBuffer)\n }\n\n // eslint-disable-next-line complexity\n private async fromHttp(url: string, sourceUrl?: string): Promise<ImageThumbnail> {\n let response: AxiosResponse\n let dnsResult: string[]\n try {\n const urlObj = new Url(url)\n dnsResult = await dnsPromises.resolve(urlObj.host)\n } catch (ex) {\n const error = ex as DnsError\n const result: ImageThumbnail = {\n http: {\n code: error.code,\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n return result\n }\n try {\n response = await axios.get(url, {\n responseType: 'arraybuffer',\n })\n } catch (ex) {\n const axiosError = ex as AxiosError\n if (axiosError.isAxiosError) {\n // selectively pick fields from AxiosError\n const result: ImageThumbnail = {\n http: {\n ipAddress: dnsResult[0],\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n if (axiosError?.response?.status !== undefined) {\n result.http = result.http ?? {}\n result.http.status = axiosError?.response?.status\n }\n if (axiosError?.code !== undefined) {\n result.http = result.http ?? {}\n result.http.code = axiosError?.code\n }\n return result\n } else {\n throw ex\n }\n }\n\n const result: ImageThumbnail = {\n http: {\n status: response.status,\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n\n if (response.status >= 200 && response.status < 300) {\n const contentType: string | undefined = response.headers['content-type']?.toString()\n const sourceBuffer = Buffer.from(response.data, 'binary')\n\n return this.processMedia(sourceBuffer, result, contentType)\n }\n return result\n }\n\n private async processMedia(sourceBuffer: ArrayBuffer, imageThumbnail: ImageThumbnail, contentType?: string): Promise<ImageThumbnail> {\n const [mediaType, fileType] = contentType?.split('/') ?? ['', '']\n imageThumbnail.mime = imageThumbnail.mime ?? {}\n imageThumbnail.mime.returned = mediaType\n\n try {\n imageThumbnail.mime.detected = await FileType.fromBuffer(sourceBuffer)\n } catch (ex) {\n const error = ex as Error\n this.logger?.error(`FileType error: ${error.message}`)\n }\n\n const processImage = async (encoding?: ImageThumbnailEncoding) => {\n imageThumbnail.sourceHash = await ImageThumbnailWitness.binaryToSha256(sourceBuffer)\n imageThumbnail.url = await this.createThumbnailDataUrl(sourceBuffer, encoding)\n }\n\n const processVideo = async () => {\n // Gracefully handle the case where ffmpeg is not installed.\n\n if (hasbin.sync('ffmpeg')) {\n imageThumbnail.sourceHash = await ImageThumbnailWitness.binaryToSha256(sourceBuffer)\n imageThumbnail.url = await this.createThumbnailFromVideo(sourceBuffer)\n } else {\n imageThumbnail.mime = imageThumbnail.mime ?? {}\n imageThumbnail.mime.invalid = true\n }\n }\n\n let encoding: ImageThumbnailEncoding = 'PNG'\n\n switch (fileType.toUpperCase()) {\n case 'GIF': {\n encoding = 'GIF'\n break\n }\n case 'JPG':\n case 'JPEG': {\n encoding = 'JPG'\n break\n }\n }\n\n switch (mediaType) {\n case 'image': {\n await processImage(encoding)\n imageThumbnail.mime.type = mediaType\n break\n }\n case 'video': {\n await processVideo()\n imageThumbnail.mime.type = mediaType\n break\n }\n default: {\n const [detectedMediaType] = imageThumbnail.mime.detected?.mime?.split('/') ?? ['', '']\n switch (detectedMediaType) {\n case 'image': {\n await processImage()\n imageThumbnail.mime.type = imageThumbnail.mime.detected?.mime\n break\n }\n case 'video': {\n await processVideo()\n imageThumbnail.mime.type = imageThumbnail.mime.detected?.mime\n break\n }\n default: {\n imageThumbnail.mime.invalid = true\n break\n }\n }\n break\n }\n }\n return imageThumbnail\n }\n}\n","import { unlink, writeFile } from 'node:fs/promises'\nimport { tmpdir } from 'node:os'\nimport { Writable, WritableOptions } from 'node:stream'\n\nimport ffmpeg from 'fluent-ffmpeg'\nimport { v4 as uuid } from 'uuid'\n\n/**\n * A Writable stream that collects output from ffmpeg.\n */\nclass FfmpegOutputStream extends Writable {\n private readonly chunks: Uint8Array[] = []\n\n constructor(options?: WritableOptions) {\n super(options)\n }\n\n override _write(chunk: never, _encoding: BufferEncoding, callback: (error?: Error | null) => void): void {\n this.chunks.push(chunk)\n callback()\n }\n\n /**\n * Collects the output from ffmpeg into a buffer.\n * @returns A buffer containing the concatenated\n * output from ffmpeg.\n */\n toBuffer = () => Buffer.concat(this.chunks)\n}\n\n/**\n * Execute FFmpeg using fluent API with provided input buffer and video thumbnail image.\n * @param videoBuffer Input video buffer.\n * @returns Output buffer containing the video thumbnail image.\n */\nexport const getVideoFrameAsImageFluent = async (videoBuffer: ArrayBuffer) => {\n // Get a temp file name\n const tmpFile = `/${tmpdir()}/${uuid()}`\n try {\n // Write videoBuffer to temp file for use as input to ffmpeg to\n // avoid issues with ffmpeg inferring premature EOF from buffer\n // passed via stdin (happens when ffmpeg is trying to infer\n // input video format)\n await writeFile(tmpFile, new Uint8Array(videoBuffer), { encoding: 'binary' })\n const imageBuffer = await new Promise<Buffer>((resolve, reject) => {\n // Create a Writable stream to collect PNG output from ffmpeg\n const ffmpegOutput = new FfmpegOutputStream()\n // Execute ffmpeg using fluent API\n ffmpeg()\n // NOTE: Uncomment to debug CLI args to ffmpeg\n // .on('start', (commandLine) => console.log('Spawned Ffmpeg with command: ' + commandLine))\n .on('error', err => reject(err.message))\n // Listen for the 'end' event to combine the output into a buffer holding the PNG image\n .on('end', () => resolve(ffmpegOutput.toBuffer()))\n .input(tmpFile) // Use temp file as input\n .takeFrames(1) // Only take 1st video frame\n .withNoAudio() // Don't include audio\n .outputOptions('-f image2pipe') // Write output to stdout\n .videoCodec('png') // Force PNG output\n // Start processing and direct ffmpeg stdout to writable stream\n .pipe(ffmpegOutput)\n })\n return imageBuffer\n } finally {\n // Cleanup temp file\n try {\n await unlink(tmpFile)\n } catch {\n // No error here since file doesn't exist\n }\n }\n}\n","import { assertEx } from '@xylabs/assert'\n\nconst allowIpfsIoRepair = true\n\n/**\n * Returns the equivalent IPFS gateway URL for the supplied URL.\n * @param urlToCheck The URL to check\n * @returns If the supplied URL is an IPFS URL, it converts the URL to the\n * equivalent IPFS gateway URL. Otherwise, returns the original URL.\n */\nexport const checkIpfsUrl = (urlToCheck: string, ipfsGateway?: string): string => {\n try {\n const url = new URL(urlToCheck)\n let protocol = url.protocol\n let host = url.host\n let path = url.pathname\n const query = url.search\n if (protocol === 'ipfs:') {\n protocol = 'https:'\n host = assertEx(ipfsGateway, () => 'No ipfsGateway provided')\n path = url.host === 'ipfs' ? `ipfs${path}` : `ipfs/${url.host}${path}`\n const root = `${protocol}//${host}/${path}`\n return query?.length > 0 ? `${root}?${query}` : root\n } else if (allowIpfsIoRepair && protocol === 'https' && host === 'ipfs.io') {\n protocol = 'https:'\n host = assertEx(ipfsGateway, () => 'No ipfsGateway provided')\n const pathParts = path.split('/')\n if (pathParts[0] === 'ipfs') {\n pathParts.shift()\n }\n path = pathParts.join('/')\n const root = `${protocol}//${host}/${path}`\n return query?.length > 0 ? `${root}?${query}` : root\n } else {\n return urlToCheck\n }\n } catch {\n // const error = ex as Error\n // console.error(`${error.name}:${error.message} [${urlToCheck}]`)\n // console.log(error.stack)\n return urlToCheck\n }\n}\n","import { fromByteArray } from 'base64-js'\n\nexport const createDataUrl = (data: ArrayBuffer, contextType: string, encoding: 'base64' = 'base64') => {\n return `data:${contextType};${encoding},${fromByteArray(new Uint8Array(data))}`\n}\n","import { axios, AxiosResponse } from '@xylabs/axios'\nimport { toByteArray } from 'base64-js'\nimport { Builder, parseStringPromise } from 'xml2js'\n\nexport const resolveDynamicSvg = async (base64Bytes: string) => {\n const decoder = new TextDecoder()\n const bytes = toByteArray(base64Bytes)\n const svg = decoder.decode(bytes)\n const svgObj = await parseStringPromise(svg)\n const svgNode = svgObj['svg']\n const imageResults = (await Promise.all(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n svgNode['image'].map(async (img: any) => [\n img.$,\n await axios.get(img.$.href, {\n responseType: 'arraybuffer',\n }),\n ]),\n )) as [string, AxiosResponse][]\n const image = imageResults.map(([href, response]) => {\n if (response.data) {\n const sourceBuffer = Buffer.from(response.data, 'binary')\n return { $: { href: `data:${response.headers['content-type']?.toString()};base64,${sourceBuffer.toString('base64')}` } }\n } else {\n return { $: { href } }\n }\n })\n const updatedSVG = { ...svgObj, svg: { ...svgNode, image } }\n const builder = new Builder()\n return builder.buildObject(updatedSVG)\n}\n","// eslint-disable-next-line import/no-default-export\nexport { ImageThumbnailPlugin as default, ImageThumbnailPlugin } from './Plugin.ts'\nexport * from './Witness/index.ts'\nexport * from '@xyo-network/diviner-image-thumbnail'\n"],"mappings":"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","names":["ImageThumbnailDiviner","ImageThumbnailSchema","PayloadSetSchema","createPayloadSetDualPlugin","ImageThumbnailSchema","ImageThumbnailWitnessConfigSchema","Buffer","promises","dnsPromises","axios","compact","AbstractWitness","PayloadHasher","ImageThumbnailSchema","UrlSchema","Semaphore","FileType","graphicsMagick","hasbin","sha256","shajs","Url","unlink","writeFile","tmpdir","Writable","ffmpeg","v4","uuid","FfmpegOutputStream","Writable","chunks","constructor","options","_write","chunk","_encoding","callback","push","toBuffer","Buffer","concat","getVideoFrameAsImageFluent","videoBuffer","tmpFile","tmpdir","uuid","writeFile","Uint8Array","encoding","imageBuffer","Promise","resolve","reject","ffmpegOutput","ffmpeg","on","err","message","input","takeFrames","withNoAudio","outputOptions","videoCodec","pipe","unlink","assertEx","allowIpfsIoRepair","checkIpfsUrl","urlToCheck","ipfsGateway","url","URL","protocol","host","path","pathname","query","search","assertEx","root","length","pathParts","split","shift","join","fromByteArray","createDataUrl","data","contextType","encoding","fromByteArray","Uint8Array","axios","toByteArray","Builder","parseStringPromise","resolveDynamicSvg","base64Bytes","decoder","TextDecoder","bytes","toByteArray","svg","decode","svgObj","parseStringPromise","svgNode","imageResults","Promise","all","map","img","$","axios","get","href","responseType","image","response","_a","data","sourceBuffer","Buffer","from","headers","toString","updatedSVG","builder","Builder","buildObject","gm","graphicsMagick","subClass","imageMagick","ImageThumbnailWitness","AbstractWitness","_semaphore","Semaphore","maxAsyncProcesses","encoding","config","height","ipfsGateway","quality","width","binaryToSha256","data","viewData","Uint8Array","PayloadHasher","wasmInitialized","wasmSupport","canUseWasm","sha256","allowWasm","shajs","update","digest","toString","bufferFromDataUrl","url","startsWith","split","from","atob","c","codePointAt","error","message","name","observeHandler","payloads","hasbin","sync","Error","urlPayloads","filter","payload","schema","UrlSchema","process","compact","Promise","all","map","result","dataBuffer","dataUrlPassthrough","ImageThumbnailSchema","sourceHash","sourceUrl","cookedDataBuffer","urlParts","contentType","byteString","newSvg","resolveDynamicSvg","newSvgDataUrl","createDataUrl","Buffer","processMedia","mutatedUrl","checkIpfsUrl","fromHttp","runExclusive","createThumbnailDataUrl","sourceBuffer","thumb","resolve","reject","resize","flatten","toBuffer","buffer","createThumbnailFromVideo","videoBuffer","imageBuffer","getVideoFrameAsImageFluent","_a","response","dnsResult","urlObj","Url","dnsPromises","host","ex","http","code","axios","get","responseType","axiosError","isAxiosError","ipAddress","status","undefined","headers","imageThumbnail","mediaType","fileType","mime","returned","detected","FileType","fromBuffer","logger","processImage","processVideo","invalid","toUpperCase","type","detectedMediaType","configSchemas","ImageThumbnailWitnessConfigSchema","defaultConfigSchema","ImageThumbnailPlugin","createPayloadSetDualPlugin","required","ImageThumbnailSchema","schema","PayloadSetSchema","diviner","params","result","ImageThumbnailDiviner","create","witness","ImageThumbnailWitness"]}
1
+ {"version":3,"sources":["../../src/Plugin.ts","../../src/Witness/Config.ts","../../src/Witness/Witness.ts","../../src/Witness/ffmpeg/fluent/getVideoFrameAsImageFluent.ts","../../src/Witness/lib/checkIpfsUrl.ts","../../src/Witness/lib/createDataUrl.ts","../../src/Witness/lib/resolveDynamicSvg.ts","../../src/index.ts"],"sourcesContent":["import { ImageThumbnailDiviner } from '@xyo-network/diviner-image-thumbnail'\nimport { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { PayloadSetSchema } from '@xyo-network/payload-model'\nimport { createPayloadSetDualPlugin } from '@xyo-network/payloadset-plugin'\n\nimport { ImageThumbnailWitness } from './Witness/index.ts'\n\nexport const ImageThumbnailPlugin = () =>\n createPayloadSetDualPlugin<ImageThumbnailWitness, ImageThumbnailDiviner>(\n { required: { [ImageThumbnailSchema]: 1 }, schema: PayloadSetSchema },\n {\n diviner: async (params) => {\n const result = await ImageThumbnailDiviner.create(params)\n return result\n },\n witness: async (params) => {\n const result = await ImageThumbnailWitness.create(params)\n return result\n },\n },\n )\n","import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { WitnessConfig } from '@xyo-network/witness-model'\n\nexport const ImageThumbnailWitnessConfigSchema = `${ImageThumbnailSchema}.witness.config` as const\nexport type ImageThumbnailWitnessConfigSchema = typeof ImageThumbnailWitnessConfigSchema\n\nexport type ImageThumbnailEncoding = 'PNG' | 'JPG' | 'GIF'\n\nexport type ImageThumbnailWitnessConfig = WitnessConfig<{\n dataUrlPassthrough?: boolean\n encoding?: ImageThumbnailEncoding\n height?: number\n ipfsGateway?: string\n maxAsyncProcesses?: number\n maxCacheBytes?: number\n maxCacheEntries?: number\n quality?: number\n runExclusive?: boolean\n schema: ImageThumbnailWitnessConfigSchema\n width?: number\n}>\n","/* eslint-disable max-statements */\nimport { Buffer } from 'node:buffer'\nimport { promises as dnsPromises } from 'node:dns'\n\nimport { compact } from '@xylabs/lodash'\nimport { AbstractWitness } from '@xyo-network/abstract-witness'\nimport { PayloadHasher } from '@xyo-network/hash'\nimport { ImageThumbnail, ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { Schema } from '@xyo-network/payload-model'\nimport { UrlPayload, UrlSchema } from '@xyo-network/url-payload-plugin'\nimport { Semaphore } from 'async-mutex'\nimport type { AxiosError, AxiosResponse } from 'axios'\nimport axios from 'axios'\nimport FileType from 'file-type'\nimport graphicsMagick from 'gm'\nimport hasbin from 'hasbin'\nimport { sha256 } from 'hash-wasm'\nimport shajs from 'sha.js'\nimport Url from 'url-parse'\n\nimport { ImageThumbnailEncoding, ImageThumbnailWitnessConfigSchema } from './Config.ts'\nimport { getVideoFrameAsImageFluent } from './ffmpeg/index.ts'\nimport { checkIpfsUrl, createDataUrl, resolveDynamicSvg } from './lib/index.ts'\nimport { ImageThumbnailWitnessParams } from './Params.ts'\n\n// TODO: Break this into two Witnesses?\n\n// setFfmpegPath(ffmpegPath)\n\nconst gm = graphicsMagick.subClass({ imageMagick: '7+' })\n\nexport interface ImageThumbnailWitnessError extends Error {\n name: 'ImageThumbnailWitnessError'\n url: string\n}\n\nexport interface DnsError extends Error {\n code: string\n}\n\nexport class ImageThumbnailWitness<TParams extends ImageThumbnailWitnessParams = ImageThumbnailWitnessParams> extends AbstractWitness<TParams> {\n static override readonly configSchemas: Schema[] = [...super.configSchemas, ImageThumbnailWitnessConfigSchema]\n static override readonly defaultConfigSchema: Schema = ImageThumbnailWitnessConfigSchema\n\n private _semaphore = new Semaphore(this.maxAsyncProcesses)\n\n get encoding() {\n return this.config.encoding ?? 'PNG'\n }\n\n get height() {\n return this.config.height ?? 128\n }\n\n get ipfsGateway() {\n return this.config.ipfsGateway ?? '5d7b6582.beta.decentralnetworkservices.com'\n }\n\n get maxAsyncProcesses() {\n return this.config.maxAsyncProcesses ?? 4\n }\n\n get quality() {\n return this.config.quality ?? 50\n }\n\n get width() {\n return this.config.width ?? 128\n }\n\n private static async binaryToSha256(data: ArrayBuffer) {\n const viewData = new Uint8Array(data)\n await PayloadHasher.wasmInitialized\n if (PayloadHasher.wasmSupport.canUseWasm) {\n try {\n return await sha256(viewData)\n } catch {\n PayloadHasher.wasmSupport.allowWasm = false\n }\n }\n\n return shajs('sha256').update(viewData).digest().toString()\n }\n\n private static bufferFromDataUrl(url: string): ArrayBuffer | undefined {\n if (url.startsWith('data:image')) {\n const data = url.split(',')[1]\n if (data) {\n return Uint8Array.from(atob(data), c => c.codePointAt(0) ?? 0)\n } else {\n const error: ImageThumbnailWitnessError = {\n message: 'Invalid data Url',\n name: 'ImageThumbnailWitnessError',\n url,\n }\n throw error\n }\n }\n }\n\n protected override async observeHandler(payloads: UrlPayload[] = []): Promise<ImageThumbnail[]> {\n if (!hasbin.sync('magick')) {\n throw new Error('ImageMagick is required for this witness')\n }\n const urlPayloads = payloads.filter(payload => payload.schema === UrlSchema)\n const process = async () => {\n return compact(\n await Promise.all(\n urlPayloads.map<Promise<ImageThumbnail>>(async ({ url }) => {\n let result: ImageThumbnail\n\n // if it is a data URL, return a Buffer\n const dataBuffer = ImageThumbnailWitness.bufferFromDataUrl(url)\n\n if (dataBuffer) {\n if (this.config.dataUrlPassthrough) {\n result = {\n schema: ImageThumbnailSchema,\n sourceHash: await ImageThumbnailWitness.binaryToSha256(dataBuffer),\n sourceUrl: url,\n url,\n }\n } else {\n let cookedDataBuffer = dataBuffer\n const urlParts = url.split(';')\n const [, contentType] = urlParts[0].split(':')\n if (contentType.startsWith('image/svg')) {\n const [encoding, byteString] = urlParts[1].split(',')\n if (encoding === 'base64') {\n const newSvg = await resolveDynamicSvg(byteString)\n const newSvgDataUrl = createDataUrl(Buffer.from(newSvg), contentType)\n cookedDataBuffer = ImageThumbnailWitness.bufferFromDataUrl(newSvgDataUrl) ?? dataBuffer\n }\n }\n result = await this.processMedia(\n cookedDataBuffer,\n {\n schema: ImageThumbnailSchema,\n sourceUrl: url,\n },\n contentType,\n )\n }\n } else {\n // if it is ipfs, go through cloud flair\n const mutatedUrl = checkIpfsUrl(url, this.ipfsGateway)\n result = await this.fromHttp(mutatedUrl, url)\n }\n return result\n }),\n ),\n )\n }\n return this.config.runExclusive ? await this._semaphore.runExclusive(() => process()) : process()\n }\n\n private async createThumbnailDataUrl(sourceBuffer: ArrayBuffer, encoding?: ImageThumbnailEncoding) {\n const thumb = await new Promise<Buffer>((resolve, reject) => {\n gm(Buffer.from(sourceBuffer))\n .quality(this.quality)\n .resize(this.width, this.height)\n .flatten()\n .toBuffer(encoding ?? this.encoding, (error, buffer) => {\n if (error) {\n reject(error)\n } else {\n resolve(buffer)\n }\n })\n })\n return createDataUrl(thumb, 'image/png')\n }\n\n /**\n * Creates an image thumbnail from a video.\n * @param videoBuffer The input video buffer.\n * @returns An buffer containing an image thumbnail for the video.\n */\n private async createThumbnailFromVideo(videoBuffer: ArrayBuffer) {\n const imageBuffer = await getVideoFrameAsImageFluent(videoBuffer)\n return this.createThumbnailDataUrl(imageBuffer)\n }\n\n // eslint-disable-next-line complexity\n private async fromHttp(url: string, sourceUrl?: string): Promise<ImageThumbnail> {\n let response: AxiosResponse\n let dnsResult: string[]\n try {\n const urlObj = new Url(url)\n dnsResult = await dnsPromises.resolve(urlObj.host)\n } catch (ex) {\n const error = ex as DnsError\n const result: ImageThumbnail = {\n http: {\n code: error.code,\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n return result\n }\n try {\n response = await axios.get(url, {\n responseType: 'arraybuffer',\n })\n } catch (ex) {\n const axiosError = ex as AxiosError\n if (axiosError.isAxiosError) {\n // selectively pick fields from AxiosError\n const result: ImageThumbnail = {\n http: {\n ipAddress: dnsResult[0],\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n if (axiosError?.response?.status !== undefined) {\n result.http = result.http ?? {}\n result.http.status = axiosError?.response?.status\n }\n if (axiosError?.code !== undefined) {\n result.http = result.http ?? {}\n result.http.code = axiosError?.code\n }\n return result\n } else {\n throw ex\n }\n }\n\n const result: ImageThumbnail = {\n http: {\n status: response.status,\n },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n\n if (response.status >= 200 && response.status < 300) {\n const contentType: string | undefined = response.headers['content-type']?.toString()\n const sourceBuffer = Buffer.from(response.data, 'binary')\n\n return this.processMedia(sourceBuffer, result, contentType)\n }\n return result\n }\n\n private async processMedia(sourceBuffer: ArrayBuffer, imageThumbnail: ImageThumbnail, contentType?: string): Promise<ImageThumbnail> {\n const [mediaType, fileType] = contentType?.split('/') ?? ['', '']\n imageThumbnail.mime = imageThumbnail.mime ?? {}\n imageThumbnail.mime.returned = mediaType\n\n try {\n imageThumbnail.mime.detected = await FileType.fromBuffer(sourceBuffer)\n } catch (ex) {\n const error = ex as Error\n this.logger?.error(`FileType error: ${error.message}`)\n }\n\n const processImage = async (encoding?: ImageThumbnailEncoding) => {\n imageThumbnail.sourceHash = await ImageThumbnailWitness.binaryToSha256(sourceBuffer)\n imageThumbnail.url = await this.createThumbnailDataUrl(sourceBuffer, encoding)\n }\n\n const processVideo = async () => {\n // Gracefully handle the case where ffmpeg is not installed.\n\n if (hasbin.sync('ffmpeg')) {\n imageThumbnail.sourceHash = await ImageThumbnailWitness.binaryToSha256(sourceBuffer)\n imageThumbnail.url = await this.createThumbnailFromVideo(sourceBuffer)\n } else {\n imageThumbnail.mime = imageThumbnail.mime ?? {}\n imageThumbnail.mime.invalid = true\n }\n }\n\n let encoding: ImageThumbnailEncoding = 'PNG'\n\n switch (fileType.toUpperCase()) {\n case 'GIF': {\n encoding = 'GIF'\n break\n }\n case 'JPG':\n case 'JPEG': {\n encoding = 'JPG'\n break\n }\n }\n\n switch (mediaType) {\n case 'image': {\n await processImage(encoding)\n imageThumbnail.mime.type = mediaType\n break\n }\n case 'video': {\n await processVideo()\n imageThumbnail.mime.type = mediaType\n break\n }\n default: {\n const [detectedMediaType] = imageThumbnail.mime.detected?.mime?.split('/') ?? ['', '']\n switch (detectedMediaType) {\n case 'image': {\n await processImage()\n imageThumbnail.mime.type = imageThumbnail.mime.detected?.mime\n break\n }\n case 'video': {\n await processVideo()\n imageThumbnail.mime.type = imageThumbnail.mime.detected?.mime\n break\n }\n default: {\n imageThumbnail.mime.invalid = true\n break\n }\n }\n break\n }\n }\n return imageThumbnail\n }\n}\n","import { unlink, writeFile } from 'node:fs/promises'\nimport { tmpdir } from 'node:os'\nimport { Writable, WritableOptions } from 'node:stream'\n\nimport ffmpeg from 'fluent-ffmpeg'\nimport { v4 as uuid } from 'uuid'\n\n/**\n * A Writable stream that collects output from ffmpeg.\n */\nclass FfmpegOutputStream extends Writable {\n private readonly chunks: Uint8Array[] = []\n\n constructor(options?: WritableOptions) {\n super(options)\n }\n\n override _write(chunk: never, _encoding: BufferEncoding, callback: (error?: Error | null) => void): void {\n this.chunks.push(chunk)\n callback()\n }\n\n /**\n * Collects the output from ffmpeg into a buffer.\n * @returns A buffer containing the concatenated\n * output from ffmpeg.\n */\n toBuffer = () => Buffer.concat(this.chunks)\n}\n\n/**\n * Execute FFmpeg using fluent API with provided input buffer and video thumbnail image.\n * @param videoBuffer Input video buffer.\n * @returns Output buffer containing the video thumbnail image.\n */\nexport const getVideoFrameAsImageFluent = async (videoBuffer: ArrayBuffer) => {\n // Get a temp file name\n const tmpFile = `/${tmpdir()}/${uuid()}`\n try {\n // Write videoBuffer to temp file for use as input to ffmpeg to\n // avoid issues with ffmpeg inferring premature EOF from buffer\n // passed via stdin (happens when ffmpeg is trying to infer\n // input video format)\n await writeFile(tmpFile, new Uint8Array(videoBuffer), { encoding: 'binary' })\n const imageBuffer = await new Promise<Buffer>((resolve, reject) => {\n // Create a Writable stream to collect PNG output from ffmpeg\n const ffmpegOutput = new FfmpegOutputStream()\n // Execute ffmpeg using fluent API\n ffmpeg()\n // NOTE: Uncomment to debug CLI args to ffmpeg\n // .on('start', (commandLine) => console.log('Spawned Ffmpeg with command: ' + commandLine))\n .on('error', err => reject(err.message))\n // Listen for the 'end' event to combine the output into a buffer holding the PNG image\n .on('end', () => resolve(ffmpegOutput.toBuffer()))\n .input(tmpFile) // Use temp file as input\n .takeFrames(1) // Only take 1st video frame\n .withNoAudio() // Don't include audio\n .outputOptions('-f image2pipe') // Write output to stdout\n .videoCodec('png') // Force PNG output\n // Start processing and direct ffmpeg stdout to writable stream\n .pipe(ffmpegOutput)\n })\n return imageBuffer\n } finally {\n // Cleanup temp file\n try {\n await unlink(tmpFile)\n } catch {\n // No error here since file doesn't exist\n }\n }\n}\n","import { assertEx } from '@xylabs/assert'\n\nconst allowIpfsIoRepair = true\n\n/**\n * Returns the equivalent IPFS gateway URL for the supplied URL.\n * @param urlToCheck The URL to check\n * @returns If the supplied URL is an IPFS URL, it converts the URL to the\n * equivalent IPFS gateway URL. Otherwise, returns the original URL.\n */\nexport const checkIpfsUrl = (urlToCheck: string, ipfsGateway?: string): string => {\n try {\n const url = new URL(urlToCheck)\n let protocol = url.protocol\n let host = url.host\n let path = url.pathname\n const query = url.search\n if (protocol === 'ipfs:') {\n protocol = 'https:'\n host = assertEx(ipfsGateway, () => 'No ipfsGateway provided')\n path = url.host === 'ipfs' ? `ipfs${path}` : `ipfs/${url.host}${path}`\n const root = `${protocol}//${host}/${path}`\n return query?.length > 0 ? `${root}?${query}` : root\n } else if (allowIpfsIoRepair && protocol === 'https' && host === 'ipfs.io') {\n protocol = 'https:'\n host = assertEx(ipfsGateway, () => 'No ipfsGateway provided')\n const pathParts = path.split('/')\n if (pathParts[0] === 'ipfs') {\n pathParts.shift()\n }\n path = pathParts.join('/')\n const root = `${protocol}//${host}/${path}`\n return query?.length > 0 ? `${root}?${query}` : root\n } else {\n return urlToCheck\n }\n } catch {\n // const error = ex as Error\n // console.error(`${error.name}:${error.message} [${urlToCheck}]`)\n // console.log(error.stack)\n return urlToCheck\n }\n}\n","import { fromByteArray } from 'base64-js'\n\nexport const createDataUrl = (data: ArrayBuffer, contextType: string, encoding: 'base64' = 'base64') => {\n return `data:${contextType};${encoding},${fromByteArray(new Uint8Array(data))}`\n}\n","import axios, { AxiosResponse } from 'axios'\nimport { toByteArray } from 'base64-js'\nimport { Builder, parseStringPromise } from 'xml2js'\n\nexport const resolveDynamicSvg = async (base64Bytes: string) => {\n const decoder = new TextDecoder()\n const bytes = toByteArray(base64Bytes)\n const svg = decoder.decode(bytes)\n const svgObj = await parseStringPromise(svg)\n const svgNode = svgObj['svg']\n const imageResults = (await Promise.all(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n svgNode['image'].map(async (img: any) => [\n img.$,\n await axios.get(img.$.href, {\n responseType: 'arraybuffer',\n }),\n ]),\n )) as [string, AxiosResponse][]\n const image = imageResults.map(([href, response]) => {\n if (response.data) {\n const sourceBuffer = Buffer.from(response.data, 'binary')\n return { $: { href: `data:${response.headers['content-type']?.toString()};base64,${sourceBuffer.toString('base64')}` } }\n } else {\n return { $: { href } }\n }\n })\n const updatedSVG = { ...svgObj, svg: { ...svgNode, image } }\n const builder = new Builder()\n return builder.buildObject(updatedSVG)\n}\n","// eslint-disable-next-line import/no-default-export\nexport { ImageThumbnailPlugin as default, ImageThumbnailPlugin } from './Plugin.ts'\nexport * from './Witness/index.ts'\nexport * from '@xyo-network/diviner-image-thumbnail'\n"],"mappings":"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","names":["ImageThumbnailSchema","Buffer","ImageThumbnailSchema","axios","ImageThumbnailSchema","Buffer","result","axios","encoding","ImageThumbnailSchema"]}
package/package.json CHANGED
@@ -10,19 +10,19 @@
10
10
  "url": "https://github.com/XYOracleNetwork/plugins/issues"
11
11
  },
12
12
  "dependencies": {
13
- "@xylabs/assert": "^3.6.8",
14
- "@xylabs/axios": "^3.6.8",
15
- "@xylabs/lodash": "^3.6.8",
16
- "@xyo-network/abstract-witness": "^2.111.2",
17
- "@xyo-network/diviner-image-thumbnail": "^2.99.5",
18
- "@xyo-network/hash": "^2.111.2",
19
- "@xyo-network/image-thumbnail-payload-plugin": "^2.99.5",
20
- "@xyo-network/module-model": "^2.111.2",
21
- "@xyo-network/payload-model": "^2.111.2",
22
- "@xyo-network/payloadset-plugin": "^2.111.2",
23
- "@xyo-network/url-payload-plugin": "^2.99.5",
24
- "@xyo-network/witness-model": "^2.111.2",
13
+ "@xylabs/assert": "^3.6.12",
14
+ "@xylabs/lodash": "^3.6.12",
15
+ "@xyo-network/abstract-witness": "^2.111.3",
16
+ "@xyo-network/diviner-image-thumbnail": "^2.99.6",
17
+ "@xyo-network/hash": "^2.111.3",
18
+ "@xyo-network/image-thumbnail-payload-plugin": "^2.99.6",
19
+ "@xyo-network/module-model": "^2.111.3",
20
+ "@xyo-network/payload-model": "^2.111.3",
21
+ "@xyo-network/payloadset-plugin": "^2.111.3",
22
+ "@xyo-network/url-payload-plugin": "^2.99.6",
23
+ "@xyo-network/witness-model": "^2.111.3",
25
24
  "async-mutex": "^0.5.0",
25
+ "axios": "^1.7.3",
26
26
  "base64-js": "^1.5.1",
27
27
  "file-type": "^16.5.4",
28
28
  "fluent-ffmpeg": "^2.1.3",
@@ -42,16 +42,16 @@
42
42
  "@types/url-parse": "^1.4.11",
43
43
  "@types/uuid": "^10.0.0",
44
44
  "@types/xml2js": "^0.4.14",
45
- "@xylabs/delay": "^3.6.8",
46
- "@xylabs/ts-scripts-yarn3": "^3.15.13",
47
- "@xylabs/tsconfig": "^3.15.13",
48
- "@xyo-network/account": "^2.111.2",
49
- "@xyo-network/archivist-memory": "^2.111.2",
50
- "@xyo-network/node-memory": "^2.111.2",
51
- "@xyo-network/payload-builder": "^2.111.2",
52
- "@xyo-network/sentinel-memory": "^2.111.2",
53
- "@xyo-network/sentinel-wrapper": "^2.111.2",
54
- "@xyo-network/witness-timestamp": "^2.111.2",
45
+ "@xylabs/delay": "^3.6.12",
46
+ "@xylabs/ts-scripts-yarn3": "^3.15.14",
47
+ "@xylabs/tsconfig": "^3.15.14",
48
+ "@xyo-network/account": "^2.111.3",
49
+ "@xyo-network/archivist-memory": "^2.111.3",
50
+ "@xyo-network/node-memory": "^2.111.3",
51
+ "@xyo-network/payload-builder": "^2.111.3",
52
+ "@xyo-network/sentinel-memory": "^2.111.3",
53
+ "@xyo-network/sentinel-wrapper": "^2.111.3",
54
+ "@xyo-network/witness-timestamp": "^2.111.3",
55
55
  "jest": "^29.7.0",
56
56
  "jest-mock-extended": "^3.0.7",
57
57
  "typescript": "^5.5.4"
@@ -85,6 +85,6 @@
85
85
  "url": "https://github.com/XYOracleNetwork/plugins.git"
86
86
  },
87
87
  "sideEffects": false,
88
- "version": "2.99.5",
88
+ "version": "2.99.6",
89
89
  "type": "module"
90
90
  }
@@ -2,7 +2,6 @@
2
2
  import { Buffer } from 'node:buffer'
3
3
  import { promises as dnsPromises } from 'node:dns'
4
4
 
5
- import { axios, AxiosError, AxiosResponse } from '@xylabs/axios'
6
5
  import { compact } from '@xylabs/lodash'
7
6
  import { AbstractWitness } from '@xyo-network/abstract-witness'
8
7
  import { PayloadHasher } from '@xyo-network/hash'
@@ -10,6 +9,8 @@ import { ImageThumbnail, ImageThumbnailSchema } from '@xyo-network/image-thumbna
10
9
  import { Schema } from '@xyo-network/payload-model'
11
10
  import { UrlPayload, UrlSchema } from '@xyo-network/url-payload-plugin'
12
11
  import { Semaphore } from 'async-mutex'
12
+ import type { AxiosError, AxiosResponse } from 'axios'
13
+ import axios from 'axios'
13
14
  import FileType from 'file-type'
14
15
  import graphicsMagick from 'gm'
15
16
  import hasbin from 'hasbin'
@@ -1,4 +1,4 @@
1
- import { axios, AxiosResponse } from '@xylabs/axios'
1
+ import axios, { AxiosResponse } from 'axios'
2
2
  import { toByteArray } from 'base64-js'
3
3
  import { Builder, parseStringPromise } from 'xml2js'
4
4