. // }}} namespace App\Controller; use App\Core\Controller; use App\Core\DB\DB; use App\Core\Event; use App\Core\GSFile; use function App\Core\I18n\_m; use App\Core\Router\Router; use App\Entity\AttachmentThumbnail; use App\Util\Common; use App\Util\Exception\ClientException; use App\Util\Exception\NotFoundException; use App\Util\Exception\ServerException; use Symfony\Component\HttpFoundation\HeaderUtils; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\Mime\MimeTypes; class Attachment extends Controller { /** * Generic function that handles getting a representation for an attachment */ private function attachment(int $id, callable $handle) { if ($id <= 0) { // This should never happen coming from the router, but let's bail if it does // @codeCoverageIgnoreStart Log::critical("Attachment controller called with {$id}, which should not be possible"); throw new ClientException(_m('No such attachment'), 404); // @codeCoverageIgnoreEnd } else { if (Event::handle('AttachmentFileInfo', [$id, &$res]) != Event::stop) { // If no one else claims this attachment, use the default representation $res = GSFile::getAttachmentFileInfo($id); } } if (!empty($res)) { return $handle($res); } else { // @codeCoverageIgnoreStart throw new ClientException(_m('No such attachment'), 404); // @codeCoverageIgnoreEnd } } /** * The page where the attachment and it's info is shown */ public function attachment_show(Request $request, int $id) { try { $attachment = DB::findOneBy('attachment', ['id' => $id]); return $this->attachment($id, function ($res) use ($id, $attachment) { return [ '_template' => 'attachments/show.html.twig', 'download' => Router::url('attachment_download', ['id' => $id]), 'attachment' => $attachment, 'right_panel_vars' => ['attachment_id' => $id], ]; }); } catch (NotFoundException) { throw new ClientException(_m('No such attachment'), 404); } } /** * Display the attachment inline */ public function attachment_view(Request $request, int $id) { return $this->attachment($id, fn (array $res) => GSFile::sendFile($res['filepath'], $res['mimetype'], GSFile::ensureFilenameWithProperExtension($res['filename'], $res['mimetype']) ?? $res['filename'], HeaderUtils::DISPOSITION_INLINE)); } public function attachment_download(Request $request, int $id) { return $this->attachment($id, fn (array $res) => GSFile::sendFile($res['filepath'], $res['mimetype'], GSFile::ensureFilenameWithProperExtension($res['filename'], $res['mimetype']) ?? $res['filename'], HeaderUtils::DISPOSITION_ATTACHMENT)); } /** * Controller to produce a thumbnail for a given attachment id * * @param Request $request * @param int $id Attachment ID * * @throws ClientException * @throws NotFoundException * @throws ServerException * @throws \App\Util\Exception\DuplicateFoundException * * @return Response */ public function attachment_thumbnail(Request $request, int $id): Response { $attachment = DB::findOneBy('attachment', ['id' => $id]); $default_width = Common::config('thumbnail', 'width'); $default_height = Common::config('thumbnail', 'height'); $default_crop = Common::config('thumbnail', 'smart_crop'); $width = $this->int('w') ?: $default_width; $height = $this->int('h') ?: $default_height; $crop = $this->bool('c') ?: $default_crop; Event::handle('GetAllowedThumbnailSizes', [&$sizes]); if (!in_array(['width' => $width, 'height' => $height], $sizes)) { throw new ClientException(_m('The requested thumbnail dimensions are not allowed'), 400); // 400 Bad Request } $thumbnail = AttachmentThumbnail::getOrCreate(attachment: $attachment, width: $width, height: $height, crop: $crop); $filename = $thumbnail->getFilename(); $path = $thumbnail->getPath(); $mimetype = $thumbnail->getMimetype(); return GSFile::sendFile(filepath: $path, mimetype: $mimetype, output_filename: $filename . '.' . MimeTypes::getDefault()->getExtensions($mimetype)[0], disposition: HeaderUtils::DISPOSITION_INLINE); } }