Allow SVGs when using Assets (#7643)
* Allow SVG files when using Assets * Fixed TypeScript error * fix: some small nits and add a test * chore: changeset --------- Co-authored-by: Princesseuh <princssdev@gmail.com>
This commit is contained in:
parent
bdde6b9f6c
commit
4b82e55cf1
8 changed files with 46 additions and 17 deletions
5
.changeset/tasty-bags-accept.md
Normal file
5
.changeset/tasty-bags-accept.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Add support for using `.svg` files with `astro:assets`'s base services. The SVGs will NOT be processed and will be return as-is, however, proper attributes, alt enforcement etc will all work correctly.
|
|
@ -18,8 +18,16 @@ export const VALID_INPUT_FORMATS = [
|
||||||
'svg',
|
'svg',
|
||||||
] as const;
|
] as const;
|
||||||
/**
|
/**
|
||||||
* Valid formats for optimizations in our base services.
|
* Valid formats that our base services support.
|
||||||
* Certain formats can be imported (namely SVGs) but not processed, so they are excluded from this list.
|
* Certain formats can be imported (namely SVGs) but will not be processed.
|
||||||
*/
|
*/
|
||||||
export const VALID_OPTIMIZABLE_FORMATS = ['jpeg', 'jpg', 'png', 'tiff', 'webp', 'gif'] as const;
|
export const VALID_SUPPORTED_FORMATS = [
|
||||||
export const VALID_OUTPUT_FORMATS = ['avif', 'png', 'webp', 'jpeg', 'jpg'] as const;
|
'jpeg',
|
||||||
|
'jpg',
|
||||||
|
'png',
|
||||||
|
'tiff',
|
||||||
|
'webp',
|
||||||
|
'gif',
|
||||||
|
'svg',
|
||||||
|
] as const;
|
||||||
|
export const VALID_OUTPUT_FORMATS = ['avif', 'png', 'webp', 'jpeg', 'jpg', 'svg'] as const;
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { AstroError, AstroErrorData } from '../../core/errors/index.js';
|
import { AstroError, AstroErrorData } from '../../core/errors/index.js';
|
||||||
import { joinPaths } from '../../core/path.js';
|
import { joinPaths } from '../../core/path.js';
|
||||||
import { VALID_OPTIMIZABLE_FORMATS } from '../consts.js';
|
import { VALID_SUPPORTED_FORMATS } from '../consts.js';
|
||||||
import { isESMImportedImage } from '../internal.js';
|
import { isESMImportedImage } from '../internal.js';
|
||||||
import type { ImageOutputFormat, ImageTransform } from '../types.js';
|
import type { ImageOutputFormat, ImageTransform } from '../types.js';
|
||||||
|
|
||||||
|
@ -143,16 +143,21 @@ export const baseService: Omit<LocalImageService, 'transform'> = {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!VALID_OPTIMIZABLE_FORMATS.includes(options.src.format as any)) {
|
if (!VALID_SUPPORTED_FORMATS.includes(options.src.format as any)) {
|
||||||
throw new AstroError({
|
throw new AstroError({
|
||||||
...AstroErrorData.UnsupportedImageFormat,
|
...AstroErrorData.UnsupportedImageFormat,
|
||||||
message: AstroErrorData.UnsupportedImageFormat.message(
|
message: AstroErrorData.UnsupportedImageFormat.message(
|
||||||
options.src.format,
|
options.src.format,
|
||||||
options.src.src,
|
options.src.src,
|
||||||
VALID_OPTIMIZABLE_FORMATS
|
VALID_SUPPORTED_FORMATS
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We currently do not support processing SVGs, so whenever the input format is a SVG, force the output to also be one
|
||||||
|
if (options.src.format === 'svg') {
|
||||||
|
options.format = 'svg';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the user didn't specify a format, we'll default to `webp`. It offers the best ratio of compatibility / quality
|
// If the user didn't specify a format, we'll default to `webp`. It offers the best ratio of compatibility / quality
|
||||||
|
|
|
@ -37,6 +37,10 @@ const sharpService: LocalImageService = {
|
||||||
|
|
||||||
const transform: BaseServiceTransform = transformOptions as BaseServiceTransform;
|
const transform: BaseServiceTransform = transformOptions as BaseServiceTransform;
|
||||||
|
|
||||||
|
// Return SVGs as-is
|
||||||
|
// TODO: Sharp has some support for SVGs, we could probably support this once Sharp is the default and only service.
|
||||||
|
if (transform.format === 'svg') return { data: inputBuffer, format: 'svg' };
|
||||||
|
|
||||||
let result = sharp(inputBuffer, { failOnError: false, pages: -1 });
|
let result = sharp(inputBuffer, { failOnError: false, pages: -1 });
|
||||||
|
|
||||||
// Never resize using both width and height at the same time, prioritizing width.
|
// Never resize using both width and height at the same time, prioritizing width.
|
||||||
|
|
|
@ -12,7 +12,7 @@ import type { Operation } from './vendor/squoosh/image.js';
|
||||||
|
|
||||||
const baseQuality = { low: 25, mid: 50, high: 80, max: 100 };
|
const baseQuality = { low: 25, mid: 50, high: 80, max: 100 };
|
||||||
const qualityTable: Record<
|
const qualityTable: Record<
|
||||||
Exclude<ImageOutputFormat, 'png'>,
|
Exclude<ImageOutputFormat, 'png' | 'svg'>,
|
||||||
Record<ImageQualityPreset, number>
|
Record<ImageQualityPreset, number>
|
||||||
> = {
|
> = {
|
||||||
avif: {
|
avif: {
|
||||||
|
@ -38,6 +38,9 @@ const service: LocalImageService = {
|
||||||
|
|
||||||
let format = transform.format;
|
let format = transform.format;
|
||||||
|
|
||||||
|
// Return SVGs as-is
|
||||||
|
if (format === 'svg') return { data: inputBuffer, format: 'svg' };
|
||||||
|
|
||||||
const operations: Operation[] = [];
|
const operations: Operation[] = [];
|
||||||
|
|
||||||
// Never resize using both width and height at the same time, prioritizing width.
|
// Never resize using both width and height at the same time, prioritizing width.
|
||||||
|
|
|
@ -503,8 +503,8 @@ See https://docs.astro.build/en/guides/server-side-rendering/ for more informati
|
||||||
message: (format: string, imagePath: string, supportedFormats: readonly string[]) =>
|
message: (format: string, imagePath: string, supportedFormats: readonly string[]) =>
|
||||||
`Received unsupported format \`${format}\` from \`${imagePath}\`. Currently only ${supportedFormats.join(
|
`Received unsupported format \`${format}\` from \`${imagePath}\`. Currently only ${supportedFormats.join(
|
||||||
', '
|
', '
|
||||||
)} are supported for optimization.`,
|
)} are supported by our image services.`,
|
||||||
hint: "If you do not need optimization, using an `img` tag directly instead of the `Image` component might be what you're looking for.",
|
hint: "Using an `img` tag directly instead of the `Image` component might be what you're looking for.",
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
* @docs
|
* @docs
|
||||||
|
|
|
@ -109,14 +109,18 @@ describe('astro:image', () => {
|
||||||
expect(res.headers.get('content-type')).to.equal('image/webp');
|
expect(res.headers.get('content-type')).to.equal('image/webp');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('errors on unsupported formats', async () => {
|
it('properly skip processing SVGs, but does not error', async () => {
|
||||||
logs.length = 0;
|
let res = await fixture.fetch('/svgSupport');
|
||||||
let res = await fixture.fetch('/unsupported-format');
|
let html = await res.text();
|
||||||
await res.text();
|
|
||||||
|
|
||||||
expect(logs).to.have.a.lengthOf(1);
|
$ = cheerio.load(html);
|
||||||
expect(logs[0].message).to.contain('Received unsupported format');
|
let $img = $('img');
|
||||||
});
|
expect($img).to.have.a.lengthOf(1);
|
||||||
|
|
||||||
|
let src = $img.attr('src');
|
||||||
|
res = await fixture.fetch(src);
|
||||||
|
expect(res.status).to.equal(200);
|
||||||
|
})
|
||||||
|
|
||||||
it("errors when an ESM imported image's src is passed to Image/getImage instead of the full import", async () => {
|
it("errors when an ESM imported image's src is passed to Image/getImage instead of the full import", async () => {
|
||||||
logs.length = 0;
|
logs.length = 0;
|
||||||
|
|
Loading…
Reference in a new issue