Core/src/Controllers/AbstractController.php

136 lines
3.8 KiB
PHP
Raw Normal View History

<?php
namespace Benzine\Controllers;
2020-06-19 07:53:59 +00:00
use Benzine\Controllers\Filters\Filter;
2020-06-22 19:38:55 +00:00
use Benzine\Exceptions\FilterDecodeException;
use League\Flysystem\Filesystem;
2020-11-13 08:19:24 +00:00
use League\MimeTypeDetection\ExtensionMimeTypeDetector;
use League\MimeTypeDetection\FinfoMimeTypeDetector;
2020-07-27 02:44:36 +00:00
use Monolog\Logger;
2020-11-13 08:50:21 +00:00
use Slim\HttpCache\CacheProvider;
2020-06-19 07:53:59 +00:00
use Slim\Psr7\Request;
use Slim\Psr7\Response;
2020-09-01 03:15:02 +00:00
abstract class AbstractController
{
2021-05-06 20:15:01 +00:00
public function __construct(
protected Logger $logger,
protected CacheProvider $cacheProvider
) {
}
public function xmlResponse(\SimpleXMLElement $root, Request $request, Response $response): Response
{
2020-06-22 19:38:55 +00:00
$response->getBody()->write($root->asXML());
return $response->withHeader('Content-type', 'text/xml');
}
public function jsonResponse($json, Request $request, Response $response): Response
{
2020-06-22 19:38:55 +00:00
$content = json_encode($json, JSON_PRETTY_PRINT);
$response->getBody()->write($content);
return $response->withHeader('Content-type', 'application/json');
}
2020-11-20 14:01:14 +00:00
public function redirect(Response $response, string $url = '/', int $code = 302): Response
2020-11-18 08:08:06 +00:00
{
$response = $response->withStatus($code);
2020-11-20 14:01:14 +00:00
return $response->withHeader('Location', $url);
2020-11-18 08:08:06 +00:00
}
public function jsonResponseException(\Exception $e, Request $request, Response $response): Response
{
return $this->jsonResponse(
[
'Status' => 'Fail',
'Reason' => $e->getMessage(),
],
$request,
$response
);
}
/**
* Decide if a request has a filter attached to it.
*
* @throws FilterDecodeException
*/
protected function requestHasFilters(Request $request, Response $response): bool
{
if ($request->hasHeader('Filter')) {
$filterText = trim($request->getHeader('Filter')[0]);
if (!empty($filterText)) {
$decode = json_decode($filterText);
if (null !== $decode) {
return true;
}
throw new FilterDecodeException('Could not decode given Filter. Reason: Not JSON. Given: "'.$filterText.'"');
}
}
return false;
}
/**
* Parse filters header into filter objects.
*/
protected function parseFilters(Request $request, Response $response): Filter
{
$filter = new Filter();
$filter->parseFromHeader(json_decode($request->getHeader('Filter')[0], true));
return $filter;
}
2020-08-27 17:24:10 +00:00
2020-08-27 17:25:48 +00:00
protected function pageNotFound(): Response
{
2020-08-27 17:24:10 +00:00
return (new Response())
2020-08-27 17:25:48 +00:00
->withStatus(404)
;
2020-08-27 17:24:10 +00:00
}
protected function returnFile(Filesystem $filesystem, string $filename): Response
{
$response = new Response();
2020-11-13 07:49:36 +00:00
2021-07-24 06:16:20 +00:00
if (!$filesystem->fileExists($filename)) {
return $this->pageNotFound();
}
2021-07-24 06:16:20 +00:00
// Generate an etag
2022-06-19 01:19:07 +00:00
$etag = md5($filesystem->lastModified($filename).$filename);
2021-07-24 06:16:20 +00:00
$response = $this->cacheProvider->withEtag($response, $etag);
2020-11-13 08:50:21 +00:00
// Detect mimetype for content-type header from file meta
2020-11-13 08:19:24 +00:00
$mimetype = (new ExtensionMimeTypeDetector())
2022-06-19 01:19:07 +00:00
->detectMimeTypeFromPath($filename)
;
// No dice? Early-load the data and interrogate that for mimetype then I GUESS.
if (!$mimetype) {
$data = $filesystem->read($filename);
$mimetype = (new FinfoMimeTypeDetector())
2022-06-19 01:19:07 +00:00
->detectMimeTypeFromBuffer($data)
;
}
// If we have mimetype by this point, send the contenttype
if ($mimetype) {
$response = $response->withHeader('Content-Type', $mimetype);
}
2021-07-24 06:16:20 +00:00
// Send back the response
$response
->getBody()
2022-06-19 01:19:07 +00:00
->write($data ?? $filesystem->read($filename))
;
2020-11-13 07:49:36 +00:00
return $response;
}
}