What are you trying to achieve?
Render an SVG from buffer that refers to an external or local image file.
For example:
<svg width="200" height="200"
xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<image xlink:href="https://mdn.mozillademos.org/files/6457/mdn_logo_only_color.png" height="200" width="200"/>
</svg>
(image taken from this Mozilla example: https://developer.mozilla.org/en-US/docs/Web/SVG/Element/image)
Have you searched for similar questions?
Yes.
Are you able to provide a standalone code sample that demonstrates this question?
This code:
const sharp = require('sharp');
var svgData = `<svg width="200" height="200"
xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<image xlink:href="https://mdn.mozillademos.org/files/6457/mdn_logo_only_color.png" height="200" width="200"/>
</svg>`;
sharp(new Buffer.from(svgData))
.toFile("test.png")
.catch(err => {
console.error(err);
});
Results in this error:
[Error: vips_realpath: unable to form filename
unix error: No such file or directory
svgload_buffer: SVG rendering failed
vips2png: unable to write "test.png"
]
I am probably missing something simple, but it seems that buffer input confuses the path resolution somehow. For local files with relative paths, would these be relative to the execution directory?
Are you able to provide a sample image that helps explain the question?
See: https://developer.mozilla.org/en-US/docs/Web/SVG/Element/image
Using a file called test-buffer.svg with the same SVG and the following code also returns an error, so something seems amiss:
const sharp = require('sharp');
// test-buffer.svg contains same SVG as OP
sharp("test-buffer.svg")
.toFile("test.png")
.catch(err => {
console.error(err);
});
Error:
[Error: svgload: SVG rendering failed
vips2png: unable to write "test.png"
]
Hello, did you see https://github.com/lovell/sharp/issues/1378?
No, I had not. Is it the same issue? For example, if I download that Mozilla image in the img directory just below my js, and modify the image tag to refer to it with a relative path, it works:
<svg width="200" height="200"
xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<image xlink:href="img/mdn_logo_only_color.png" height="200" width="200"/>
</svg>
But if I put this SVG in a variable and load it with Buffer.from like my first example, it does not. My issue seems to be path resolution rather than nesting. Or maybe I'm wrong and all this is related to pixbuf.
librsvg needs a base URL for referenced files. If a base URL is not given, then any referenced files will not be allowed.
https://gitlab.gnome.org/GNOME/librsvg/blob/b4aa8d277d30f4042797dc8b64c073b10f8bac4b/rsvg_internals/src/allowed_url.rs#L60
When loading from a file, librsvg will set the base file to the file itself. But when a SVG image is loaded from a buffer (Buffer.from) a base file isn't set.
https://github.com/libvips/libvips/blob/3b0d44be51fac4d04355f409e12ffed34d886efd/libvips/foreign/svgload.c#L582
This will require a new optional argument within vips_svgload_buffer (for e.g. base_uri) that will set the base URL (this can be done with rsvg_handle_set_base_uri).
To workaround this for now you'll need to base64 encode the relevant image directly within the SVG file. data: URLs doesn't require a base URL.
I see, thanks for the explanation. I'm creating SVG on the fly, so my workaround would be to save a temporary file and load that with sharp. Unfortunate.
As an added note, even for loading as an SVG file, when the xlink:href is a full URL, an error is generated. When the image is locally available and referenced with a relative path, it works. Surely that is a bug?
librsvg will not load http resources. As librsvg is shared across the GNOME environment, it's pretty important to have a strict deny on remote resources. This to keep malicious SVG data from "phoning home".
Interesting, even though Chrome renders it. I will work with local files then.
The current behaviour of librsvg is expected and by design. Should network access be required when rendering a given SVG then librsvg+libvips is probably not the right solution. Ensure you trust both the source and contents of any SVG before attempting to rasterise it when using alternatives.
It might be possible to improve the error messaging; I had a quick look and it appears librsvg does not update the internal cairo status when an error occurs. For this example the message obtained via cairo_status_to_string is "no error has occurred", which suggests an improvement to librsvg itself may be possible, but this is very much left as an exercise for the reader.
Hi, is there not a simple way to embed/overlay a base64 encoded PNG on to an SVG and then render the whole thing as a PNG? Phantomjs includes the logo in the final PNG (see image below). sharp ignores the logo. Is this possible? Thanks!
sharp(Buffer.from(`
<svg width="200" height="230">
<path fill="none" stroke="steelblue" stroke-width="1.5" d="M0,0 L0,229 L199,229 L199,1 L0,0Z"></path>
<g transform="translate(5,5)">
<image width="100" height="9" xlink:href=""></image><g fill="none" font-size="10" font-family="sans-serif" text-anchor="middle" transform="translate(0, 25) scale(4.761904761904762, -45.5) translate(0, -4)"><path fill="#8a89a6" stroke="steelblue" stroke-width="0.1" d="M0,2L1,2 L2,2 L3,4 L4,4 L5,3 L6,3 L7,3 L8,2 L9,3 L10,2 L11,3 L12,3 L13,3 L14,3 L15,3 L16,3 L17,3 L18,3 L19,3 L20,2 L21,0 L22,3 L23,2 L24,3 L25,1 L26,2 L27,3 L28,2 L29,3 L30,3 L31,3 L32,3 L33,3 L34,3 L35,3 L36,3 L37,3 L38,3 L39,3 L40,3 M0,0Z"></path></g></g></svg>
`
))
.png()
.toBuffer()


@mahesh2000 Your example is not valid XML e.g. it is missing the xmlns:xlink="http://www.w3.org/1999/xlink" attribute from the svg element.