This repository has been archived on 2024-09-05. You can view files and clone it, but cannot push or open issues or pull requests.
shimmie2/ext/handle_svg/main.php

114 lines
3.4 KiB
PHP
Raw Normal View History

2021-12-14 18:32:47 +00:00
<?php
declare(strict_types=1);
namespace Shimmie2;
2018-02-20 21:35:43 +00:00
use enshrined\svgSanitize\Sanitizer;
class SVGFileHandler extends DataHandlerExtension
{
protected array $SUPPORTED_MIME = [MimeType::SVG];
2020-02-23 18:37:22 +00:00
2020-02-04 00:46:36 +00:00
/** @var SVGFileHandlerTheme */
protected ?Themelet $theme;
2020-02-04 00:46:36 +00:00
2020-02-23 18:37:22 +00:00
public function onPageRequest(PageRequestEvent $event)
{
2020-02-23 18:37:22 +00:00
global $page;
if ($event->page_matches("get_svg")) {
$id = int_escape($event->get_arg(0));
$image = Image::by_id($id);
$hash = $image->hash;
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);
2020-02-23 18:37:22 +00:00
$sanitizer = new Sanitizer();
$sanitizer->removeRemoteReferences(true);
$dirtySVG = file_get_contents(warehouse_path(Image::IMAGE_DIR, $hash));
$cleanSVG = $sanitizer->sanitize($dirtySVG);
$page->set_data($cleanSVG);
}
}
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;
$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;
}
protected function move_upload_to_archive(DataUploadEvent $event)
{
$sanitizer = new Sanitizer();
$sanitizer->removeRemoteReferences(true);
$dirtySVG = file_get_contents($event->tmpname);
$cleanSVG = $sanitizer->sanitize($dirtySVG);
$event->hash = md5($cleanSVG);
file_put_contents(warehouse_path(Image::IMAGE_DIR, $event->hash), $cleanSVG);
}
protected function create_thumb(string $hash, string $mime): bool
{
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')) {
create_image_thumb($hash, $mime);
2020-02-01 21:20:32 +00:00
} else {
create_image_thumb($hash, $mime, MediaEngine::IMAGICK);
2020-02-01 21:20:32 +00:00
}
return true;
} catch (MediaException $e) {
log_warning("handle_svg", "Could not generate thumbnail. " . $e->getMessage());
copy("ext/handle_svg/thumb.jpg", warehouse_path(Image::THUMBNAIL_DIR, $hash));
return false;
}
}
protected function check_contents(string $tmpname): bool
{
if (MimeType::get_for_file($tmpname)!==MimeType::SVG) {
return false;
}
$msp = new MiniSVGParser($tmpname);
return bool_escape($msp->valid);
}
}
class MiniSVGParser
{
public bool $valid = false;
public int $width=0;
public int $height=0;
private int $xml_depth=0;
public function __construct(string $file)
{
$xml_parser = xml_parser_create();
xml_set_element_handler($xml_parser, [$this, "startElement"], [$this, "endElement"]);
$this->valid = bool_escape(xml_parse($xml_parser, file_get_contents($file), true));
xml_parser_free($xml_parser);
}
2023-01-11 11:15:26 +00:00
public function startElement($parser, $name, $attrs): void
{
if ($name == "SVG" && $this->xml_depth == 0) {
$this->width = int_escape($attrs["WIDTH"]);
$this->height = int_escape($attrs["HEIGHT"]);
}
$this->xml_depth++;
}
2023-01-11 11:15:26 +00:00
public function endElement($parser, $name): void
{
$this->xml_depth--;
}
}