@xyo-network/image-thumbnail-plugin 5.1.3 → 5.1.5

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 +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 type { 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 { exists } from '@xylabs/exists'\nimport { AbstractWitness } from '@xyo-network/abstract-witness'\nimport { ObjectHasher } from '@xyo-network/hash'\nimport type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport type { Schema } from '@xyo-network/payload-model'\nimport type { UrlPayload } from '@xyo-network/url-payload-plugin'\nimport { UrlSchema } from '@xyo-network/url-payload-plugin'\nimport { Semaphore } from 'async-mutex'\nimport type { AxiosError, AxiosResponse } from 'axios'\nimport axios from 'axios'\nimport { fileTypeFromBuffer } 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 type { ImageThumbnailEncoding } from './Config.ts'\nimport { ImageThumbnailWitnessConfigSchema } from './Config.ts'\nimport { getVideoFrameAsImageFluent } from './ffmpeg/index.ts'\nimport {\n checkIpfsUrl, createDataUrl, resolveDynamicSvg,\n} from './lib/index.ts'\nimport type { 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: ArrayBufferLike) {\n const viewData = new Uint8Array(data)\n await ObjectHasher.wasmInitialized\n if (ObjectHasher.wasmSupport.canUseWasm) {\n try {\n return await sha256(viewData)\n } catch {\n ObjectHasher.wasmSupport.allowWasm = false\n }\n }\n\n return shajs('sha256').update(viewData).digest().toString()\n }\n\n private static bufferFromDataUrl(url: string): ArrayBufferLike | 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).buffer\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 (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).buffer, 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 )).filter(exists)\n }\n return this.config.runExclusive ? await this._semaphore.runExclusive(() => process()) : process()\n }\n\n private async createThumbnailDataUrl(sourceBuffer: ArrayBufferLike, 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.buffer, '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: ArrayBufferLike) {\n const imageBuffer = await getVideoFrameAsImageFluent(videoBuffer)\n return this.createThumbnailDataUrl(imageBuffer.buffer)\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: { code: error.code },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n return result\n }\n try {\n response = await axios.get(url, { responseType: 'arraybuffer' })\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: { ipAddress: dnsResult[0] },\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: { status: response.status },\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').buffer\n\n return this.processMedia(sourceBuffer, result, contentType)\n }\n return result\n }\n\n private async processMedia(sourceBuffer: ArrayBufferLike, 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 fileTypeFromBuffer(sourceBuffer as ArrayBuffer)\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 type { WritableOptions } from 'node:stream'\nimport { Writable } 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: ArrayBufferLike) => {\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: ArrayBufferLike, contextType: string, encoding: 'base64' = 'base64') => {\n return `data:${contextType};${encoding},${fromByteArray(new Uint8Array(data))}`\n}\n","import type { AxiosResponse } from 'axios'\nimport axios 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, { responseType: 'arraybuffer' }),\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","export { 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"]}
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 type { 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 { exists } from '@xylabs/exists'\nimport { AbstractWitness } from '@xyo-network/abstract-witness'\nimport { ObjectHasher } from '@xyo-network/hash'\nimport type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'\nimport { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'\nimport type { Schema } from '@xyo-network/payload-model'\nimport type { UrlPayload } from '@xyo-network/url-payload-plugin'\nimport { UrlSchema } from '@xyo-network/url-payload-plugin'\nimport { Semaphore } from 'async-mutex'\nimport type { AxiosError, AxiosResponse } from 'axios'\nimport axios from 'axios'\nimport { fileTypeFromBuffer } 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 type { ImageThumbnailEncoding } from './Config.ts'\nimport { ImageThumbnailWitnessConfigSchema } from './Config.ts'\nimport { getVideoFrameAsImageFluent } from './ffmpeg/index.ts'\nimport {\n checkIpfsUrl, createDataUrl, resolveDynamicSvg,\n} from './lib/index.ts'\nimport type { 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: ArrayBufferLike) {\n const viewData = new Uint8Array(data)\n await ObjectHasher.wasmInitialized\n if (ObjectHasher.wasmSupport.canUseWasm) {\n try {\n return await sha256(viewData)\n } catch {\n ObjectHasher.wasmSupport.allowWasm = false\n }\n }\n\n return shajs('sha256').update(viewData).digest().toString()\n }\n\n private static bufferFromDataUrl(url: string): ArrayBufferLike | 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).buffer\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 (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).buffer, 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 )).filter(exists)\n }\n return this.config.runExclusive ? await this._semaphore.runExclusive(() => process()) : process()\n }\n\n private async createThumbnailDataUrl(sourceBuffer: ArrayBufferLike, 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.buffer, '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: ArrayBufferLike) {\n const imageBuffer = await getVideoFrameAsImageFluent(videoBuffer)\n return this.createThumbnailDataUrl(imageBuffer.buffer)\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: { code: error.code },\n schema: ImageThumbnailSchema,\n sourceUrl: sourceUrl ?? url,\n }\n return result\n }\n try {\n response = await axios.get(url, { responseType: 'arraybuffer' })\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: { ipAddress: dnsResult[0] },\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: { status: response.status },\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').buffer\n\n return this.processMedia(sourceBuffer, result, contentType)\n }\n return result\n }\n\n private async processMedia(sourceBuffer: ArrayBufferLike, 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 fileTypeFromBuffer(sourceBuffer as ArrayBuffer)\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 type { WritableOptions } from 'node:stream'\nimport { Writable } 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: ArrayBufferLike) => {\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 // 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: ArrayBufferLike, contextType: string, encoding: 'base64' = 'base64') => {\n return `data:${contextType};${encoding},${fromByteArray(new Uint8Array(data))}`\n}\n","import type { AxiosResponse } from 'axios'\nimport axios 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, { responseType: 'arraybuffer' }),\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","export { 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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@xyo-network/image-thumbnail-plugin",
3
- "version": "5.1.3",
3
+ "version": "5.1.5",
4
4
  "description": "Typescript/Javascript Plugins for XYO Platform",
5
5
  "homepage": "https://xyo.network",
6
6
  "bugs": {
@@ -32,54 +32,57 @@
32
32
  "types": "./dist/node/index.d.ts",
33
33
  "files": [
34
34
  "dist",
35
- "src"
35
+ "src",
36
+ "!**/*.bench.*",
37
+ "!**/*.spec.*",
38
+ "!**/*.test.*"
36
39
  ],
37
40
  "dependencies": {
38
- "@xylabs/assert": "~5.0.11",
39
- "@xylabs/exists": "~5.0.11",
40
- "@xyo-network/abstract-witness": "~5.1.2",
41
- "@xyo-network/diviner-image-thumbnail": "~5.1.3",
42
- "@xyo-network/hash": "~5.1.2",
43
- "@xyo-network/image-thumbnail-payload-plugin": "~5.1.3",
44
- "@xyo-network/module-model": "~5.1.2",
45
- "@xyo-network/payload-model": "~5.1.2",
46
- "@xyo-network/payloadset-plugin": "~5.1.2",
47
- "@xyo-network/url-payload-plugin": "~5.1.3",
48
- "@xyo-network/witness-model": "~5.1.2",
41
+ "@xylabs/assert": "~5.0.37",
42
+ "@xylabs/exists": "~5.0.37",
43
+ "@xyo-network/abstract-witness": "~5.1.23",
44
+ "@xyo-network/diviner-image-thumbnail": "~5.1.5",
45
+ "@xyo-network/hash": "~5.1.23",
46
+ "@xyo-network/image-thumbnail-payload-plugin": "~5.1.5",
47
+ "@xyo-network/module-model": "~5.1.23",
48
+ "@xyo-network/payload-model": "~5.1.23",
49
+ "@xyo-network/payloadset-plugin": "~5.1.23",
50
+ "@xyo-network/url-payload-plugin": "~5.1.5",
51
+ "@xyo-network/witness-model": "~5.1.23",
49
52
  "async-mutex": "~0.5.0",
50
- "axios": "~1.11.0",
53
+ "axios": "~1.13.2",
51
54
  "base64-js": "~1.5.1",
52
- "file-type": "~21.0.0",
55
+ "file-type": "~21.1.0",
53
56
  "fluent-ffmpeg": "~2.1.3",
54
57
  "gm": "~1.25.1",
55
58
  "hasbin": "~1.2.3",
56
59
  "hash-wasm": "~4.12.0",
57
60
  "sha.js": "~2.4.12",
58
61
  "url-parse": "~1.5.10",
59
- "uuid": "~11.1.0",
62
+ "uuid": "~13.0.0",
60
63
  "xml2js": "~0.6.2"
61
64
  },
62
65
  "devDependencies": {
63
- "@types/fluent-ffmpeg": "~2.1.27",
66
+ "@types/fluent-ffmpeg": "~2.1.28",
64
67
  "@types/gm": "~1.25.4",
65
68
  "@types/hasbin": "~1.2.2",
66
69
  "@types/sha.js": "~2.4.4",
67
70
  "@types/url-parse": "~1.4.11",
68
- "@types/uuid": "~10.0.0",
71
+ "@types/uuid": "~11.0.0",
69
72
  "@types/xml2js": "~0.4.14",
70
- "@xylabs/delay": "~5.0.11",
71
- "@xylabs/ts-scripts-yarn3": "~7.1.7",
72
- "@xylabs/tsconfig": "~7.1.7",
73
- "@xylabs/vitest-extended": "~5.0.11",
74
- "@xyo-network/account": "~5.1.2",
75
- "@xyo-network/archivist-memory": "~5.1.2",
76
- "@xyo-network/node-memory": "~5.1.2",
77
- "@xyo-network/payload-builder": "~5.1.2",
78
- "@xyo-network/sentinel-memory": "~5.1.2",
79
- "@xyo-network/sentinel-wrapper": "~5.1.2",
80
- "@xyo-network/witness-timestamp": "~5.1.2",
81
- "typescript": "~5.9.2",
82
- "vitest": "~3.2.4"
73
+ "@xylabs/delay": "~5.0.37",
74
+ "@xylabs/ts-scripts-yarn3": "~7.2.8",
75
+ "@xylabs/tsconfig": "~7.2.8",
76
+ "@xylabs/vitest-extended": "~5.0.37",
77
+ "@xyo-network/account": "~5.1.23",
78
+ "@xyo-network/archivist-memory": "~5.1.23",
79
+ "@xyo-network/node-memory": "~5.1.23",
80
+ "@xyo-network/payload-builder": "~5.1.23",
81
+ "@xyo-network/sentinel-memory": "~5.1.23",
82
+ "@xyo-network/sentinel-wrapper": "~5.1.23",
83
+ "@xyo-network/witness-timestamp": "~5.1.23",
84
+ "typescript": "~5.9.3",
85
+ "vitest": "~4.0.10"
83
86
  },
84
87
  "publishConfig": {
85
88
  "access": "public"
@@ -48,7 +48,7 @@ export const getVideoFrameAsImageFluent = async (videoBuffer: ArrayBufferLike) =
48
48
  const ffmpegOutput = new FfmpegOutputStream()
49
49
  // Execute ffmpeg using fluent API
50
50
  ffmpeg()
51
- // NOTE: Uncomment to debug CLI args to ffmpeg
51
+ // Uncomment to debug CLI args to ffmpeg
52
52
  // .on('start', (commandLine) => console.log('Spawned Ffmpeg with command: ' + commandLine))
53
53
  .on('error', err => reject(err.message))
54
54
  // Listen for the 'end' event to combine the output into a buffer holding the PNG image
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.data.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.data.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.data.spec.ts"],"names":[],"mappings":"AACA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.dynamic-svg.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.dynamic-svg.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.dynamic-svg.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.https.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.https.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.https.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.ipfs.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.ipfs.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.ipfs.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.sentinel.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.sentinel.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.sentinel.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,2 +0,0 @@
1
- import '@xylabs/vitest-extended';
2
- //# sourceMappingURL=Witness.video.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"Witness.video.spec.d.ts","sourceRoot":"","sources":["../../../../src/Witness/spec/Witness.video.spec.ts"],"names":[],"mappings":"AAAA,OAAO,yBAAyB,CAAA"}
@@ -1,47 +0,0 @@
1
- /* eslint-disable @stylistic/max-len */
2
- import '@xylabs/vitest-extended'
3
-
4
- import type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
5
- import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'
6
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
7
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
8
- import hasbin from 'hasbin'
9
- import {
10
- describe, expect,
11
- it,
12
- } from 'vitest'
13
-
14
- import { ImageThumbnailWitness } from '../Witness.ts'
15
-
16
- const testIfHasBin = (bin: string) => (hasbin.sync(bin) ? it : it.skip)
17
-
18
- /**
19
- * @group thumbnail
20
- */
21
-
22
- describe('ImageThumbnailWitness', () => {
23
- testIfHasBin('magick')('DATA [medium/png]', async () => {
24
- const witness = await ImageThumbnailWitness.create({ account: 'random' })
25
- const httpsPayload: UrlPayload = {
26
- schema: UrlSchema,
27
- url: 'data:image/jpeg;base64,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',
28
- }
29
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
30
- expect(result.length).toBe(1)
31
- // console.log(`DATA/PNG Size: ${result[0].url?.length}}`)
32
- expect(result[0].url?.length).toBeLessThan(64_000)
33
- expect(result[0].schema).toBe(ImageThumbnailSchema)
34
- })
35
-
36
- testIfHasBin('magick')('DATA [medium/svg]', async () => {
37
- const witness = await ImageThumbnailWitness.create({ account: 'random' })
38
- const httpsPayload: UrlPayload = {
39
- schema: UrlSchema,
40
- url: 'data:image/svg+xml;utf8, <svg viewBox='0 0 512 512' preserveAspectRatio='xMidYMid meet' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink'><image xlink:href='data:image/png;base64,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'/></svg>',
41
- }
42
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
43
- expect(result.length).toBe(1)
44
- expect(result[0].url?.length).toBeLessThan(80_858)
45
- expect(result[0].schema).toBe(ImageThumbnailSchema)
46
- })
47
- })
@@ -1,35 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
4
- import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'
5
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
6
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
7
- import hasbin from 'hasbin'
8
- import {
9
- describe, expect,
10
- it,
11
- } from 'vitest'
12
-
13
- import { ImageThumbnailWitness } from '../Witness.ts'
14
-
15
- const testIfHasBin = (bin: string) => (hasbin.sync(bin) ? it : it.skip)
16
-
17
- /**
18
- * @group thumbnail
19
- */
20
-
21
- describe('ImageThumbnailWitness', () => {
22
- testIfHasBin('magick')('Dynamic SVG [medium/png]', async () => {
23
- const witness = await ImageThumbnailWitness.create({ account: 'random' })
24
- const httpsPayload: UrlPayload = {
25
- schema: UrlSchema,
26
- // eslint-disable-next-line @stylistic/max-len
27
- url: 'data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHByZXNlcnZlQXNwZWN0UmF0aW89InhNaW5ZTWluIG1lZXQiIHZpZXdCb3g9IjAgMCAxMjAwIDEyMDAiPjxpbWFnZSBocmVmPSJodHRwczovL25lb3Rva3lvLm15cGluYXRhLmNsb3VkL2lwZnMvUW1aeGhEd0xjb0s3Y2lwWDNZMXFNRXBXVUhFeE43RjJqQ3o3UU5mR2NleFVFdS9iYWNrZ3JvdW5kLzgucG5nIi8+PGltYWdlIGhyZWY9Imh0dHBzOi8vbmVvdG9reW8ubXlwaW5hdGEuY2xvdWQvaXBmcy9RbVp4aER3TGNvSzdjaXBYM1kxcU1FcFdVSEV4TjdGMmpDejdRTmZHY2V4VUV1L2JvZHkvMy5wbmciLz48aW1hZ2UgaHJlZj0iaHR0cHM6Ly9uZW90b2t5by5teXBpbmF0YS5jbG91ZC9pcGZzL1FtWnhoRHdMY29LN2NpcFgzWTFxTUVwV1VIRXhON0YyakN6N1FOZkdjZXhVRXUvY2xvdGgvNC5wbmciLz48aW1hZ2UgaHJlZj0iaHR0cHM6Ly9uZW90b2t5by5teXBpbmF0YS5jbG91ZC9pcGZzL1FtWnhoRHdMY29LN2NpcFgzWTFxTUVwV1VIRXhON0YyakN6N1FOZkdjZXhVRXUvaGFuZC9maXN0LzMucG5nIi8+PGltYWdlIGhyZWY9Imh0dHBzOi8vbmVvdG9reW8ubXlwaW5hdGEuY2xvdWQvaXBmcy9RbVp4aER3TGNvSzdjaXBYM1kxcU1FcFdVSEV4TjdGMmpDejdRTmZHY2V4VUV1L2hlYWQvMy5wbmciLz48aW1hZ2UgaHJlZj0iaHR0cHM6Ly9uZW90b2t5by5teXBpbmF0YS5jbG91ZC9pcGZzL1FtWnhoRHdMY29LN2NpcFgzWTFxTUVwV1VIRXhON0YyakN6N1FOZkdjZXhVRXUvaGVsbS8yLnBuZyIvPjwvc3ZnPg==',
28
- }
29
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
30
- expect(result.length).toBe(1)
31
- // console.log(`DATA/PNG Size: ${result[0].url?.length}}`)
32
- expect(result[0].url?.length).toBeLessThan(128_000)
33
- expect(result[0].schema).toBe(ImageThumbnailSchema)
34
- }, 20_000)
35
- })
@@ -1,137 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import { removeEmptyFields } from '@xyo-network/hash'
4
- import type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
5
- import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'
6
- import { PayloadBuilder } from '@xyo-network/payload-builder'
7
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
8
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
9
- import hasbin from 'hasbin'
10
- import {
11
- beforeAll,
12
- describe, expect, it,
13
- } from 'vitest'
14
-
15
- import { ImageThumbnailWitness } from '../Witness.ts'
16
-
17
- const describeIfHasBin = (bin: string) => (hasbin.sync(bin) ? describe : describe.skip)
18
-
19
- /**
20
- * @group thumbnail
21
- */
22
-
23
- describeIfHasBin('magick')('ImageThumbnailWitness', () => {
24
- let witness: ImageThumbnailWitness
25
- beforeAll(async () => {
26
- witness = await ImageThumbnailWitness.create({ account: 'random' })
27
- })
28
- it('HTTPS [medium/avif]', async () => {
29
- const httpsPayload: UrlPayload = {
30
- schema: UrlSchema,
31
- url: 'https://i.seadn.io/gae/sOGk14HmHMajXRrra4X7Y1ZWncAPyidZap5StDRFCtKHHNSiIUNMpa12v4wqmyp1lEe4CxSxpWgpfiIdh-b_Mn3fq9LDM68i9gof9w?w=500&auto=format',
32
- }
33
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
34
- expect(result.length).toBe(1)
35
- expect(result[0].url?.length).toBeLessThan(64_000)
36
- expect(result[0].schema).toBe(ImageThumbnailSchema)
37
- }, 20_000)
38
- it.skip('HTTPS [medium/png/unsafe]', async () => {
39
- const httpsPayload: UrlPayload = {
40
- schema: UrlSchema,
41
- url: 'https://ethercb.com/image.png',
42
- }
43
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
44
- expect(result.length).toBe(1)
45
- expect(result[0].schema).toBe(ImageThumbnailSchema)
46
- expect(result[0].url?.length).toBeLessThan(64_000)
47
- expect(result[0].schema).toBe(ImageThumbnailSchema)
48
- }, 20_000)
49
- it('HTTPS [medium/svg]', async () => {
50
- const httpsPayload: UrlPayload = {
51
- schema: UrlSchema,
52
- url: 'https://dev.w3.org/SVG/tools/svgweb/samples/svg-files/AJ_Digital_Camera.svg',
53
- }
54
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
55
- expect(result.length).toBe(1)
56
- expect(result[0].url?.length).toBeLessThan(64_000)
57
-
58
- // do a second pass and make sure we get cached result
59
- const result2 = (await witness.observe([httpsPayload])) as ImageThumbnail[]
60
- expect(result2.length).toBe(1)
61
- expect(result2[0].url?.length).toEqual(result[0].url?.length)
62
- expect(result[0].schema).toBe(ImageThumbnailSchema)
63
- }, 20_000)
64
- it('HTTPS [medium/ens]', async () => {
65
- const httpsPayload: UrlPayload = {
66
- schema: UrlSchema,
67
-
68
- url: 'https://metadata.ens.domains/mainnet/0x57f1887a8bf19b14fc0df6fd9b2acc9af147ea85/ens.eth/image',
69
- }
70
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
71
- console.log(`ENS-SourceHash: ${result[0].sourceHash}`)
72
- console.log(`ENS-Hash: ${await PayloadBuilder.hash(result[0])}`)
73
- console.log(`ENS-DataHash: ${await PayloadBuilder.dataHash(removeEmptyFields(result[0]))}`)
74
- console.log(`ENS-Result: ${JSON.stringify(result[0], null, 2)}`)
75
- expect(result.length).toBe(1)
76
- expect(result[0].url?.length).toBeLessThan(128_000)
77
-
78
- // do a second pass and make sure we get cached result
79
- const result2 = (await witness.observe([httpsPayload])) as ImageThumbnail[]
80
- expect(result2.length).toBe(1)
81
- expect(result2[0].url?.length).toEqual(result[0].url?.length)
82
- expect(result[0].schema).toBe(ImageThumbnailSchema)
83
- }, 20_000)
84
- it('HTTPS [large/gif (animated)]', async () => {
85
- const httpsPayload: UrlPayload = {
86
- schema: UrlSchema,
87
- url: 'https://lh3.googleusercontent.com/N3uFgyMt0xOew9YjD8GiOLQEbbQ2Y7WJOqoHdUdZZSljKrbuKNt6VGkAByzyPAI80y81tELH6tKatSZvFXKfcbBdm6GfCyZhFWxgOTw',
88
- }
89
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
90
- expect(result.length).toBe(1)
91
- expect(result[0].url?.length).toBeLessThan(64_000)
92
- expect(result[0].schema).toBe(ImageThumbnailSchema)
93
- }, 20_000)
94
- it('HTTPS [html/error]', async () => {
95
- const httpsPayload: UrlPayload = {
96
- schema: UrlSchema,
97
- url: 'https://espn.com',
98
- }
99
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
100
- expect(result.length).toBe(1)
101
- expect(result[0]?.mime?.invalid).toBe(true)
102
- }, 20_000)
103
- it('HTTPS [dns/error]', async () => {
104
- const httpsPayload: UrlPayload = {
105
- schema: UrlSchema,
106
- url: 'https://sdjkfsdljkfhdskfsd.com',
107
- }
108
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
109
- expect(result.length).toBe(1)
110
- expect(result[0]?.http?.code).toBe('ENOTFOUND')
111
- }, 20_000)
112
- it('HTTPS [other/error]', async () => {
113
- const httpsPayload: UrlPayload = {
114
- schema: UrlSchema,
115
- url: 'https://profilesetting.in/mier/logo.gif',
116
- }
117
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
118
- expect(result.length).toBe(1)
119
- console.log(`HTTPS [other/error]: ${JSON.stringify(result)}`)
120
- expect(result[0]?.http?.code).toBe('EPROTO')
121
- }, 20_000)
122
- it.skip('HTTPS [medium/png]', async () => {
123
- const httpsPayload: UrlPayload = {
124
- schema: UrlSchema,
125
- url: 'https://usdclive.org/usdc.png',
126
- }
127
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
128
- expect(result.length).toBe(1)
129
- expect(result[0].url?.length).toBeLessThan(64_000)
130
-
131
- // do a second pass and make sure we get cached result
132
- const result2 = (await witness.observe([httpsPayload])) as ImageThumbnail[]
133
- expect(result2.length).toBe(1)
134
- expect(result2[0].url?.length).toEqual(result[0].url?.length)
135
- expect(result[0].schema).toBe(ImageThumbnailSchema)
136
- }, 20_000)
137
- })
@@ -1,64 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
4
- import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'
5
- import type { ModuleError } from '@xyo-network/payload-model'
6
- import { ModuleErrorSchema } from '@xyo-network/payload-model'
7
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
8
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
9
- import hasbin from 'hasbin'
10
- import {
11
- beforeAll,
12
- describe, expect, it,
13
- } from 'vitest'
14
-
15
- import { ImageThumbnailWitness } from '../Witness.ts'
16
-
17
- const testIfHasBin = (bin: string) => (hasbin.sync(bin) ? it : it.skip)
18
-
19
- /**
20
- * @group thumbnail
21
- */
22
-
23
- describe('ImageThumbnailWitness', () => {
24
- let witness: ImageThumbnailWitness
25
- beforeAll(async () => {
26
- witness = await ImageThumbnailWitness.create({ account: 'random' })
27
- })
28
- testIfHasBin('magick')('IPFS [jpeg]', async () => {
29
- const ipfsPayload: UrlPayload = {
30
- schema: UrlSchema,
31
- url: 'ipfs://ipfs/QmewywDQGqz9WuWfT11ueSR3Mu86MejfD64v3KtFRzGP2G/image.jpeg',
32
- }
33
- const result = (await witness.observe([ipfsPayload])) as (ImageThumbnail | ModuleError)[]
34
- expect(result.length).toBe(1)
35
- const thumb = result[0] as ImageThumbnail
36
- expect(thumb.url?.length).toBeLessThan(64_000)
37
- const error = result[0] as ModuleError
38
- if (result[0].schema === ModuleErrorSchema) {
39
- console.log(`Error: ${error.message}`)
40
- }
41
- expect(result[0].schema).toBe(ImageThumbnailSchema)
42
- })
43
- testIfHasBin('magick')('IPFS [png]', async () => {
44
- const ipfsPayload: UrlPayload = {
45
- schema: UrlSchema,
46
- url: 'ipfs://bafybeie3vrrqcq7iugzmsdsdxscvmvqnfkqkk7if2ywxu5h436wf72usga/470.png',
47
- }
48
- const result = (await witness.observe([ipfsPayload])) as ImageThumbnail[]
49
- expect(result.length).toBe(1)
50
- const thumb = result[0] as ImageThumbnail
51
- expect(thumb.url?.length).toBeLessThan(64_000)
52
- expect(result[0].schema).toBe(ImageThumbnailSchema)
53
- })
54
- testIfHasBin('magick')('IPFS [bad (ipfs.io)]', async () => {
55
- const ipfsPayload: UrlPayload = {
56
- schema: UrlSchema,
57
- url: 'https://ipfs.io/ipfs/QmTkxZExjY97XSCTGoqGNxgYUE4kDakMykjbL1NVnH9xE9',
58
- }
59
- const result = (await witness.observe([ipfsPayload])) as ImageThumbnail[]
60
- expect(result.length).toBe(1)
61
- console.log(`result: ${JSON.stringify(result, null, 2)}`)
62
- expect(result[0].schema).toBe(ImageThumbnailSchema)
63
- })
64
- })
@@ -1,96 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import { delay } from '@xylabs/delay'
4
- import { Account } from '@xyo-network/account'
5
- import { MemoryArchivist } from '@xyo-network/archivist-memory'
6
- import { isImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
7
- import { MemoryNode } from '@xyo-network/node-memory'
8
- import { PayloadBuilder } from '@xyo-network/payload-builder'
9
- import { MemorySentinel } from '@xyo-network/sentinel-memory'
10
- import { SentinelWrapper } from '@xyo-network/sentinel-wrapper'
11
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
12
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
13
- import { isTimestamp, TimestampWitness } from '@xyo-network/witness-timestamp'
14
- import {
15
- beforeAll,
16
- describe, expect, it,
17
- } from 'vitest'
18
-
19
- import { ImageThumbnailWitness } from '../Witness.ts'
20
-
21
- /**
22
- * @group thumbnail
23
- * @group sentinel
24
- */
25
-
26
- describe('Witness', () => {
27
- describe('when behind sentinel', () => {
28
- const archivistName = 'archivist'
29
- let thumbnailWitness: ImageThumbnailWitness
30
- let timestampWitness: TimestampWitness
31
- let archivist: MemoryArchivist
32
- let sentinel: MemorySentinel
33
- let node: MemoryNode
34
- // const logger = mock<Console>()
35
-
36
- beforeAll(async () => {
37
- thumbnailWitness = await ImageThumbnailWitness.create({
38
- account: 'random',
39
- config: { schema: ImageThumbnailWitness.defaultConfigSchema },
40
- // logger,
41
- })
42
- timestampWitness = await TimestampWitness.create({
43
- account: 'random',
44
- config: { schema: TimestampWitness.defaultConfigSchema },
45
- // logger,
46
- })
47
- archivist = await MemoryArchivist.create({
48
- account: 'random',
49
- config: { name: archivistName, schema: MemoryArchivist.defaultConfigSchema },
50
- })
51
- sentinel = await MemorySentinel.create({
52
- account: 'random',
53
- config: {
54
- archiving: { archivists: [archivist.address] },
55
- schema: MemorySentinel.defaultConfigSchema,
56
- synchronous: true,
57
- tasks: [{ input: true, mod: thumbnailWitness.address }, { mod: timestampWitness.address }],
58
- },
59
- // logger,
60
- })
61
- const modules = [timestampWitness, thumbnailWitness, archivist, sentinel]
62
- node = await MemoryNode.create({
63
- account: 'random',
64
- config: { schema: MemoryNode.defaultConfigSchema },
65
- // logger,
66
- })
67
- await node.start()
68
- await Promise.all(
69
- modules.map(async (mod) => {
70
- await node.register(mod)
71
- await node.attach(mod.address, true)
72
- }),
73
- )
74
- })
75
- it('witnesses thumbnail for url', async () => {
76
- // TODO: Replace with data url for speed
77
- // const url =
78
- // eslint-disable-next-line @stylistic/max-len
79
- // "data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='100' height='100' viewBox='0 0 100 100'><circle cx='50' cy='50' r='48' fill='yellow' stroke='black' stroke-width='2'/><circle cx='35' cy='35' r='5' fill='black'/><circle cx='65' cy='35' r='5' fill='black'/><path d='M 35 70 Q 50 85, 65 70' fill='none' stroke='black' stroke-width='2'/></svg>"
80
- const url = 'https://placekitten.com/200/300'
81
- const query = new PayloadBuilder<UrlPayload>({ schema: UrlSchema }).fields({ url }).build()
82
- const sentinelWrapper = SentinelWrapper.wrap(sentinel, await Account.random())
83
- // using wrapper for archiving
84
- const values = await sentinelWrapper.report([query])
85
- const timestamps = values.filter(isTimestamp)
86
- expect(timestamps.length).toBe(1)
87
- const thumbnails = values.filter(isImageThumbnail)
88
- expect(thumbnails.length).toBe(1)
89
- const thumbnail = thumbnails[0]
90
- expect(thumbnail.sourceUrl).toBe(url)
91
- await delay(1000)
92
- const payloads = await archivist?.all()
93
- expect(payloads?.length).toBeGreaterThan(0)
94
- })
95
- })
96
- })
@@ -1,32 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import {
4
- describe, expect,
5
- it,
6
- } from 'vitest'
7
-
8
- import { checkIpfsUrl } from '../lib/index.ts'
9
-
10
- /**
11
- * @group thumbnail
12
- */
13
-
14
- describe('Witness', () => {
15
- describe('checkIpfsUrl', () => {
16
- const cases: [uri: string, expected: string][] = [
17
- [
18
- 'ipfs://ipfs/QmewywDQGqz9WuWfT11ueSR3Mu86MejfD64v3KtFRzGP2G/image.jpeg',
19
- 'https://5d7b6582.beta.decentralnetworkservices.com/ipfs/QmewywDQGqz9WuWfT11ueSR3Mu86MejfD64v3KtFRzGP2G/image.jpeg',
20
- ],
21
- [
22
- 'ipfs://QmWX3Kx2NX3AK8WxTQwktVYLMFHX3pHm77ThynhgmU8dP8',
23
- 'https://5d7b6582.beta.decentralnetworkservices.com/ipfs/QmWX3Kx2NX3AK8WxTQwktVYLMFHX3pHm77ThynhgmU8dP8',
24
- ],
25
- ]
26
-
27
- it.each(cases)('%s', (input, expected) => {
28
- const actual = checkIpfsUrl(input, '5d7b6582.beta.decentralnetworkservices.com')
29
- expect(actual).toBe(expected)
30
- })
31
- })
32
- })
@@ -1,124 +0,0 @@
1
- import '@xylabs/vitest-extended'
2
-
3
- import type { ImageThumbnail } from '@xyo-network/image-thumbnail-payload-plugin'
4
- import { ImageThumbnailSchema } from '@xyo-network/image-thumbnail-payload-plugin'
5
- import type { UrlPayload } from '@xyo-network/url-payload-plugin'
6
- import { UrlSchema } from '@xyo-network/url-payload-plugin'
7
- import { fileTypeFromBuffer } from 'file-type'
8
- import hasbin from 'hasbin'
9
- import {
10
- beforeAll,
11
- describe, expect, it,
12
- } from 'vitest'
13
-
14
- import { ImageThumbnailWitness } from '../Witness.ts'
15
-
16
- const describeIfHasBin = (bin: string) => (hasbin.sync(bin) ? describe : describe.skip)
17
-
18
- type MimeWithUrl = [mime: string, url: string]
19
-
20
- const testVideoFormat = async (witness: ImageThumbnailWitness, url: string) => {
21
- const httpsPayload: UrlPayload = { schema: UrlSchema, url }
22
- const result = (await witness.observe([httpsPayload])) as ImageThumbnail[]
23
- expect(result.length).toBe(1)
24
- expect(result[0].schema).toBe(ImageThumbnailSchema)
25
- expect(result[0].url?.length).toBeGreaterThan(0)
26
- const imageData = result[0].url?.split(',')[1] ?? ''
27
- const buffer = Buffer.from(Uint8Array.from(atob(imageData), c => c.codePointAt(0) ?? 0))
28
- const fileType = await fileTypeFromBuffer(buffer)
29
- expect(fileType?.mime).toBe('image/png')
30
- }
31
-
32
- /**
33
- * @group thumbnail
34
- */
35
-
36
- describe.skip('ImageThumbnailWitness', () => {
37
- describeIfHasBin('magick')('ImageThumbnailWitness', () => {
38
- describe('with video type', () => {
39
- let witness: ImageThumbnailWitness
40
- beforeAll(async () => {
41
- witness = await ImageThumbnailWitness.create({ account: 'random' })
42
- })
43
- describe('3GPP', () => {
44
- const cases: MimeWithUrl[] = [['video/3gpp', 'https://filesamples.com/samples/video/3gp/sample_640x360.3gp']]
45
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
46
- })
47
- describe.skip('3GPP2', () => {
48
- const cases: MimeWithUrl[] = [['video/3gpp2', '']]
49
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
50
- })
51
- describe.skip('Advanced Video Coding / H.264', () => {
52
- const cases: MimeWithUrl[] = [['video/h264', '']]
53
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
54
- })
55
- describe.skip('High Efficiency Video Coding / H.265', () => {
56
- const cases: MimeWithUrl[] = [['video/h265', '']]
57
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
58
- })
59
- describe.skip('MPEG-2 Transport Stream', () => {
60
- const cases: MimeWithUrl[] = [['video/mp2t', 'https://filesamples.com/samples/video/m2ts/sample_640x360.m2ts']]
61
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
62
- })
63
- describe('MPEG-4 Part 14', () => {
64
- const cases: MimeWithUrl[] = [
65
- ['video/mp4', 'https://cdn-longterm.mee6.xyz/assets/avatars-presale.mp4'],
66
- ['video/mp4;codecs=avc1', 'https://media.niftygateway.com/video/upload/v1649189105/Abigail/FEWO/Paint/Paint/006266_paint_hf9cft.mp4'],
67
- ]
68
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
69
- })
70
- describe('MPEG Moving Picture Experts Group Phase 1', () => {
71
- const cases: MimeWithUrl[] = [['video/mpeg', 'https://filesamples.com/samples/video/mpeg/sample_640x360.mpeg']]
72
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
73
- })
74
- describe.skip('OGG', () => {
75
- const cases: MimeWithUrl[] = [['video/ogg', 'https://filesamples.com/samples/video/ogv/sample_640x360.ogv']]
76
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
77
- })
78
- describe('QuickTime File Format (QTFF)', () => {
79
- const cases: MimeWithUrl[] = [['video/quicktime', 'https://filesamples.com/samples/video/mov/sample_640x360.mov']]
80
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
81
- })
82
- describe.skip('RealMedia', () => {
83
- const cases: MimeWithUrl[] = [['video/vnd.rn-realvideo', '.rm']]
84
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
85
- })
86
- describe('WebM', () => {
87
- const cases: MimeWithUrl[] = [['video/webm', 'https://filesamples.com/samples/video/webm/sample_640x360.webm']]
88
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
89
- })
90
- describe('Flash Video', () => {
91
- const cases: MimeWithUrl[] = [
92
- ['video/x-flv', 'https://filesamples.com/samples/video/flv/sample_640x360.flv'],
93
- // NOTE: Because of restrictions in the FLV file format, Adobe Systems created new file formats
94
- // in 2007, based on the ISO base media file format (MPEG-4 Part 12). In this way, the F4V format
95
- // shares a common base with the MP4 format, which is why F4V is sometimes informally called "Flash MP4".
96
- // https://en.m.wikipedia.org/wiki/Flash_Video#History
97
- // ['video/mp4', 'https://filesamples.com/samples/video/f4v/sample_640x360.f4v'],
98
- ]
99
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
100
- })
101
- describe('MPEG-4 Video', () => {
102
- const cases: MimeWithUrl[] = [['video/x-m4v', 'https://filesamples.com/samples/video/m4v/sample_640x360.m4v']]
103
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
104
- })
105
- describe.skip('Matroska Multimedia Container', () => {
106
- const cases: MimeWithUrl[] = [['video/x-matroska', 'https://filesamples.com/samples/video/mkv/sample_640x360.mkv']]
107
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
108
- })
109
- describe('Windows Media Video', () => {
110
- const cases: MimeWithUrl[] = [['video/x-ms-wmv', 'https://filesamples.com/samples/video/wmv/sample_640x360.wmv']]
111
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
112
- })
113
- describe('Audio Video Interleave', () => {
114
- const cases: MimeWithUrl[] = [
115
- // ['video/vnd.avi', 'https://filesamples.com/samples/video/avi/sample_640x360.avi'],
116
- ['video/avi', 'https://filesamples.com/samples/video/avi/sample_640x360.avi'],
117
- // ['video/msvideo', 'https://filesamples.com/samples/video/avi/sample_640x360.avi'],
118
- // ['video/x-msvideo', 'https://filesamples.com/samples/video/avi/sample_640x360.avi'],
119
- ]
120
- it.each(cases)('video [mime (%s)]', async (_mime, url) => await testVideoFormat(witness, url))
121
- })
122
- })
123
- })
124
- })