From ef3548ae2f24dafacfd6af37ca8057d7a0209980 Mon Sep 17 00:00:00 2001
From: =?utf8?q?Rapha=C3=ABl=20Gertz?= <git@rapsys.eu>
Date: Sun, 8 Dec 2024 03:19:02 +0100
Subject: [PATCH] Merge Image and Map into Controller Add alias, config, ctx
 and version member variables Support jpeg, png and webp formats Store files
 in new cache tree Cleanup

---
 Controller.php                 | 839 +++++++++++++++++++++++++++++++++
 Controller/ImageController.php | 291 ------------
 Controller/MapController.php   | 507 --------------------
 3 files changed, 839 insertions(+), 798 deletions(-)
 create mode 100644 Controller.php
 delete mode 100644 Controller/ImageController.php
 delete mode 100644 Controller/MapController.php

diff --git a/Controller.php b/Controller.php
new file mode 100644
index 0000000..523c950
--- /dev/null
+++ b/Controller.php
@@ -0,0 +1,839 @@
+<?php declare(strict_types=1);
+
+/*
+ * This file is part of the Rapsys PackBundle package.
+ *
+ * (c) Raphaël Gertz <symfony@rapsys.eu>
+ *
+ * For the full copyright and license information, please view the LICENSE
+ * file that was distributed with this source code.
+ */
+
+namespace Rapsys\PackBundle;
+
+use Rapsys\PackBundle\Util\ImageUtil;
+use Rapsys\PackBundle\Util\SluggerUtil;
+
+use Psr\Container\ContainerInterface;
+
+use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
+use Symfony\Component\Filesystem\Exception\IOExceptionInterface;
+use Symfony\Component\Filesystem\Filesystem;
+use Symfony\Component\HttpFoundation\BinaryFileResponse;
+use Symfony\Component\HttpFoundation\HeaderUtils;
+use Symfony\Component\HttpFoundation\Request;
+use Symfony\Component\HttpFoundation\Response;
+use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
+use Symfony\Component\Routing\RequestContext;
+use Symfony\Contracts\Service\ServiceSubscriberInterface;
+
+/**
+ * {@inheritdoc}
+ */
+class Controller extends AbstractController implements ServiceSubscriberInterface {
+	/**
+	 * Alias string
+	 */
+	protected string $alias;
+
+	/**
+	 * Config array
+	 */
+	protected array $config;
+
+	/**
+	 * Stream context
+	 */
+	protected mixed $ctx;
+
+	/**
+	 * Version string
+	 */
+	protected string $version;
+
+	/**
+	 * Creates a new image controller
+	 *
+	 * @param ContainerInterface $container The ContainerInterface instance
+	 * @param ImageUtil $image The MapUtil instance
+	 * @param SluggerUtil $slugger The SluggerUtil instance
+	 */
+	function __construct(protected ContainerInterface $container, protected ImageUtil $image, protected SluggerUtil $slugger) {
+		//Retrieve config
+		$this->config = $container->getParameter($this->alias = RapsysPackBundle::getAlias());
+
+		//Set ctx
+		$this->ctx = stream_context_create(
+			[
+				'http' => [
+					'max_redirects' => $_ENV['RAPSYSPACK_REDIRECT'] ?? 20,
+					'timeout' => $_ENV['RAPSYSPACK_TIMEOUT'] ?? (($timeout = ini_get('default_socket_timeout')) !== false && $timeout !== '' ? (float)$timeout : 60),
+					'user_agent' => $_ENV['RAPSYSPACK_AGENT'] ?? (($agent = ini_get('user_agent')) !== false && $agent !== '' ? (string)$agent : $this->alias.'/'.($this->version = RapsysPackBundle::getVersion()))
+				]
+			]
+		);
+	}
+
+	/**
+	 * Return captcha image
+	 *
+	 * @param Request $request The Request instance
+	 * @param string $hash The hash
+	 * @param string $equation The shorted equation
+	 * @param int $width The width
+	 * @param int $height The height
+	 * @return Response The rendered image
+	 */
+	public function captcha(Request $request, string $hash, string $equation, int $width, int $height, string $_format): Response {
+		//Without matching hash
+		if ($hash !== $this->slugger->serialize([$equation, $width, $height])) {
+			//Throw new exception
+			throw new NotFoundHttpException(sprintf('Unable to match captcha hash: %s', $hash));
+		//Without valid format
+		} elseif ($_format !== 'jpeg' && $_format !== 'png' && $_format !== 'webp') {
+			//Throw new exception
+			throw new NotFoundHttpException('Invalid thumb format');
+		}
+
+		//Unshort equation
+		$equation = $this->slugger->unshort($short = $equation);
+
+		//Set hashed tree
+		$hashed = str_split(strval($equation));
+
+		//Set captcha
+		$captcha = $this->config['cache'].'/'.$this->config['prefixes']['captcha'].'/'.$hashed[0].'/'.$hashed[4].'/'.$hashed[8].'/'.$short.'.'.$_format;
+
+		//Without captcha up to date file
+		if (!is_file($captcha) || !($mtime = stat($captcha)['mtime']) || $mtime < (new \DateTime('-1 hour'))->getTimestamp()) {
+			//Without existing captcha path
+			if (!is_dir($dir = dirname($captcha))) {
+				//Create filesystem object
+				$filesystem = new Filesystem();
+
+				try {
+					//Create path
+					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+					//XXX: on CoW filesystems execute a chattr +C before filling
+					$filesystem->mkdir($dir, 0775);
+				} catch (IOExceptionInterface $e) {
+					//Throw error
+					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
+				}
+			}
+
+			//Create image instance
+			$image = new \Imagick();
+
+			//Add imagick draw instance
+			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
+			$draw = new \ImagickDraw();
+
+			//Set text antialias
+			$draw->setTextAntialias(true);
+
+			//Set stroke antialias
+			$draw->setStrokeAntialias(true);
+
+			//Set text alignment
+			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
+
+			//Set gravity
+			$draw->setGravity(\Imagick::GRAVITY_CENTER);
+
+			//Set fill color
+			$draw->setFillColor($this->config['captcha']['fill']);
+
+			//Set stroke color
+			$draw->setStrokeColor($this->config['captcha']['border']);
+
+			//Set font size
+			$draw->setFontSize($this->config['captcha']['size'] / 1.5);
+
+			//Set stroke width
+			$draw->setStrokeWidth($this->config['captcha']['thickness'] / 3);
+
+			//Set rotation
+			$draw->rotate($rotate = (rand(25, 75)*(rand(0,1)?-.1:.1)));
+
+			//Get font metrics
+			$metrics2 = $image->queryFontMetrics($draw, strval('stop spam'));
+
+			//Add annotation
+			$draw->annotation($width / 2 - ceil(rand(intval(-$metrics2['textWidth']), intval($metrics2['textWidth'])) / 2) - abs($rotate), ceil($metrics2['textHeight'] + $metrics2['descender'] + $metrics2['ascender']) / 2 - $this->config['captcha']['thickness'] - $rotate, strval('stop spam'));
+
+			//Set rotation
+			$draw->rotate(-$rotate);
+
+			//Set font size
+			$draw->setFontSize($this->config['captcha']['size']);
+
+			//Set stroke width
+			$draw->setStrokeWidth($this->config['captcha']['thickness']);
+
+			//Set rotation
+			$draw->rotate($rotate = (rand(25, 50)*(rand(0,1)?-.1:.1)));
+
+			//Get font metrics
+			$metrics = $image->queryFontMetrics($draw, strval($equation));
+
+			//Add annotation
+			$draw->annotation($width / 2, ceil($metrics['textHeight'] + $metrics['descender'] + $metrics['ascender']) / 2 - $this->config['captcha']['thickness'], strval($equation));
+
+			//Set rotation
+			$draw->rotate(-$rotate);
+
+			//Add new image
+			#$image->newImage(intval(ceil($metrics['textWidth'])), intval(ceil($metrics['textHeight'] + $metrics['descender'])), new \ImagickPixel($this->config['captcha']['background']), 'jpeg');
+			$image->newImage($width, $height, new \ImagickPixel($this->config['captcha']['background']), $_format);
+
+			//Draw on image
+			$image->drawImage($draw);
+
+			//Strip image exif data and properties
+			$image->stripImage();
+
+			//Set compression quality
+			$image->setImageCompressionQuality(70);
+
+			//Save captcha
+			if (!$image->writeImage($captcha)) {
+				//Throw error
+				throw new \Exception(sprintf('Unable to write image "%s"', $captcha));
+			}
+
+			//Set mtime
+			$mtime = stat($captcha)['mtime'];
+		}
+
+		//Read captcha from cache
+		$response = new BinaryFileResponse($captcha);
+
+		//Set file name
+		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'captcha-stop-spam-'.str_replace([' ', '*', '+'], ['-', 'mul', 'add'], $equation).'.'.$_format);
+
+		//Set etag
+		$response->setEtag(md5($hash));
+
+		//Set last modified
+		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
+
+		//Set as public
+		$response->setPublic();
+
+		//Return 304 response if not modified
+		$response->isNotModified($request);
+
+		//Return response
+		return $response;
+	}
+
+	/**
+	 * Return facebook image
+	 *
+	 * @param Request $request The Request instance
+	 * @param string $hash The hash
+	 * @param string $path The image path
+	 * @param int $width The width
+	 * @param int $height The height
+	 * @return Response The rendered image
+	 */
+	public function facebook(Request $request, string $hash, string $path, int $width, int $height, string $_format): Response {
+		//Without matching hash
+		if ($hash !== $this->slugger->serialize([$path, $width, $height])) {
+			//Throw new exception
+			throw new NotFoundHttpException(sprintf('Unable to match facebook hash: %s', $hash));
+		//Without matching format
+		} elseif ($_format !== 'jpeg' && $_format !== 'png' && $_format !== 'webp') {
+			//Throw new exception
+			throw new NotFoundHttpException(sprintf('Invalid facebook format: %s', $_format));
+		}
+
+		//Unshort path
+		$path = $this->slugger->unshort($short = $path);
+
+		//Without facebook file
+		if (!is_file($facebook = $this->config['cache'].'/'.$this->config['prefixes']['facebook'].$path.'.'.$_format)) {
+			//Throw new exception
+			throw new NotFoundHttpException('Unable to get facebook file');
+		}
+
+		//Read facebook from cache
+		$response = new BinaryFileResponse($facebook);
+
+		//Set file name
+		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'facebook-'.$hash.'.'.$_format);
+
+		//Set etag
+		$response->setEtag(md5($hash));
+
+		//Set last modified
+		$response->setLastModified(\DateTime::createFromFormat('U', strval(stat($facebook)['mtime'])));
+
+		//Set as public
+		$response->setPublic();
+
+		//Return 304 response if not modified
+		$response->isNotModified($request);
+
+		//Return response
+		return $response;
+	}
+
+	/**
+	 * Return map image
+	 *
+	 * @param Request $request The Request instance
+	 * @param string $hash The hash
+	 * @param int $updated The updated timestamp
+	 * @param float $latitude The latitude
+	 * @param float $longitude The longitude
+	 * @param int $zoom The zoom
+	 * @param int $width The width
+	 * @param int $height The height
+	 * @return Response The rendered image
+	 */
+	public function map(Request $request, string $hash, float $latitude, float $longitude, int $height, int $width, int $zoom, string $_format): Response {
+		//Without matching hash
+		if ($hash !== $this->slugger->hash([$height, $width, $zoom, $latitude, $longitude])) {
+			//Throw new exception
+			throw new NotFoundHttpException(sprintf('Unable to match map hash: %s', $hash));
+		}
+
+		//Set map
+		$map = $this->config['cache'].'/'.$this->config['prefixes']['map'].'/'.$zoom.'/'.($latitude*1000000%10).'/'.($longitude*1000000%10).'/'.$latitude.','.$longitude.'-'.$zoom.'-'.$width.'x'.$height.'.'.$_format;
+
+		//Without map file
+		//TODO: refresh after config modification ?
+		if (!is_file($map)) {
+			//Without existing map path
+			if (!is_dir($dir = dirname($map))) {
+				//Create filesystem object
+				$filesystem = new Filesystem();
+
+				try {
+					//Create path
+					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+					//XXX: on CoW filesystems execute a chattr +C before filling
+					$filesystem->mkdir($dir, 0775);
+				} catch (IOExceptionInterface $e) {
+					//Throw error
+					throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
+				}
+			}
+
+			//Create image instance
+			$image = new \Imagick();
+
+			//Add new image
+			$image->newImage($width, $height, new \ImagickPixel('transparent'), $_format);
+
+			//Create tile instance
+			$tile = new \Imagick();
+
+			//Get tile xy
+			$centerX = $this->image->longitudeToX($longitude, $zoom);
+			$centerY = $this->image->latitudeToY($latitude, $zoom);
+
+			//Calculate start xy
+			$startX = floor(floor($centerX) - $width / $this->config['map']['tz']);
+			$startY = floor(floor($centerY) - $height / $this->config['map']['tz']);
+
+			//Calculate end xy
+			$endX = ceil(ceil($centerX) + $width / $this->config['map']['tz']);
+			$endY = ceil(ceil($centerY) + $height / $this->config['map']['tz']);
+
+			for($x = $startX; $x <= $endX; $x++) {
+				for($y = $startY; $y <= $endY; $y++) {
+					//Set cache path
+					$cache = $this->config['cache'].'/'.$this->config['prefixes']['map'].'/'.$zoom.'/'.($x%10).'/'.($y%10).'/'.$x.','.$y.'.png';
+
+					//Without cache image
+					if (!is_file($cache)) {
+						//Set tile url
+						$tileUri = str_replace(['{Z}', '{X}', '{Y}'], [$zoom, $x, $y], $this->config['servers'][$this->config['map']['server']]);
+
+						//Without cache path
+						if (!is_dir($dir = dirname($cache))) {
+							//Create filesystem object
+							$filesystem = new Filesystem();
+
+							try {
+								//Create path
+								//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+								$filesystem->mkdir($dir, 0775);
+							} catch (IOExceptionInterface $e) {
+								//Throw error
+								throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
+							}
+						}
+
+						//Store tile in cache
+						file_put_contents($cache, file_get_contents($tileUri, false, $this->ctx));
+					}
+
+					//Set dest x
+					$destX = intval(floor($width / 2 - $this->config['map']['tz'] * ($centerX - $x)));
+
+					//Set dest y
+					$destY = intval(floor($height / 2 - $this->config['map']['tz'] * ($centerY - $y)));
+
+					//Read tile from cache
+					$tile->readImage($cache);
+
+					//Compose image
+					$image->compositeImage($tile, \Imagick::COMPOSITE_OVER, $destX, $destY);
+
+					//Clear tile
+					$tile->clear();
+				}
+			}
+
+			//Add imagick draw instance
+			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
+			$draw = new \ImagickDraw();
+
+			//Set text antialias
+			$draw->setTextAntialias(true);
+
+			//Set stroke antialias
+			$draw->setStrokeAntialias(true);
+
+			//Set text alignment
+			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
+
+			//Set gravity
+			$draw->setGravity(\Imagick::GRAVITY_CENTER);
+
+			//Set fill color
+			$draw->setFillColor($this->config['map']['fill']);
+
+			//Set stroke color
+			$draw->setStrokeColor($this->config['map']['border']);
+
+			//Set stroke width
+			$draw->setStrokeWidth($this->config['map']['thickness']);
+
+			//Draw circle
+			$draw->circle($width/2 - $this->config['map']['radius'], $height/2 - $this->config['map']['radius'], $width/2 + $this->config['map']['radius'], $height/2 + $this->config['map']['radius']);
+
+			//Draw on image
+			$image->drawImage($draw);
+
+			//Strip image exif data and properties
+			$image->stripImage();
+
+			//Add latitude
+			//XXX: not supported by imagick :'(
+			$image->setImageProperty('exif:GPSLatitude', $this->image->latitudeToSexagesimal($latitude));
+
+			//Add longitude
+			//XXX: not supported by imagick :'(
+			$image->setImageProperty('exif:GPSLongitude', $this->image->longitudeToSexagesimal($longitude));
+
+			//Set progressive jpeg
+			$image->setInterlaceScheme(\Imagick::INTERLACE_PLANE);
+
+			//Set compression quality
+			$image->setImageCompressionQuality($this->config['map']['quality']);
+
+			//Save image
+			if (!$image->writeImage($map)) {
+				//Throw error
+				throw new \Exception(sprintf('Unable to write image "%s"', $map));
+			}
+		}
+
+		//Read map from cache
+		$response = new BinaryFileResponse($map);
+
+		//Set file name
+		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, basename($map));
+
+		//Set etag
+		$response->setEtag(md5(serialize([$height, $width, $zoom, $latitude, $longitude])));
+
+		//Set last modified
+		$response->setLastModified(\DateTime::createFromFormat('U', strval(stat($map)['mtime'])));
+
+		//Disable robot index
+		$response->headers->set('X-Robots-Tag', 'noindex');
+
+		//Set as public
+		$response->setPublic();
+
+		//Return 304 response if not modified
+		$response->isNotModified($request);
+
+		//Return response
+		return $response;
+	}
+
+	/**
+	 * Return multi map image
+	 *
+	 * @param Request $request The Request instance
+	 * @param string $hash The hash
+	 * @param int $updated The updated timestamp
+	 * @param float $latitude The latitude
+	 * @param float $longitude The longitude
+	 * @param string $coordinates The coordinates
+	 * @param int $zoom The zoom
+	 * @param int $width The width
+	 * @param int $height The height
+	 * @return Response The rendered image
+	 */
+	public function multi(Request $request, string $hash, string $coordinate, int $height, int $width, int $zoom, string $_format): Response {
+		//Without matching hash
+		if ($hash !== $this->slugger->hash([$height, $width, $zoom, $coordinate])) {
+			//Throw new exception
+			throw new NotFoundHttpException(sprintf('Unable to match multi map hash: %s', $hash));
+		}
+
+		//Set latitudes and longitudes array
+		$latitudes = $longitudes = [];
+
+		//Set coordinates
+		$coordinates = array_map(
+			function ($v) use (&$latitudes, &$longitudes) {
+				list($latitude, $longitude) = explode(',', $v);
+				$latitudes[] = $latitude;
+				$longitudes[] = $longitude;
+				return [ $latitude, $longitude ];
+			},
+			explode('-', $coordinate)
+		);
+
+		//Set latitude
+		$latitude = round((min($latitudes)+max($latitudes))/2, 6);
+
+		//Set longitude
+		$longitude = round((min($longitudes)+max($longitudes))/2, 6);
+
+		//Set map
+		$map = $this->config['cache'].'/'.$this->config['prefixes']['multi'].'/'.$zoom.'/'.($latitude*1000000%10).'/'.($longitude*1000000%10).'/'.$coordinate.'-'.$zoom.'-'.$width.'x'.$height.'.'.$_format;
+
+		//Without map file
+		if (!is_file($map)) {
+			//Without existing multi path
+			if (!is_dir($dir = dirname($map))) {
+				//Create filesystem object
+				$filesystem = new Filesystem();
+
+				try {
+					//Create path
+					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+					//XXX: on CoW filesystems execute a chattr +C before filling
+					$filesystem->mkdir($dir, 0775);
+				} catch (IOExceptionInterface $e) {
+					//Throw error
+					throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
+				}
+			}
+
+			//Create image instance
+			$image = new \Imagick();
+
+			//Add new image
+			$image->newImage($width, $height, new \ImagickPixel('transparent'), $_format);
+
+			//Create tile instance
+			$tile = new \Imagick();
+
+			//Get tile xy
+			$centerX = $this->image->longitudeToX($longitude, $zoom);
+			$centerY = $this->image->latitudeToY($latitude, $zoom);
+
+			//Calculate start xy
+			$startX = floor(floor($centerX) - $width / $this->config['multi']['tz']);
+			$startY = floor(floor($centerY) - $height / $this->config['multi']['tz']);
+
+			//Calculate end xy
+			$endX = ceil(ceil($centerX) + $width / $this->config['multi']['tz']);
+			$endY = ceil(ceil($centerY) + $height / $this->config['multi']['tz']);
+
+			for($x = $startX; $x <= $endX; $x++) {
+				for($y = $startY; $y <= $endY; $y++) {
+					//Set cache path
+					$cache = $this->config['cache'].'/'.$this->config['prefixes']['multi'].'/'.$zoom.'/'.($x%10).'/'.($y%10).'/'.$x.','.$y.'.png';
+
+					//Without cache image
+					if (!is_file($cache)) {
+						//Set tile url
+						$tileUri = str_replace(['{Z}', '{X}', '{Y}'], [$zoom, $x, $y], $this->config['servers'][$this->config['multi']['server']]);
+
+						//Without cache path
+						if (!is_dir($dir = dirname($cache))) {
+							//Create filesystem object
+							$filesystem = new Filesystem();
+
+							try {
+								//Create path
+								//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+								$filesystem->mkdir($dir, 0775);
+							} catch (IOExceptionInterface $e) {
+								//Throw error
+								throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
+							}
+						}
+
+						//Store tile in cache
+						file_put_contents($cache, file_get_contents($tileUri, false, $this->ctx));
+					}
+
+					//Set dest x
+					$destX = intval(floor($width / 2 - $this->config['multi']['tz'] * ($centerX - $x)));
+
+					//Set dest y
+					$destY = intval(floor($height / 2 - $this->config['multi']['tz'] * ($centerY - $y)));
+
+					//Read tile from cache
+					$tile->readImage($cache);
+
+					//Compose image
+					$image->compositeImage($tile, \Imagick::COMPOSITE_OVER, $destX, $destY);
+
+					//Clear tile
+					$tile->clear();
+				}
+			}
+
+			//Add imagick draw instance
+			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
+			$draw = new \ImagickDraw();
+
+			//Set text antialias
+			$draw->setTextAntialias(true);
+
+			//Set stroke antialias
+			$draw->setStrokeAntialias(true);
+
+			//Set text alignment
+			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
+
+			//Set gravity
+			$draw->setGravity(\Imagick::GRAVITY_CENTER);
+
+			//Iterate on locations
+			foreach($coordinates as $id => $coordinate) {
+				//Get coordinates
+				list($clatitude, $clongitude) = $coordinate;
+
+				//Set dest x
+				$destX = intval(floor($width / 2 - $this->config['multi']['tz'] * ($centerX - $this->image->longitudeToX(floatval($clongitude), $zoom))));
+
+				//Set dest y
+				$destY = intval(floor($height / 2 - $this->config['multi']['tz'] * ($centerY - $this->image->latitudeToY(floatval($clatitude), $zoom))));
+
+				//Set fill color
+				$draw->setFillColor($this->config['multi']['fill']);
+
+				//Set font size
+				$draw->setFontSize($this->config['multi']['size']);
+
+				//Set stroke color
+				$draw->setStrokeColor($this->config['multi']['border']);
+
+				//Set circle radius
+				$radius = $this->config['multi']['radius'];
+
+				//Set stroke width
+				$stroke = $this->config['multi']['thickness'];
+
+				//With matching position
+				if ($clatitude === $latitude && $clongitude == $longitude) {
+					//Set fill color
+					$draw->setFillColor($this->config['multi']['highfill']);
+
+					//Set font size
+					$draw->setFontSize($this->config['multi']['highsize']);
+
+					//Set stroke color
+					$draw->setStrokeColor($this->config['multi']['highborder']);
+
+					//Set circle radius
+					$radius = $this->config['multi']['highradius'];
+
+					//Set stroke width
+					$stroke = $this->config['multi']['highthickness'];
+				}
+
+				//Set stroke width
+				$draw->setStrokeWidth($stroke);
+
+				//Draw circle
+				$draw->circle($destX - $radius, $destY - $radius, $destX + $radius, $destY + $radius);
+
+				//Set fill color
+				$draw->setFillColor($draw->getStrokeColor());
+
+				//Set stroke width
+				$draw->setStrokeWidth($stroke / 4);
+
+				//Get font metrics
+				#$metrics = $image->queryFontMetrics($draw, strval($id));
+
+				//Add annotation
+				$draw->annotation($destX - $radius, $destY + $stroke, strval($id));
+			}
+
+			//Draw on image
+			$image->drawImage($draw);
+
+			//Strip image exif data and properties
+			$image->stripImage();
+
+			//Add latitude
+			//XXX: not supported by imagick :'(
+			$image->setImageProperty('exif:GPSLatitude', $this->image->latitudeToSexagesimal($latitude));
+
+			//Add longitude
+			//XXX: not supported by imagick :'(
+			$image->setImageProperty('exif:GPSLongitude', $this->image->longitudeToSexagesimal($longitude));
+
+			//Add description
+			//XXX: not supported by imagick :'(
+			#$image->setImageProperty('exif:Description', $caption);
+
+			//Set progressive jpeg
+			$image->setInterlaceScheme(\Imagick::INTERLACE_PLANE);
+
+			//Set compression quality
+			$image->setImageCompressionQuality($this->config['multi']['quality']);
+
+			//Save image
+			if (!$image->writeImage($map)) {
+				//Throw error
+				throw new \Exception(sprintf('Unable to write image "%s"', $path));
+			}
+		}
+
+		//Read map from cache
+		$response = new BinaryFileResponse($map);
+
+		//Set file name
+		#$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'multimap-'.$latitude.','.$longitude.'-'.$zoom.'-'.$width.'x'.$height.'.jpeg');
+		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, basename($map));
+
+		//Set etag
+		$response->setEtag(md5(serialize([$height, $width, $zoom, $coordinate])));
+
+		//Set last modified
+		$response->setLastModified(\DateTime::createFromFormat('U', strval(stat($map)['mtime'])));
+
+		//Disable robot index
+		$response->headers->set('X-Robots-Tag', 'noindex');
+
+		//Set as public
+		$response->setPublic();
+
+		//Return 304 response if not modified
+		$response->isNotModified($request);
+
+		//Return response
+		return $response;
+	}
+
+	/**
+	 * Return thumb image
+	 *
+	 * @param Request $request The Request instance
+	 * @param string $hash The hash
+	 * @param string $path The image path
+	 * @param int $width The width
+	 * @param int $height The height
+	 * @return Response The rendered image
+	 */
+	public function thumb(Request $request, string $hash, string $path, int $width, int $height, string $_format): Response {
+		//Without matching hash
+		if ($hash !== $this->slugger->serialize([$path, $width, $height])) {
+			//Throw new exception
+			throw new NotFoundHttpException('Invalid thumb hash');
+		//Without valid format
+		} elseif ($_format !== 'jpeg' && $_format !== 'png' && $_format !== 'webp') {
+			//Throw new exception
+			throw new NotFoundHttpException('Invalid thumb format');
+		}
+
+		//Unshort path
+		$path = $this->slugger->unshort($short = $path);
+
+		//Set thumb
+		$thumb = $this->config['cache'].'/'.$this->config['prefixes']['thumb'].$path.'.'.$_format;
+
+		//Without file
+		if (!is_file($path) || !($updated = stat($path)['mtime'])) {
+			//Throw new exception
+			throw new NotFoundHttpException('Unable to get thumb file');
+		}
+
+		//Without thumb up to date file
+		if (!is_file($thumb) || !($mtime = stat($thumb)['mtime']) || $mtime < $updated) {
+			//Without existing thumb path
+			if (!is_dir($dir = dirname($thumb))) {
+				//Create filesystem object
+				$filesystem = new Filesystem();
+
+				try {
+					//Create path
+					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
+					//XXX: on CoW filesystems execute a chattr +C before filling
+					$filesystem->mkdir($dir, 0775);
+				} catch (IOExceptionInterface $e) {
+					//Throw error
+					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
+				}
+			}
+
+			//Create image instance
+			$image = new \Imagick();
+
+			//Read image
+			$image->readImage(realpath($path));
+
+			//Crop using aspect ratio
+			//XXX: for better result upload image directly in aspect ratio :)
+			$image->cropThumbnailImage($width, $height);
+
+			//Strip image exif data and properties
+			$image->stripImage();
+
+			//Set compression quality
+			//TODO: ajust that
+			$image->setImageCompressionQuality(70);
+
+			//Set image format
+			#$image->setImageFormat($_format);
+
+			//Save thumb
+			if (!$image->writeImage($thumb)) {
+				//Throw error
+				throw new \Exception(sprintf('Unable to write image "%s"', $thumb));
+			}
+
+			//Set mtime
+			$mtime = stat($thumb)['mtime'];
+		}
+
+		//Read thumb from cache
+		$response = new BinaryFileResponse($thumb);
+
+		//Set file name
+		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'thumb-'.$hash.'.'.$_format);
+
+		//Set etag
+		$response->setEtag(md5($hash));
+
+		//Set last modified
+		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
+
+		//Set as public
+		$response->setPublic();
+
+		//Return 304 response if not modified
+		$response->isNotModified($request);
+
+		//Return response
+		return $response;
+	}
+}
diff --git a/Controller/ImageController.php b/Controller/ImageController.php
deleted file mode 100644
index 13865a7..0000000
--- a/Controller/ImageController.php
+++ /dev/null
@@ -1,291 +0,0 @@
-<?php declare(strict_types=1);
-
-/*
- * This file is part of the Rapsys PackBundle package.
- *
- * (c) Raphaël Gertz <symfony@rapsys.eu>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
-
-namespace Rapsys\PackBundle\Controller;
-
-use Rapsys\PackBundle\Util\ImageUtil;
-use Rapsys\PackBundle\Util\SluggerUtil;
-
-use Psr\Container\ContainerInterface;
-
-use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
-use Symfony\Component\Filesystem\Exception\IOExceptionInterface;
-use Symfony\Component\Filesystem\Filesystem;
-use Symfony\Component\HttpFoundation\BinaryFileResponse;
-use Symfony\Component\HttpFoundation\HeaderUtils;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\HttpFoundation\Response;
-use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
-use Symfony\Component\Routing\RequestContext;
-use Symfony\Contracts\Service\ServiceSubscriberInterface;
-
-/**
- * {@inheritdoc}
- */
-class ImageController extends AbstractController implements ServiceSubscriberInterface {
-	/**
-	 * Creates a new image controller
-	 *
-	 * @param ContainerInterface $container The ContainerInterface instance
-	 * @param ImageUtil $image The MapUtil instance
-	 * @param SluggerUtil $slugger The SluggerUtil instance
-	 * @param string $cache The cache path
-	 * @param string $path The public path
-	 * @param string $prefix The prefix
-	 */
-	function __construct(protected ContainerInterface $container, protected ImageUtil $image, protected SluggerUtil $slugger, protected string $cache = '../var/cache', protected string $path = './bundles/rapsyspack', protected string $prefix = 'image') {
-	}
-
-	/**
-	 * Return captcha image
-	 *
-	 * @param Request $request The Request instance
-	 * @param string $hash The hash
-	 * @param int $updated The updated timestamp
-	 * @param string $equation The shorted equation
-	 * @param int $width The width
-	 * @param int $height The height
-	 * @return Response The rendered image
-	 */
-	public function captcha(Request $request, string $hash, int $updated, string $equation, int $width, int $height): Response {
-		//Without matching hash
-		if ($hash !== $this->slugger->serialize([$updated, $equation, $width, $height])) {
-			//Throw new exception
-			throw new NotFoundHttpException(sprintf('Unable to match captcha hash: %s', $hash));
-		}
-
-		//Set hashed tree
-		$hashed = array_reverse(str_split(strval($updated)));
-
-		//Set captcha
-		$captcha = $this->path.'/'.$this->prefix.'/'.$hashed[0].'/'.$hashed[1].'/'.$hashed[2].'/'.$updated.'/'.$equation.'/'.$width.'x'.$height.'.jpeg';
-
-		//Unshort equation
-		$equation = $this->slugger->unshort($equation);
-
-		//Without captcha up to date file
-		if (!is_file($captcha) || !($mtime = stat($captcha)['mtime']) || $mtime < $updated) {
-			//Without existing captcha path
-			if (!is_dir($dir = dirname($captcha))) {
-				//Create filesystem object
-				$filesystem = new Filesystem();
-
-				try {
-					//Create path
-					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-					//XXX: on CoW filesystems execute a chattr +C before filling
-					$filesystem->mkdir($dir, 0775);
-				} catch (IOExceptionInterface $e) {
-					//Throw error
-					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
-				}
-			}
-
-			//Create image instance
-			$image = new \Imagick();
-
-			//Add imagick draw instance
-			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
-			$draw = new \ImagickDraw();
-
-			//Set text antialias
-			$draw->setTextAntialias(true);
-
-			//Set stroke antialias
-			$draw->setStrokeAntialias(true);
-
-			//Set text alignment
-			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
-
-			//Set gravity
-			$draw->setGravity(\Imagick::GRAVITY_CENTER);
-
-			//Set fill color
-			$draw->setFillColor($this->image->getFill());
-
-			//Set stroke color
-			$draw->setStrokeColor($this->image->getStroke());
-
-			//Set font size
-			$draw->setFontSize($this->image->getFontSize() / 1.5);
-
-			//Set stroke width
-			$draw->setStrokeWidth($this->image->getStrokeWidth() / 3);
-
-			//Set rotation
-			$draw->rotate($rotate = (rand(25, 75)*(rand(0,1)?-.1:.1)));
-
-			//Get font metrics
-			$metrics2 = $image->queryFontMetrics($draw, strval('stop spam'));
-
-			//Add annotation
-			$draw->annotation($width / 2 - ceil(rand(intval(-$metrics2['textWidth']), intval($metrics2['textWidth'])) / 2) - abs($rotate), ceil($metrics2['textHeight'] + $metrics2['descender'] + $metrics2['ascender']) / 2 - $this->image->getStrokeWidth() - $rotate, strval('stop spam'));
-
-			//Set rotation
-			$draw->rotate(-$rotate);
-
-			//Set font size
-			$draw->setFontSize($this->image->getFontSize());
-
-			//Set stroke width
-			$draw->setStrokeWidth($this->image->getStrokeWidth());
-
-			//Set rotation
-			$draw->rotate($rotate = (rand(25, 50)*(rand(0,1)?-.1:.1)));
-
-			//Get font metrics
-			$metrics = $image->queryFontMetrics($draw, strval($equation));
-
-			//Add annotation
-			$draw->annotation($width / 2, ceil($metrics['textHeight'] + $metrics['descender'] + $metrics['ascender']) / 2 - $this->image->getStrokeWidth(), strval($equation));
-
-			//Set rotation
-			$draw->rotate(-$rotate);
-
-			//Add new image
-			#$image->newImage(intval(ceil($metrics['textWidth'])), intval(ceil($metrics['textHeight'] + $metrics['descender'])), new \ImagickPixel($this->image->getBackground()), 'jpeg');
-			$image->newImage($width, $height, new \ImagickPixel($this->image->getBackground()), 'jpeg');
-
-			//Draw on image
-			$image->drawImage($draw);
-
-			//Strip image exif data and properties
-			$image->stripImage();
-
-			//Set compression quality
-			$image->setImageCompressionQuality(70);
-
-			//Save captcha
-			if (!$image->writeImage($captcha)) {
-				//Throw error
-				throw new \Exception(sprintf('Unable to write image "%s"', $captcha));
-			}
-
-			//Set mtime
-			$mtime = stat($captcha)['mtime'];
-		}
-
-		//Read captcha from cache
-		$response = new BinaryFileResponse($captcha);
-
-		//Set file name
-		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'captcha-stop-spam-'.str_replace([' ', '*', '+'], ['-', 'mul', 'add'], $equation).'-'.$width.'x'.$height.'.jpeg');
-
-		//Set etag
-		$response->setEtag(md5($hash));
-
-		//Set last modified
-		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
-
-		//Set as public
-		$response->setPublic();
-
-		//Return 304 response if not modified
-		$response->isNotModified($request);
-
-		//Return response
-		return $response;
-	}
-
-	/**
-	 * Return thumb image
-	 *
-	 * @param Request $request The Request instance
-	 * @param string $hash The hash
-	 * @param int $updated The updated timestamp
-	 * @param string $path The image path
-	 * @param int $width The width
-	 * @param int $height The height
-	 * @return Response The rendered image
-	 */
-	public function thumb(Request $request, string $hash, int $updated, string $path, int $width, int $height): Response {
-		//Without matching hash
-		if ($hash !== $this->slugger->serialize([$updated, $path, $width, $height])) {
-			//Throw new exception
-			throw new NotFoundHttpException(sprintf('Unable to match thumb hash: %s', $hash));
-		}
-
-		//Set hashed tree
-		$hashed = array_reverse(str_split(strval($updated)));
-
-		//Set thumb
-		$thumb = $this->path.'/'.$this->prefix.'/'.$hashed[0].'/'.$hashed[1].'/'.$hashed[2].'/'.$updated.'/'.$path.'/'.$width.'x'.$height.'.jpeg';
-
-		//Unshort path
-		$path = $this->slugger->unshort($path);
-
-		//Without thumb up to date file
-		if (!is_file($thumb) || !($mtime = stat($thumb)['mtime']) || $mtime < $updated) {
-			//Without existing thumb path
-			if (!is_dir($dir = dirname($thumb))) {
-				//Create filesystem object
-				$filesystem = new Filesystem();
-
-				try {
-					//Create path
-					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-					//XXX: on CoW filesystems execute a chattr +C before filling
-					$filesystem->mkdir($dir, 0775);
-				} catch (IOExceptionInterface $e) {
-					//Throw error
-					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
-				}
-			}
-
-			//Create image instance
-			$image = new \Imagick();
-
-			//Read image
-			$image->readImage(realpath($path));
-
-			//Crop using aspect ratio
-			//XXX: for better result upload image directly in aspect ratio :)
-			$image->cropThumbnailImage($width, $height);
-
-			//Strip image exif data and properties
-			$image->stripImage();
-
-			//Set compression quality
-			//TODO: ajust that
-			$image->setImageCompressionQuality(70);
-
-			//Save thumb
-			if (!$image->writeImage($thumb)) {
-				//Throw error
-				throw new \Exception(sprintf('Unable to write image "%s"', $thumb));
-			}
-
-			//Set mtime
-			$mtime = stat($thumb)['mtime'];
-		}
-
-		//Read thumb from cache
-		$response = new BinaryFileResponse($thumb);
-
-		//Set file name
-		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'thumb-'.str_replace('/', '_', $path).'-'.$width.'x'.$height.'.jpeg');
-
-		//Set etag
-		$response->setEtag(md5($hash));
-
-		//Set last modified
-		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
-
-		//Set as public
-		$response->setPublic();
-
-		//Return 304 response if not modified
-		$response->isNotModified($request);
-
-		//Return response
-		return $response;
-	}
-}
diff --git a/Controller/MapController.php b/Controller/MapController.php
deleted file mode 100644
index 8d38aa6..0000000
--- a/Controller/MapController.php
+++ /dev/null
@@ -1,507 +0,0 @@
-<?php declare(strict_types=1);
-
-/*
- * This file is part of the Rapsys PackBundle package.
- *
- * (c) Raphaël Gertz <symfony@rapsys.eu>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
-
-namespace Rapsys\PackBundle\Controller;
-
-use Rapsys\PackBundle\Util\MapUtil;
-use Rapsys\PackBundle\Util\SluggerUtil;
-
-use Psr\Container\ContainerInterface;
-
-use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
-use Symfony\Component\Filesystem\Exception\IOExceptionInterface;
-use Symfony\Component\Filesystem\Filesystem;
-use Symfony\Component\HttpFoundation\BinaryFileResponse;
-use Symfony\Component\HttpFoundation\HeaderUtils;
-use Symfony\Component\HttpFoundation\Request;
-use Symfony\Component\HttpFoundation\Response;
-use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
-use Symfony\Component\Routing\RequestContext;
-use Symfony\Contracts\Service\ServiceSubscriberInterface;
-
-/**
- * {@inheritdoc}
- */
-class MapController extends AbstractController implements ServiceSubscriberInterface {
-	/**
-	 * The stream context instance
-	 */
-	protected mixed $ctx;
-
-	/**
-	 * Creates a new osm controller
-	 *
-	 * @param ContainerInterface $container The ContainerInterface instance
-	 * @param MapUtil $map The MapUtil instance
-	 * @param SluggerUtil $slugger The SluggerUtil instance
-	 * @param string $cache The cache path
-	 * @param string $path The public path
-	 * @param string $prefix The prefix
-	 * @param string $url The tile server url
-	 */
-	function __construct(protected ContainerInterface $container, protected MapUtil $map, protected SluggerUtil $slugger, protected string $cache = '../var/cache', protected string $path = './bundles/rapsyspack', protected string $prefix = 'map', protected string $url = MapUtil::osm) {
-		//Set ctx
-		$this->ctx = stream_context_create(
-			[
-				'http' => [
-					#'header' => ['Referer: https://www.openstreetmap.org/'],
-					'max_redirects' => $_ENV['RAPSYSPACK_REDIRECT'] ?? 20,
-					'timeout' => $_ENV['RAPSYSPACK_TIMEOUT'] ?? (($timeout = ini_get('default_socket_timeout')) !== false && $timeout !== "" ? (float)$timeout : 60),
-					'user_agent' => $_ENV['RAPSYSPACK_AGENT'] ?? (($agent = ini_get('user_agent')) !== false && $agent !== "" ? (string)$agent : RapsysPackBundle::getAlias().'/'.RapsysPackBundle::getVersion())
-				]
-			]
-		);
-	}
-
-	/**
-	 * Return map image
-	 *
-	 * @param Request $request The Request instance
-	 * @param string $hash The hash
-	 * @param int $updated The updated timestamp
-	 * @param float $latitude The latitude
-	 * @param float $longitude The longitude
-	 * @param int $zoom The zoom
-	 * @param int $width The width
-	 * @param int $height The height
-	 * @return Response The rendered image
-	 */
-	public function map(Request $request, string $hash, int $updated, float $latitude, float $longitude, int $zoom, int $width, int $height): Response {
-		//Without matching hash
-		if ($hash !== $this->slugger->hash([$updated, $latitude, $longitude, $zoom, $width, $height])) {
-			//Throw new exception
-			throw new NotFoundHttpException(sprintf('Unable to match map hash: %s', $hash));
-		}
-
-		//Set map
-		$map = $this->path.'/'.$this->prefix.'/'.$zoom.'/'.$latitude.'/'.$longitude.'/'.$width.'x'.$height.'.jpeg';
-
-		//Without multi up to date file
-		if (!is_file($map) || !($mtime = stat($map)['mtime']) || $mtime < $updated) {
-			//Without existing map path
-			if (!is_dir($dir = dirname($map))) {
-				//Create filesystem object
-				$filesystem = new Filesystem();
-
-				try {
-					//Create path
-					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-					//XXX: on CoW filesystems execute a chattr +C before filling
-					$filesystem->mkdir($dir, 0775);
-				} catch (IOExceptionInterface $e) {
-					//Throw error
-					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
-				}
-			}
-
-			//Create image instance
-			$image = new \Imagick();
-
-			//Add new image
-			$image->newImage($width, $height, new \ImagickPixel('transparent'), 'jpeg');
-
-			//Create tile instance
-			$tile = new \Imagick();
-
-			//Get tile xy
-			$centerX = $this->map->longitudeToX($longitude, $zoom);
-			$centerY = $this->map->latitudeToY($latitude, $zoom);
-
-			//Calculate start xy
-			$startX = floor(floor($centerX) - $width / MapUtil::tz);
-			$startY = floor(floor($centerY) - $height / MapUtil::tz);
-
-			//Calculate end xy
-			$endX = ceil(ceil($centerX) + $width / MapUtil::tz);
-			$endY = ceil(ceil($centerY) + $height / MapUtil::tz);
-
-			for($x = $startX; $x <= $endX; $x++) {
-				for($y = $startY; $y <= $endY; $y++) {
-					//Set cache path
-					$cache = $this->cache.'/'.$this->prefix.'/'.$zoom.'/'.$x.'/'.$y.'.png';
-
-					//Without cache image
-					if (!is_file($cache)) {
-						//Set tile url
-						$tileUri = str_replace(['{Z}', '{X}', '{Y}'], [$zoom, $x, $y], $this->url);
-
-						//Without cache path
-						if (!is_dir($dir = dirname($cache))) {
-							//Create filesystem object
-							$filesystem = new Filesystem();
-
-							try {
-								//Create path
-								//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-								$filesystem->mkdir($dir, 0775);
-							} catch (IOExceptionInterface $e) {
-								//Throw error
-								throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
-							}
-						}
-
-						//Store tile in cache
-						file_put_contents($cache, file_get_contents($tileUri, false, $this->ctx));
-					}
-
-					//Set dest x
-					$destX = intval(floor($width / 2 - MapUtil::tz * ($centerX - $x)));
-
-					//Set dest y
-					$destY = intval(floor($height / 2 - MapUtil::tz * ($centerY - $y)));
-
-					//Read tile from cache
-					$tile->readImage($cache);
-
-					//Compose image
-					$image->compositeImage($tile, \Imagick::COMPOSITE_OVER, $destX, $destY);
-
-					//Clear tile
-					$tile->clear();
-				}
-			}
-
-			//Add imagick draw instance
-			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
-			$draw = new \ImagickDraw();
-
-			//Set text antialias
-			$draw->setTextAntialias(true);
-
-			//Set stroke antialias
-			$draw->setStrokeAntialias(true);
-
-			//Set text alignment
-			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
-
-			//Set gravity
-			$draw->setGravity(\Imagick::GRAVITY_CENTER);
-
-			//Set fill color
-			$draw->setFillColor('#cff');
-
-			//Set stroke color
-			$draw->setStrokeColor('#00c3f9');
-
-			//Set stroke width
-			$draw->setStrokeWidth(2);
-
-			//Draw circle
-			$draw->circle($width/2 - 5, $height/2 - 5, $width/2 + 5, $height/2 + 5);
-
-			//Draw on image
-			$image->drawImage($draw);
-
-			//Strip image exif data and properties
-			$image->stripImage();
-
-			//Add latitude
-			//XXX: not supported by imagick :'(
-			$image->setImageProperty('exif:GPSLatitude', $this->map->latitudeToSexagesimal($latitude));
-
-			//Add longitude
-			//XXX: not supported by imagick :'(
-			$image->setImageProperty('exif:GPSLongitude', $this->map->longitudeToSexagesimal($longitude));
-
-			//Add description
-			//XXX: not supported by imagick :'(
-			#$image->setImageProperty('exif:Description', $caption);
-
-			//Set progressive jpeg
-			$image->setInterlaceScheme(\Imagick::INTERLACE_PLANE);
-
-			//Set compression quality
-			//TODO: ajust that
-			$image->setImageCompressionQuality(70);
-
-			//Save image
-			if (!$image->writeImage($map)) {
-				//Throw error
-				throw new \Exception(sprintf('Unable to write image "%s"', $path));
-			}
-
-			//Set mtime
-			$mtime = stat($map)['mtime'];
-		}
-
-		//Read map from cache
-		$response = new BinaryFileResponse($map);
-
-		//Set file name
-		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'map-'.$latitude.','.$longitude.'-'.$zoom.'-'.$width.'x'.$height.'.jpeg');
-
-		//Set etag
-		$response->setEtag(md5(serialize([$updated, $latitude, $longitude, $zoom, $width, $height])));
-
-		//Set last modified
-		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
-
-		//Disable robot index
-		$response->headers->set('X-Robots-Tag', 'noindex');
-
-		//Set as public
-		$response->setPublic();
-
-		//Return 304 response if not modified
-		$response->isNotModified($request);
-
-		//Return response
-		return $response;
-	}
-
-	/**
-	 * Return multi map image
-	 *
-	 * @param Request $request The Request instance
-	 * @param string $hash The hash
-	 * @param int $updated The updated timestamp
-	 * @param float $latitude The latitude
-	 * @param float $longitude The longitude
-	 * @param string $coordinates The coordinates
-	 * @param int $zoom The zoom
-	 * @param int $width The width
-	 * @param int $height The height
-	 * @return Response The rendered image
-	 */
-	public function multiMap(Request $request, string $hash, int $updated, float $latitude, float $longitude, string $coordinates, int $zoom, int $width, int $height): Response {
-		//Without matching hash
-		if ($hash !== $this->slugger->hash([$updated, $latitude, $longitude, $coordinate = $this->slugger->hash($coordinates), $zoom, $width, $height])) {
-			//Throw new exception
-			throw new NotFoundHttpException(sprintf('Unable to match multi map hash: %s', $hash));
-		}
-
-		//Set multi
-		$map = $this->path.'/'.$this->prefix.'/'.$zoom.'/'.$latitude.'/'.$longitude.'/'.$coordinate.'/'.$width.'x'.$height.'.jpeg';
-
-		//Without multi up to date file
-		if (!is_file($map) || !($mtime = stat($map)['mtime']) || $mtime < $updated) {
-			//Without existing multi path
-			if (!is_dir($dir = dirname($map))) {
-				//Create filesystem object
-				$filesystem = new Filesystem();
-
-				try {
-					//Create path
-					//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-					//XXX: on CoW filesystems execute a chattr +C before filling
-					$filesystem->mkdir($dir, 0775);
-				} catch (IOExceptionInterface $e) {
-					//Throw error
-					throw new \Exception(sprintf('Output path "%s" do not exists and unable to create it', $dir), 0, $e);
-				}
-			}
-
-			//Create image instance
-			$image = new \Imagick();
-
-			//Add new image
-			$image->newImage($width, $height, new \ImagickPixel('transparent'), 'jpeg');
-
-			//Create tile instance
-			$tile = new \Imagick();
-
-			//Get tile xy
-			$centerX = $this->map->longitudeToX($longitude, $zoom);
-			$centerY = $this->map->latitudeToY($latitude, $zoom);
-
-			//Calculate start xy
-			$startX = floor(floor($centerX) - $width / MapUtil::tz);
-			$startY = floor(floor($centerY) - $height / MapUtil::tz);
-
-			//Calculate end xy
-			$endX = ceil(ceil($centerX) + $width / MapUtil::tz);
-			$endY = ceil(ceil($centerY) + $height / MapUtil::tz);
-
-			for($x = $startX; $x <= $endX; $x++) {
-				for($y = $startY; $y <= $endY; $y++) {
-					//Set cache path
-					$cache = $this->cache.'/'.$this->prefix.'/'.$zoom.'/'.$x.'/'.$y.'.png';
-
-					//Without cache image
-					if (!is_file($cache)) {
-						//Set tile url
-						$tileUri = str_replace(['{Z}', '{X}', '{Y}'], [$zoom, $x, $y], $this->url);
-
-						//Without cache path
-						if (!is_dir($dir = dirname($cache))) {
-							//Create filesystem object
-							$filesystem = new Filesystem();
-
-							try {
-								//Create path
-								//XXX: set as 0775, symfony umask (0022) will reduce rights (0755)
-								$filesystem->mkdir($dir, 0775);
-							} catch (IOExceptionInterface $e) {
-								//Throw error
-								throw new \Exception(sprintf('Output directory "%s" do not exists and unable to create it', $dir), 0, $e);
-							}
-						}
-
-						//Store tile in cache
-						file_put_contents($cache, file_get_contents($tileUri, false, $this->ctx));
-					}
-
-					//Set dest x
-					$destX = intval(floor($width / 2 - MapUtil::tz * ($centerX - $x)));
-
-					//Set dest y
-					$destY = intval(floor($height / 2 - MapUtil::tz * ($centerY - $y)));
-
-					//Read tile from cache
-					$tile->readImage($cache);
-
-					//Compose image
-					$image->compositeImage($tile, \Imagick::COMPOSITE_OVER, $destX, $destY);
-
-					//Clear tile
-					$tile->clear();
-				}
-			}
-
-			//Add imagick draw instance
-			//XXX: see https://www.php.net/manual/fr/imagick.examples-1.php#example-3916
-			$draw = new \ImagickDraw();
-
-			//Set text antialias
-			$draw->setTextAntialias(true);
-
-			//Set stroke antialias
-			$draw->setStrokeAntialias(true);
-
-			//Set text alignment
-			$draw->setTextAlignment(\Imagick::ALIGN_CENTER);
-
-			//Set gravity
-			$draw->setGravity(\Imagick::GRAVITY_CENTER);
-
-			//Convert to array
-			$coordinates = array_reverse(array_map(function ($v) { $p = strpos($v, ','); return ['latitude' => floatval(substr($v, 0, $p)), 'longitude' => floatval(substr($v, $p + 1))]; }, explode('-', $coordinates)), true);
-
-			//Iterate on locations
-			foreach($coordinates as $id => $coordinate) {
-				//Set dest x
-				$destX = intval(floor($width / 2 - MapUtil::tz * ($centerX - $this->map->longitudeToX(floatval($coordinate['longitude']), $zoom))));
-
-				//Set dest y
-				$destY = intval(floor($height / 2 - MapUtil::tz * ($centerY - $this->map->latitudeToY(floatval($coordinate['latitude']), $zoom))));
-
-				//Set fill color
-				$draw->setFillColor($this->map->getFill());
-
-				//Set font size
-				$draw->setFontSize($this->map->getFontSize());
-
-				//Set stroke color
-				$draw->setStrokeColor($this->map->getStroke());
-
-				//Set circle radius
-				$radius = $this->map->getRadius();
-
-				//Set stroke width
-				$stroke = $this->map->getStrokeWidth();
-
-				//With matching position
-				if ($coordinate['latitude'] === $latitude && $coordinate['longitude'] == $longitude) {
-					//Set fill color
-					$draw->setFillColor($this->map->getHighFill());
-
-					//Set font size
-					$draw->setFontSize($this->map->getHighFontSize());
-
-					//Set stroke color
-					$draw->setStrokeColor($this->map->getHighStroke());
-
-					//Set circle radius
-					$radius = $this->map->getHighRadius();
-
-					//Set stroke width
-					$stroke = $this->map->getHighStrokeWidth();
-				}
-
-				//Set stroke width
-				$draw->setStrokeWidth($stroke);
-
-				//Draw circle
-				$draw->circle($destX - $radius, $destY - $radius, $destX + $radius, $destY + $radius);
-
-				//Set fill color
-				$draw->setFillColor($draw->getStrokeColor());
-
-				//Set stroke width
-				$draw->setStrokeWidth($stroke / 4);
-
-				//Get font metrics
-				#$metrics = $image->queryFontMetrics($draw, strval($id));
-
-				//Add annotation
-				$draw->annotation($destX - $radius, $destY + $stroke, strval($id));
-			}
-
-			//Draw on image
-			$image->drawImage($draw);
-
-			//Strip image exif data and properties
-			$image->stripImage();
-
-			//Add latitude
-			//XXX: not supported by imagick :'(
-			$image->setImageProperty('exif:GPSLatitude', $this->map->latitudeToSexagesimal($latitude));
-
-			//Add longitude
-			//XXX: not supported by imagick :'(
-			$image->setImageProperty('exif:GPSLongitude', $this->map->longitudeToSexagesimal($longitude));
-
-			//Add description
-			//XXX: not supported by imagick :'(
-			#$image->setImageProperty('exif:Description', $caption);
-
-			//Set progressive jpeg
-			$image->setInterlaceScheme(\Imagick::INTERLACE_PLANE);
-
-			//Set compression quality
-			//TODO: ajust that
-			$image->setImageCompressionQuality(70);
-
-			//Save image
-			if (!$image->writeImage($map)) {
-				//Throw error
-				throw new \Exception(sprintf('Unable to write image "%s"', $path));
-			}
-
-			//Set mtime
-			$mtime = stat($map)['mtime'];
-		}
-
-		//Read map from cache
-		$response = new BinaryFileResponse($map);
-
-		//Set file name
-		$response->setContentDisposition(HeaderUtils::DISPOSITION_INLINE, 'multimap-'.$latitude.','.$longitude.'-'.$zoom.'-'.$width.'x'.$height.'.jpeg');
-
-		//Set etag
-		$response->setEtag(md5(serialize([$updated, $latitude, $longitude, $zoom, $width, $height])));
-
-		//Set last modified
-		$response->setLastModified(\DateTime::createFromFormat('U', strval($mtime)));
-
-		//Disable robot index
-		$response->headers->set('X-Robots-Tag', 'noindex');
-
-		//Set as public
-		$response->setPublic();
-
-		//Return 304 response if not modified
-		$response->isNotModified($request);
-
-		//Return response
-		return $response;
-	}
-}
-- 
2.41.3