我正在使用 d3.js 作为 SVG 动态创建图表。这些图表是根据经过身份验证的用户的选择动态生成的。生成这些图表后,用户可以选择将生成的 SVG 下载为 PNG 或 PDF。
当前的工作流程如下:
// JAVASC
// get the element containing generated SVG
var svg = document.getElementById("chart-container");
// Extract the data as SVG text string
var svg_xml = (new XMLSerializer).serializeToString(svg);
// Submit the <FORM> to the server.
var form = document.getElementById("svgform");
form['output_format'].value = output_format; // can be either "pdf" or "png"
form['data'].value = svg_xml ;
form.submit();
FORM 元素是一个隐藏的表单,用于 POST 数据:
<form id="svgform" method="post" action="conversion.php">
<input type="hidden" id="output_format" name="output_format" value="">
<input type="hidden" id="data" name="data" value="">
</form>
PHP 文件将提供的 SVG 数据保存为临时文件:
// check for valid session, etc - omitted for brevity
$xmldat = $_POST['data']; // serialized XML representing the SVG element
if(simplexml_load_string($xmldat)===FALSE) { die; } // reject invalid XML
$fileformat = $_POST['output_format']; // chosen format for output; PNG or PDF
if ($fileformat != "pdf" && $fileformat != "png" ){ die; } // limited options for format
$fileformat = escapeshellarg($fileformat); // escape shell arguments that might have snuck in
// generate temporary file names with tempnam() - omitted for brevity
$handle = fopen($infile, "w");
fwrite($handle, $xmldat);
fclose($handle);
运行一个转换实用程序,它读取临时文件 ($infile) 并以指定的 $fileformat(PDF 或 PNG)创建一个新文件 ($outfile)。然后将生成的新文件返回给浏览器,并删除临时文件:
// headers etc generated - omitted for brevity
readfile($outfile);
unlink($infile); // delete temporary infile
unlink($outfile); // delete temporary outfile
我研究了使用 JavaScript (canvg(), 然后 toDataURL, 然后 document.write) 将 SVG 转换为 PNG,并且可以使用它来生成 PNG,但它不允许转换为 PDF。
那么:在将其写入文件之前,如何最好地清理或过滤提供给 conversion.php 的 SVG 数据?SVG 清理的当前状态是什么?PHP中有什么可用的?我应该使用基于白名单的方法来清理提供给 conversion.php 的 SVG 数据,还是有更好的方法?
(我不知道 XSLT,虽然我可以尝试学习它;我希望尽可能多地在 PHP 中进行清理。使用 Windows Server 2008,因此任何使用外部工具的解决方案都需要在该生态系统中可用。)