diff --git a/src/handler/image.ts b/src/handler/image.ts index cd0fb584..84e73e8e 100644 --- a/src/handler/image.ts +++ b/src/handler/image.ts @@ -59,7 +59,7 @@ function parsePNG(buf: ArrayBuffer) { import { createLRU, parseViewBox } from '../utils.js' -type ResolvedImageData = [string, number?, number?] +type ResolvedImageData = [string, number?, number?] | readonly [] const cache = createLRU(100) const inflightRequests = new Map>() @@ -249,7 +249,8 @@ export async function resolveImageData( return result }) .catch((err) => { - throw new Error(`Can't load image ${url}: ` + err.message) + console.error(`Can't load image ${url}: ` + err.message) + return [] as const }) inflightRequests.set(url, promise) diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-correctly-position-the-background-pattern-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-correctly-position-the-background-pattern-1-snap.png new file mode 100644 index 00000000..858ed5db Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-correctly-position-the-background-pattern-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-1-snap.png new file mode 100644 index 00000000..1fae2c81 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-2-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-2-snap.png new file mode 100644 index 00000000..1fae2c81 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-2-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-3-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-3-snap.png new file mode 100644 index 00000000..7f07e25a Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-3-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-4-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-4-snap.png new file mode 100644 index 00000000..7b0e604b Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-4-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-5-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-5-snap.png new file mode 100644 index 00000000..296bc4c7 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-data-uris-with-size-for-supported-image-formats-5-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-image-data-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-image-data-1-snap.png new file mode 100644 index 00000000..dc8af41d Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-resolve-image-data-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-double-quotes-inside-url-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-double-quotes-inside-url-1-snap.png new file mode 100644 index 00000000..dc8af41d Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-double-quotes-inside-url-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-single-quotes-inside-url-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-single-quotes-inside-url-1-snap.png new file mode 100644 index 00000000..dc8af41d Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-single-quotes-inside-url-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-stretched-background-size-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-stretched-background-size-1-snap.png new file mode 100644 index 00000000..7badec0a Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-background-image-url-should-support-stretched-background-size-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-and-padding-areas-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-and-padding-areas-1-snap.png new file mode 100644 index 00000000..bb34fed7 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-and-padding-areas-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-area-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-area-1-snap.png new file mode 100644 index 00000000..453d8521 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-clip-content-in-the-border-area-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-deduplicate-image-data-requests-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-deduplicate-image-data-requests-1-snap.png new file mode 100644 index 00000000..2d5bea1b Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-deduplicate-image-data-requests-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-not-throw-when-image-is-not-valid-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-not-throw-when-image-is-not-valid-1-snap.png new file mode 100644 index 00000000..e0f16792 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-not-throw-when-image-is-not-valid-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-image-data-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-image-data-1-snap.png new file mode 100644 index 00000000..dc8af41d Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-image-data-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-non-square-image-size-correctly-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-non-square-image-size-correctly-1-snap.png new file mode 100644 index 00000000..e2321723 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-non-square-image-size-correctly-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-the-image-size-and-scale-automatically-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-the-image-size-and-scale-automatically-1-snap.png new file mode 100644 index 00000000..bf5ad3b0 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-resolve-the-image-size-and-scale-automatically-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-array-buffer-as-src-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-array-buffer-as-src-1-snap.png new file mode 100644 index 00000000..e70c43ed Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-array-buffer-as-src-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-opacity-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-opacity-1-snap.png new file mode 100644 index 00000000..2d6df5d3 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-opacity-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-styles-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-styles-1-snap.png new file mode 100644 index 00000000..e8c0df08 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-styles-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-svg-images-and-percentage-with-correct-aspect-ratio-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-svg-images-and-percentage-with-correct-aspect-ratio-1-snap.png new file mode 100644 index 00000000..9eeca7be Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-svg-images-and-percentage-with-correct-aspect-ratio-1-snap.png differ diff --git a/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-transparent-image-with-background-1-snap.png b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-transparent-image-with-background-1-snap.png new file mode 100644 index 00000000..0200e7d0 Binary files /dev/null and b/test/__image_snapshots__/image-test-tsx-test-image-test-tsx-image-should-support-transparent-image-with-background-1-snap.png differ diff --git a/test/image.test.tsx b/test/image.test.tsx index 80238f78..9737147a 100644 --- a/test/image.test.tsx +++ b/test/image.test.tsx @@ -18,57 +18,25 @@ function dataUriToArrayBuffer(dataUri: string): ArrayBuffer { const PNG_SAMPLE_ARRAYBUFFER = dataUriToArrayBuffer(PNG_SAMPLE) -describe('Image', () => { - let fonts - initFonts((f) => (fonts = f)) - - let requests = [] - - beforeEach(() => { - // Polyfill fetch - requests = [] - ;(globalThis as any).fetch = async (url) => { - requests.push(url) - if (url.startsWith('data:')) { - return { - headers: { - get: () => 'image/png', - }, - text: async () => { - const binary_string = atob( - url.replace('data:image/png;base64,', '') - ) - const len = binary_string.length - const bytes = new Uint8Array(len) - for (let i = 0; i < len; i++) { - bytes[i] = binary_string.charCodeAt(i) - } - return bytes.buffer - }, - } - } - - if (url.endsWith('.svg')) { - return { - headers: { - get: () => 'image/svg+xml', - }, - text: async () => - '', - } - } - +let fonts +initFonts((f) => (fonts = f)) + +let requests = [] + +beforeEach(() => { + // Polyfill fetch + requests = [] + ;(globalThis as any).fetch = async (url) => { + requests.push(url) + if (url.includes('wrong-url')) { + throw Error('wrong url') + } else if (url.startsWith('data:')) { return { headers: { - get: (key) => { - if (key === 'content-type') return 'image/png' - }, + get: () => 'image/png', }, - arrayBuffer: async () => { - // 1x1 #00F blue image. - const binary_string = atob( - `iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPj/HwADBwIAMCbHYQAAAABJRU5ErkJggg==` - ) + text: async () => { + const binary_string = atob(url.replace('data:image/png;base64,', '')) const len = binary_string.length const bytes = new Uint8Array(len) for (let i = 0; i < len; i++) { @@ -78,280 +46,386 @@ describe('Image', () => { }, } } - }) - afterEach(() => { - delete globalThis.fetch - }) + if (url.endsWith('.svg')) { + return { + headers: { + get: () => 'image/svg+xml', + }, + text: async () => + '', + } + } - describe('img', () => { - it('should resolve image data', async () => { - const svg = await satori( -
- -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() + return { + headers: { + get: (key) => { + if (key === 'content-type') return 'image/png' + }, + }, + arrayBuffer: async () => { + // 1x1 #00F blue image. + const binary_string = atob( + `iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNkYPj/HwADBwIAMCbHYQAAAABJRU5ErkJggg==` + ) + const len = binary_string.length + const bytes = new Uint8Array(len) + for (let i = 0; i < len; i++) { + bytes[i] = binary_string.charCodeAt(i) + } + return bytes.buffer + }, + } + } +}) - expect(requests).toEqual(['https://via.placeholder.com/150']) - }) +afterEach(() => { + delete globalThis.fetch +}) - it('should deduplicate image data requests', async () => { - const svg = await satori( -
- - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() +describe('Image', () => { + it('should resolve image data', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + + expect(requests).toEqual(['https://via.placeholder.com/150']) + }) - expect(requests).toEqual(['https://via.placeholder.com/200']) - }) + it('should deduplicate image data requests', async () => { + const svg = await satori( +
+ + +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + + expect(requests).toEqual(['https://via.placeholder.com/200']) + }) - it('should resolve the image size and scale automatically', async () => { - const svg = await satori( -
- -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + it('should resolve the image size and scale automatically', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should resolve non-square image size correctly', async () => { - const svg = await satori( -
- -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + it('should resolve non-square image size correctly', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should support styles', async () => { - const svg = await satori( -
{ + const svg = await satori( +
+ - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + /> +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should support opacity', async () => { - const svg = await satori( -
{ + const svg = await satori( +
+ - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + /> +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should support SVG images and percentage with correct aspect ratio', async () => { - const svg = await satori( -
- -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + it('should support SVG images and percentage with correct aspect ratio', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should clip content in the border area', async () => { - const svg = await satori( -
{ + const svg = await satori( +
+ - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + /> +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should clip content in the border and padding areas', async () => { - const svg = await satori( -
{ + const svg = await satori( +
+ - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + /> +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should support transparent image with background', async () => { - const svg = await satori( -
{ + const svg = await satori( +
+ - -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + /> +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - it('should support ArrayBuffer as src', async () => { - const svg = await satori( -
- -
, - { width: 100, height: 100, fonts } - ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + it('should support ArrayBuffer as src', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() }) - describe('background-image: url()', () => { - it('should resolve image data', async () => { - const svg = await satori( -
, - { width: 100, height: 100, fonts } - ) + it('should not throw when image is not valid', async () => { + const svg = await satori( +
+ +
, + { width: 100, height: 100, fonts } + ) + + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) +}) + +describe('background-image: url()', () => { + it('should resolve image data', async () => { + const svg = await satori( +
, + { width: 100, height: 100, fonts } + ) + + expect(toImage(svg, 100)).toMatchImageSnapshot() + + expect(requests).toEqual(['https://via.placeholder.com/300']) + }) - expect(toImage(svg, 100)).toMatchImageSnapshot() + it('should support single quotes inside url()', async () => { + const svg = await satori( +
, + { width: 100, height: 100, fonts } + ) + expect(toImage(svg, 100)).toMatchImageSnapshot() + + expect(requests).toEqual(['https://via.placeholder.com/301']) + }) - expect(requests).toEqual(['https://via.placeholder.com/300']) - }) + it('should support double quotes inside url()', async () => { + const svg = await satori( +
, + { width: 100, height: 100, fonts } + ) + + expect(toImage(svg, 100)).toMatchImageSnapshot() + + expect(requests).toEqual(['https://via.placeholder.com/302']) + }) - it('should support single quotes inside url()', async () => { - const svg = await satori( -
, - { width: 100, height: 100, fonts } + it('should support SVG data uris with various quotes inside url()', async () => { + const backgroundImagesWithDoubleQuotes = [ + `url(data:image/svg+xml,)`, + `url(data:image/svg+xml;utf8,)`, + `url(data:image/svg+xml,%3Csvg width="116" height="100" fill="white" xmlns="http://www.w3.org/2000/svg"%3E%3Cpath d="M57.5 0L115 100H0L57.5 0z" /%3E%3C/svg%3E)`, + `url(data:image/svg+xml;utf8,%3Csvg width="116" height="100" fill="white" xmlns="http://www.w3.org/2000/svg"%3E%3Cpath d="M57.5 0L115 100H0L57.5 0z" /%3E%3C/svg%3E)`, + `url(data:image/svg+xml,%3Csvg%20width%3D%22116%22%20height%3D%22100%22%20fill%3D%22white%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M57.5%200L115%20100H0L57.5%200z%22%20%2F%3E%3C%2Fsvg%3E)`, + `url(data:image/svg+xml;utf8,%3Csvg%20width%3D%22116%22%20height%3D%22100%22%20fill%3D%22white%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M57.5%200L115%20100H0L57.5%200z%22%20%2F%3E%3C%2Fsvg%3E)`, + `url()`, + ] + + const backgroundImagesWithSingleQuotes = [ + `url(data:image/svg+xml,)`, + `url(data:image/svg+xml;utf8,)`, + `url(data:image/svg+xml,%3Csvg width='116' height='100' fill='white' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M57.5 0L115 100H0L57.5 0z' /%3E%3C/svg%3E)`, + `url(data:image/svg+xml;utf8,%3Csvg width='116' height='100' fill='white' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M57.5 0L115 100H0L57.5 0z' /%3E%3C/svg%3E)`, + `url(data:image/svg+xml,%3Csvg%20width%3D%27116%27%20height%3D%27100%27%20fill%3D%27white%27%20xmlns%3D%27http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%27%3E%3Cpath%20d%3D%27M57.5%200L115%20100H0L57.5%200z%27%20%2F%3E%3C%2Fsvg%3E)`, + `url(data:image/svg+xml;utf8,%3Csvg%20width%3D%27116%27%20height%3D%27100%27%20fill%3D%27white%27%20xmlns%3D%27http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%27%3E%3Cpath%20d%3D%27M57.5%200L115%20100H0L57.5%200z%27%20%2F%3E%3C%2Fsvg%3E)`, + `url()`, + ] + + const backgroundImagesWithDoubleQuotesWrappedInSingleQuotes = + backgroundImagesWithDoubleQuotes.map((b) => + b.replace(/^url\(/, `url('`).replace(/\)$/, `')`) + ) + const backgroundImagesWithSingleQuotesWrappedInDoubleQuotes = + backgroundImagesWithDoubleQuotes.map((b) => + b.replace(/^url\(/, `url("`).replace(/\)$/, `")`) ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - expect(requests).toEqual(['https://via.placeholder.com/301']) - }) + const backgroundImages = [ + ...backgroundImagesWithDoubleQuotes, + ...backgroundImagesWithSingleQuotes, + ...backgroundImagesWithDoubleQuotesWrappedInSingleQuotes, + ...backgroundImagesWithSingleQuotesWrappedInDoubleQuotes, + ] - it('should support double quotes inside url()', async () => { + let lastImageBuffer = null + for (const backgroundImage of backgroundImages) { const svg = await satori(
{ width: '50%', height: '50%', display: 'flex', - backgroundImage: 'url("https://via.placeholder.com/302")', + backgroundImage, + backgroundSize: '50px 50px', }} >
, { width: 100, height: 100, fonts } ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - - expect(requests).toEqual(['https://via.placeholder.com/302']) - }) - - it('should support SVG data uris with various quotes inside url()', async () => { - const backgroundImagesWithDoubleQuotes = [ - `url(data:image/svg+xml,)`, - `url(data:image/svg+xml;utf8,)`, - `url(data:image/svg+xml,%3Csvg width="116" height="100" fill="white" xmlns="http://www.w3.org/2000/svg"%3E%3Cpath d="M57.5 0L115 100H0L57.5 0z" /%3E%3C/svg%3E)`, - `url(data:image/svg+xml;utf8,%3Csvg width="116" height="100" fill="white" xmlns="http://www.w3.org/2000/svg"%3E%3Cpath d="M57.5 0L115 100H0L57.5 0z" /%3E%3C/svg%3E)`, - `url(data:image/svg+xml,%3Csvg%20width%3D%22116%22%20height%3D%22100%22%20fill%3D%22white%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M57.5%200L115%20100H0L57.5%200z%22%20%2F%3E%3C%2Fsvg%3E)`, - `url(data:image/svg+xml;utf8,%3Csvg%20width%3D%22116%22%20height%3D%22100%22%20fill%3D%22white%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M57.5%200L115%20100H0L57.5%200z%22%20%2F%3E%3C%2Fsvg%3E)`, - `url()`, - ] - - const backgroundImagesWithSingleQuotes = [ - `url(data:image/svg+xml,)`, - `url(data:image/svg+xml;utf8,)`, - `url(data:image/svg+xml,%3Csvg width='116' height='100' fill='white' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M57.5 0L115 100H0L57.5 0z' /%3E%3C/svg%3E)`, - `url(data:image/svg+xml;utf8,%3Csvg width='116' height='100' fill='white' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M57.5 0L115 100H0L57.5 0z' /%3E%3C/svg%3E)`, - `url(data:image/svg+xml,%3Csvg%20width%3D%27116%27%20height%3D%27100%27%20fill%3D%27white%27%20xmlns%3D%27http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%27%3E%3Cpath%20d%3D%27M57.5%200L115%20100H0L57.5%200z%27%20%2F%3E%3C%2Fsvg%3E)`, - `url(data:image/svg+xml;utf8,%3Csvg%20width%3D%27116%27%20height%3D%27100%27%20fill%3D%27white%27%20xmlns%3D%27http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%27%3E%3Cpath%20d%3D%27M57.5%200L115%20100H0L57.5%200z%27%20%2F%3E%3C%2Fsvg%3E)`, - `url()`, - ] - - const backgroundImagesWithDoubleQuotesWrappedInSingleQuotes = - backgroundImagesWithDoubleQuotes.map((b) => - b.replace(/^url\(/, `url('`).replace(/\)$/, `')`) - ) - const backgroundImagesWithSingleQuotesWrappedInDoubleQuotes = - backgroundImagesWithDoubleQuotes.map((b) => - b.replace(/^url\(/, `url("`).replace(/\)$/, `")`) - ) - - const backgroundImages = [ - ...backgroundImagesWithDoubleQuotes, - ...backgroundImagesWithSingleQuotes, - ...backgroundImagesWithDoubleQuotesWrappedInSingleQuotes, - ...backgroundImagesWithSingleQuotesWrappedInDoubleQuotes, - ] - - let lastImageBuffer = null - for (const backgroundImage of backgroundImages) { - const svg = await satori( -
, - { width: 100, height: 100, fonts } - ) - - const newImageBuffer = toImage(svg, 100) - if (lastImageBuffer) { - expect(newImageBuffer.equals(lastImageBuffer)).toBe(true) - } - lastImageBuffer = newImageBuffer + const newImageBuffer = toImage(svg, 100) + if (lastImageBuffer) { + expect(newImageBuffer.equals(lastImageBuffer)).toBe(true) } - }) - - it('should resolve data uris with size for supported image formats', async () => { - // tests with all the supported image data uri formats. - const renderSvg = (imageUri) => - satori( -
, - { width: 100, height: 100, fonts } - ) - - const basedOnPlainSvg = await renderSvg( - '' - ) - const basedOnPng = await renderSvg( - '' - ) - const basedOnJpeg = await renderSvg( - '' - ) - const basedOnGif = await renderSvg( - '' - ) - - expect(toImage(basedOnPlainSvg, 100)).toMatchImageSnapshot() - expect(toImage(basedOnEncodedSvg, 100)).toMatchImageSnapshot() - expect(toImage(basedOnPng, 100)).toMatchImageSnapshot() - expect(toImage(basedOnJpeg, 100)).toMatchImageSnapshot() - expect(toImage(basedOnGif, 100)).toMatchImageSnapshot() - }) + lastImageBuffer = newImageBuffer + } + }) - it('should support stretched backgroundSize', async () => { - const svg = await satori( + it('should resolve data uris with size for supported image formats', async () => { + // tests with all the supported image data uri formats. + const renderSvg = (imageUri) => + satori(
, - { width: 100, height: 50, fonts } + { width: 100, height: 100, fonts } ) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + const basedOnPlainSvg = await renderSvg( + '' + ) + const basedOnPng = await renderSvg( + '' + ) + const basedOnJpeg = await renderSvg( + '' + ) + const basedOnGif = await renderSvg( + '' + ) + + expect(toImage(basedOnPlainSvg, 100)).toMatchImageSnapshot() + expect(toImage(basedOnEncodedSvg, 100)).toMatchImageSnapshot() + expect(toImage(basedOnPng, 100)).toMatchImageSnapshot() + expect(toImage(basedOnJpeg, 100)).toMatchImageSnapshot() + expect(toImage(basedOnGif, 100)).toMatchImageSnapshot() + }) - it('should correctly position the background pattern', async () => { - const svg = await satori( -
, - { width: 100, height: 100, fonts } - ) + it('should support stretched backgroundSize', async () => { + const svg = await satori( +
, + { width: 100, height: 50, fonts } + ) + + expect(toImage(svg, 100)).toMatchImageSnapshot() + }) - expect(toImage(svg, 100)).toMatchImageSnapshot() - }) + it('should correctly position the background pattern', async () => { + const svg = await satori( +
, + { width: 100, height: 100, fonts } + ) + + expect(toImage(svg, 100)).toMatchImageSnapshot() }) })