2021-12-14 18:32:47 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2023-01-10 22:44:09 +00:00
|
|
|
|
|
|
|
namespace Shimmie2;
|
|
|
|
|
2018-02-20 21:35:43 +00:00
|
|
|
use enshrined\svgSanitize\Sanitizer;
|
|
|
|
|
2019-06-09 13:22:48 -05:00
|
|
|
class SVGFileHandler extends DataHandlerExtension
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
protected array $SUPPORTED_MIME = [MimeType::SVG];
|
2020-02-23 18:37:22 +00:00
|
|
|
|
2020-02-04 00:46:36 +00:00
|
|
|
/** @var SVGFileHandlerTheme */
|
2023-06-27 15:56:49 +01:00
|
|
|
protected Themelet $theme;
|
2020-02-04 00:46:36 +00:00
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onPageRequest(PageRequestEvent $event): void
|
2019-06-25 15:17:13 -05:00
|
|
|
{
|
2020-02-23 18:37:22 +00:00
|
|
|
global $page;
|
2024-02-11 11:34:09 +00:00
|
|
|
if ($event->page_matches("get_svg/{id}")) {
|
|
|
|
$id = $event->get_iarg('id');
|
2024-02-20 00:22:25 +00:00
|
|
|
$image = Image::by_id_ex($id);
|
2020-02-23 18:37:22 +00:00
|
|
|
$hash = $image->hash;
|
2019-06-25 15:17:13 -05:00
|
|
|
|
2020-06-14 11:05:55 -05:00
|
|
|
$page->set_mime(MimeType::SVG);
|
2020-02-23 18:37:22 +00:00
|
|
|
$page->set_mode(PageMode::DATA);
|
2019-06-25 15:17:13 -05:00
|
|
|
|
2020-02-23 18:37:22 +00:00
|
|
|
$sanitizer = new Sanitizer();
|
|
|
|
$sanitizer->removeRemoteReferences(true);
|
2024-02-20 00:22:25 +00:00
|
|
|
$dirtySVG = \Safe\file_get_contents(warehouse_path(Image::IMAGE_DIR, $hash));
|
2020-02-23 18:37:22 +00:00
|
|
|
$cleanSVG = $sanitizer->sanitize($dirtySVG);
|
|
|
|
$page->set_data($cleanSVG);
|
2019-06-25 15:17:13 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-15 11:52:35 +00:00
|
|
|
public function onDataUpload(DataUploadEvent $event): void
|
2024-01-09 04:22:59 +00:00
|
|
|
{
|
|
|
|
global $config;
|
|
|
|
|
|
|
|
if ($this->supported_mime($event->mime)) {
|
|
|
|
// If the SVG handler intends to handle this file,
|
|
|
|
// then sanitise it before touching it
|
|
|
|
$sanitizer = new Sanitizer();
|
|
|
|
$sanitizer->removeRemoteReferences(true);
|
2024-02-20 00:22:25 +00:00
|
|
|
$dirtySVG = \Safe\file_get_contents($event->tmpname);
|
2024-01-20 20:48:47 +00:00
|
|
|
$cleanSVG = false_throws($sanitizer->sanitize($dirtySVG));
|
2024-01-09 04:22:59 +00:00
|
|
|
$event->hash = md5($cleanSVG);
|
2024-01-20 20:48:47 +00:00
|
|
|
$new_tmpname = shm_tempnam("svg");
|
2024-01-09 04:22:59 +00:00
|
|
|
file_put_contents($new_tmpname, $cleanSVG);
|
|
|
|
$event->set_tmpname($new_tmpname);
|
|
|
|
|
|
|
|
parent::onDataUpload($event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-23 18:37:22 +00:00
|
|
|
protected function media_check_properties(MediaCheckPropertiesEvent $event): void
|
|
|
|
{
|
|
|
|
$event->image->lossless = true;
|
|
|
|
$event->image->video = false;
|
|
|
|
$event->image->audio = false;
|
|
|
|
$event->image->image = true;
|
|
|
|
|
2023-02-22 23:37:37 +00:00
|
|
|
$msp = new MiniSVGParser($event->image->get_image_filename());
|
2020-02-23 18:37:22 +00:00
|
|
|
$event->image->width = $msp->width;
|
|
|
|
$event->image->height = $msp->height;
|
|
|
|
}
|
|
|
|
|
2024-01-09 04:49:19 +00:00
|
|
|
protected function create_thumb(Image $image): bool
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2019-06-18 13:45:59 -05:00
|
|
|
try {
|
2020-02-01 18:51:57 +00:00
|
|
|
// Normally we require imagemagick, but for unit tests we can use a no-op engine
|
2020-02-01 21:20:32 +00:00
|
|
|
if (defined('UNITTEST')) {
|
2024-01-09 04:49:19 +00:00
|
|
|
create_image_thumb($image);
|
2020-02-01 21:20:32 +00:00
|
|
|
} else {
|
2024-01-09 04:49:19 +00:00
|
|
|
create_image_thumb($image, MediaEngine::IMAGICK);
|
2020-02-01 21:20:32 +00:00
|
|
|
}
|
2019-06-18 13:45:59 -05:00
|
|
|
return true;
|
2019-06-24 10:05:16 -05:00
|
|
|
} catch (MediaException $e) {
|
2019-06-18 13:45:59 -05:00
|
|
|
log_warning("handle_svg", "Could not generate thumbnail. " . $e->getMessage());
|
2024-01-09 04:49:19 +00:00
|
|
|
copy("ext/handle_svg/thumb.jpg", $image->get_thumb_filename());
|
2019-06-18 13:45:59 -05:00
|
|
|
return false;
|
2019-05-28 17:59:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-14 23:43:50 +00:00
|
|
|
protected function check_contents(string $tmpname): bool
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
2023-11-11 21:49:12 +00:00
|
|
|
if (MimeType::get_for_file($tmpname) !== MimeType::SVG) {
|
2020-02-24 14:29:27 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-03-14 23:43:50 +00:00
|
|
|
$msp = new MiniSVGParser($tmpname);
|
2019-05-28 17:59:38 +01:00
|
|
|
return bool_escape($msp->valid);
|
|
|
|
}
|
2007-12-11 18:40:44 +00:00
|
|
|
}
|
2008-04-11 06:56:31 +00:00
|
|
|
|
2019-05-28 17:59:38 +01:00
|
|
|
class MiniSVGParser
|
|
|
|
{
|
2021-03-14 23:43:50 +00:00
|
|
|
public bool $valid = false;
|
2023-11-11 21:49:12 +00:00
|
|
|
public int $width = 0;
|
|
|
|
public int $height = 0;
|
|
|
|
private int $xml_depth = 0;
|
2019-05-28 17:59:38 +01:00
|
|
|
|
|
|
|
public function __construct(string $file)
|
|
|
|
{
|
|
|
|
$xml_parser = xml_parser_create();
|
|
|
|
xml_set_element_handler($xml_parser, [$this, "startElement"], [$this, "endElement"]);
|
2024-02-20 00:22:25 +00:00
|
|
|
$this->valid = bool_escape(xml_parse($xml_parser, \Safe\file_get_contents($file), true));
|
2019-05-28 17:59:38 +01:00
|
|
|
xml_parser_free($xml_parser);
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
/**
|
|
|
|
* @param array<string, mixed> $attrs
|
|
|
|
*/
|
|
|
|
public function startElement(mixed $parser, string $name, array $attrs): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
if ($name == "SVG" && $this->xml_depth == 0) {
|
|
|
|
$this->width = int_escape($attrs["WIDTH"]);
|
|
|
|
$this->height = int_escape($attrs["HEIGHT"]);
|
|
|
|
}
|
|
|
|
$this->xml_depth++;
|
|
|
|
}
|
|
|
|
|
2024-01-20 14:10:59 +00:00
|
|
|
public function endElement(mixed $parser, string $name): void
|
2019-05-28 17:59:38 +01:00
|
|
|
{
|
|
|
|
$this->xml_depth--;
|
|
|
|
}
|
2008-04-11 06:56:31 +00:00
|
|
|
}
|