123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675 |
- <?php
- declare(strict_types=1);
- namespace Grav\Plugin\FlexObjects\Controllers;
- use Exception;
- use Grav\Common\Debugger;
- use Grav\Common\Page\Interfaces\PageInterface;
- use Grav\Common\Page\Medium\Medium;
- use Grav\Common\Page\Medium\MediumFactory;
- use Grav\Common\Utils;
- use Grav\Framework\Flex\FlexObject;
- use Grav\Framework\Flex\Interfaces\FlexAuthorizeInterface;
- use Grav\Framework\Flex\Interfaces\FlexObjectInterface;
- use Grav\Framework\Media\Interfaces\MediaInterface;
- use LogicException;
- use Psr\Http\Message\ResponseInterface;
- use Psr\Http\Message\UploadedFileInterface;
- use RocketTheme\Toolbox\Event\Event;
- use RuntimeException;
- use function is_array;
- use function is_string;
- /**
- * Class MediaController
- * @package Grav\Plugin\FlexObjects\Controllers
- */
- class MediaController extends AbstractController
- {
- /**
- * @return ResponseInterface
- */
- public function taskMediaUpload(): ResponseInterface
- {
- $this->checkAuthorization('media.create');
- $object = $this->getObject();
- if (null === $object) {
- throw new RuntimeException('Not Found', 404);
- }
- if (!method_exists($object, 'checkUploadedMediaFile')) {
- throw new RuntimeException('Not Found', 404);
- }
- // Get updated object from Form Flash.
- $flash = $this->getFormFlash($object);
- if ($flash->exists()) {
- $object = $flash->getObject() ?? $object;
- $object->update([], $flash->getFilesByFields());
- }
- // Get field for the uploaded media.
- $field = $this->getPost('name', 'undefined');
- if ($field === 'undefined') {
- $field = null;
- }
- $request = $this->getRequest();
- $files = $request->getUploadedFiles();
- if ($field && isset($files['data'])) {
- $files = $files['data'];
- $parts = explode('.', $field);
- $last = array_pop($parts);
- foreach ($parts as $name) {
- if (!is_array($files[$name])) {
- throw new RuntimeException($this->translate('PLUGIN_ADMIN.INVALID_PARAMETERS'), 400);
- }
- $files = $files[$name];
- }
- $file = $files[$last] ?? null;
- } else {
- // Legacy call with name being the filename instead of field name.
- $file = $files['file'] ?? null;
- $field = null;
- }
- /** @var UploadedFileInterface $file */
- if (is_array($file)) {
- $file = reset($file);
- }
- if (!$file instanceof UploadedFileInterface) {
- throw new RuntimeException($this->translate('PLUGIN_ADMIN.INVALID_PARAMETERS'), 400);
- }
- $filename = $file->getClientFilename();
- $object->checkUploadedMediaFile($file, $filename, $field);
- try {
- // TODO: This only merges main level data, but is good for ordering (for now).
- $data = $flash->getData() ?? [];
- $data = array_replace($data, (array)$this->getPost('data'));
- $crop = $this->getPost('crop');
- if (is_string($crop)) {
- $crop = json_decode($crop, true, 512, JSON_THROW_ON_ERROR);
- }
- $flash->setData($data);
- $flash->addUploadedFile($file, $field, $crop);
- $flash->save();
- } catch (Exception $e) {
- throw new RuntimeException($e->getMessage(), $e->getCode(), $e);
- }
- // Include exif metadata into the response if configured to do so
- $metadata = [];
- $include_metadata = $this->grav['config']->get('system.media.auto_metadata_exif', false);
- if ($include_metadata) {
- $medium = MediumFactory::fromUploadedFile($file);
- $media = $object->getMedia();
- $media->add($filename, $medium);
- $basename = str_replace(['@3x', '@2x'], '', Utils::pathinfo($filename, PATHINFO_BASENAME));
- if (isset($media[$basename])) {
- $metadata = $media[$basename]->metadata() ?: [];
- }
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FILE_UPLOADED_SUCCESSFULLY'),
- 'filename' => htmlspecialchars($filename, ENT_QUOTES | ENT_HTML5, 'UTF-8'),
- 'metadata' => $metadata
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * @return ResponseInterface
- */
- public function taskMediaUploadMeta(): ResponseInterface
- {
- try {
- $this->checkAuthorization('media.create');
- $object = $this->getObject();
- if (null === $object) {
- throw new RuntimeException('Not Found', 404);
- }
- if (!method_exists($object, 'getMediaField')) {
- throw new RuntimeException('Not Found', 404);
- }
- $object->refresh();
- // Get updated object from Form Flash.
- $flash = $this->getFormFlash($object);
- if ($flash->exists()) {
- $object = $flash->getObject() ?? $object;
- $object->update([], $flash->getFilesByFields());
- }
- // Get field and data for the uploaded media.
- $field = (string)$this->getPost('field');
- $media = $object->getMediaField($field);
- if (!$media) {
- throw new RuntimeException('Media field not found: ' . $field, 404);
- }
- $data = $this->getPost('data');
- if (is_string($data)) {
- $data = json_decode($data, true);
- }
- $filename = Utils::basename($data['name'] ?? '');
- // Update field.
- $files = $object->getNestedProperty($field, []);
- // FIXME: Do we want to save something into the field as well?
- $files[$filename] = [];
- $object->setNestedProperty($field, $files);
- $info = [
- 'modified' => $data['modified'] ?? null,
- 'size' => $data['size'] ?? null,
- 'mime' => $data['mime'] ?? null,
- 'width' => $data['width'] ?? null,
- 'height' => $data['height'] ?? null,
- 'duration' => $data['duration'] ?? null,
- 'orientation' => $data['orientation'] ?? null,
- 'meta' => array_filter($data, static function ($val) { return $val !== null; })
- ];
- $info = array_filter($info, static function ($val) { return $val !== null; });
- // As the file may not be saved locally, we need to update the index.
- $media->updateIndex([$filename => $info]);
- $object->save();
- $flash->save();
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FILE_UPLOADED_SUCCESSFULLY'),
- 'field' => $field,
- 'filename' => $filename,
- 'metadata' => $data
- ];
- } catch (\Exception $e) {
- /** @var Debugger $debugger */
- $debugger = $this->grav['debugger'];
- $debugger->addException($e);
- return $this->createJsonErrorResponse($e);
- }
- return $this->createJsonResponse($response);
- }
- /**
- * @return ResponseInterface
- */
- public function taskMediaReorder(): ResponseInterface
- {
- try {
- $this->checkAuthorization('media.update');
- $object = $this->getObject();
- if (null === $object) {
- throw new RuntimeException('Not Found', 404);
- }
- if (!method_exists($object, 'getMediaField')) {
- throw new RuntimeException('Not Found', 404);
- }
- $object->refresh();
- // Get updated object from Form Flash.
- $flash = $this->getFormFlash($object);
- if ($flash->exists()) {
- $object = $flash->getObject() ?? $object;
- $object->update([], $flash->getFilesByFields());
- }
- // Get field and data for the uploaded media.
- $field = (string)$this->getPost('field');
- $media = $object->getMediaField($field);
- if (!$media) {
- throw new RuntimeException('Media field not found: ' . $field, 404);
- }
- // Create id => filename map from all files in the media.
- $map = [];
- foreach ($media as $name => $medium) {
- $id = $medium->get('meta.id');
- if ($id) {
- $map[$id] = $name;
- }
- }
- // Get reorder list and reorder the map.
- $data = $this->getPost('data');
- if (is_string($data)) {
- $data = json_decode($data, true);
- }
- $data = array_fill_keys($data, null);
- $map = array_filter(array_merge($data, $map), static function($val) { return $val !== null; });
- // Reorder the files.
- $files = $object->getNestedProperty($field, []);
- $map = array_fill_keys($map, null);
- $files = array_filter(array_merge($map, $files), static function($val) { return $val !== null; });
- // Update field.
- $object->setNestedProperty($field, $files);
- $object->save();
- $flash->save();
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FIELD_REORDER_SUCCESSFUL'),
- 'field' => $field,
- 'ordering' => array_keys($files)
- ];
- } catch (\Exception $e) {
- /** @var Debugger $debugger */
- $debugger = $this->grav['debugger'];
- $debugger->addException($e);
- $ex = new RuntimeException($this->translate('PLUGIN_ADMIN.FIELD_REORDER_FAILED', $field), $e->getCode(), $e);
- return $this->createJsonErrorResponse($ex);
- }
- return $this->createJsonResponse($response);
- }
- /**
- * @return ResponseInterface
- */
- public function taskMediaDelete(): ResponseInterface
- {
- $this->checkAuthorization('media.delete');
- /** @var FlexObjectInterface|null $object */
- $object = $this->getObject();
- if (!$object) {
- throw new RuntimeException('Not Found', 404);
- }
- $filename = $this->getPost('filename');
- // Handle bad filenames.
- if (!Utils::checkFilename($filename)) {
- throw new RuntimeException($this->translate('PLUGIN_ADMIN.NO_FILE_FOUND'), 400);
- }
- try {
- $field = $this->getPost('name');
- $flash = $this->getFormFlash($object);
- $flash->removeFile($filename, $field);
- $flash->save();
- } catch (Exception $e) {
- throw new RuntimeException($e->getMessage(), $e->getCode(), $e);
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FILE_DELETED') . ': ' . htmlspecialchars($filename, ENT_QUOTES | ENT_HTML5, 'UTF-8')
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * Used in pagemedia field.
- *
- * @return ResponseInterface
- */
- public function taskMediaCopy(): ResponseInterface
- {
- $this->checkAuthorization('media.create');
- /** @var FlexObjectInterface|null $object */
- $object = $this->getObject();
- if (!$object) {
- throw new RuntimeException('Not Found', 404);
- }
- if (!method_exists($object, 'uploadMediaFile')) {
- throw new RuntimeException('Not Found', 404);
- }
- $request = $this->getRequest();
- $files = $request->getUploadedFiles();
- $file = $files['file'] ?? null;
- if (!$file instanceof UploadedFileInterface) {
- throw new RuntimeException($this->translate('PLUGIN_ADMIN.INVALID_PARAMETERS'), 400);
- }
- $post = $request->getParsedBody();
- $filename = $post['name'] ?? $file->getClientFilename();
- // Upload media right away.
- $object->uploadMediaFile($file, $filename);
- // Include exif metadata into the response if configured to do so
- $metadata = [];
- $include_metadata = $this->grav['config']->get('system.media.auto_metadata_exif', false);
- if ($include_metadata) {
- $basename = str_replace(['@3x', '@2x'], '', Utils::pathinfo($filename, PATHINFO_BASENAME));
- $media = $object->getMedia();
- if (isset($media[$basename])) {
- $metadata = $media[$basename]->metadata() ?: [];
- }
- }
- if ($object instanceof PageInterface) {
- // Backwards compatibility to existing plugins.
- // DEPRECATED: page
- $this->grav->fireEvent('onAdminAfterAddMedia', new Event(['object' => $object, 'page' => $object]));
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FILE_UPLOADED_SUCCESSFULLY'),
- 'filename' => htmlspecialchars($filename, ENT_QUOTES | ENT_HTML5, 'UTF-8'),
- 'metadata' => $metadata
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * Used in pagemedia field.
- *
- * @return ResponseInterface
- */
- public function taskMediaRemove(): ResponseInterface
- {
- $this->checkAuthorization('media.delete');
- /** @var FlexObjectInterface|null $object */
- $object = $this->getObject();
- if (!$object) {
- throw new RuntimeException('Not Found', 404);
- }
- if (!method_exists($object, 'deleteMediaFile')) {
- throw new RuntimeException('Not Found', 404);
- }
- $field = $this->getPost('field');
- $filename = $this->getPost('filename');
- // Handle bad filenames.
- if (!Utils::checkFilename($filename)) {
- throw new RuntimeException($this->translate('PLUGIN_ADMIN.NO_FILE_FOUND'), 400);
- }
- $object->deleteMediaFile($filename, $field);
- if ($field) {
- $order = $object->getNestedProperty($field);
- unset($order[$filename]);
- $object->setNestedProperty($field, $order);
- $object->save();
- }
- if ($object instanceof PageInterface) {
- // Backwards compatibility to existing plugins.
- // DEPRECATED: page
- $this->grav->fireEvent('onAdminAfterDelMedia', new Event(['object' => $object, 'page' => $object, 'media' => $object->getMedia(), 'filename' => $filename]));
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'message' => $this->translate('PLUGIN_ADMIN.FILE_DELETED') . ': ' . htmlspecialchars($filename, ENT_QUOTES | ENT_HTML5, 'UTF-8')
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * @return ResponseInterface
- */
- public function actionMediaList(): ResponseInterface
- {
- $this->checkAuthorization('media.list');
- /** @var MediaInterface|FlexObjectInterface $object */
- $object = $this->getObject();
- if (!$object) {
- throw new RuntimeException('Not Found', 404);
- }
- // Get updated object from Form Flash.
- $flash = $this->getFormFlash($object);
- if ($flash->exists()) {
- $object = $flash->getObject() ?? $object;
- $object->update([], $flash->getFilesByFields());
- }
- $media = $object->getMedia();
- $media_list = [];
- /**
- * @var string $name
- * @var Medium $medium
- */
- foreach ($media->all() as $name => $medium) {
- $media_list[$name] = [
- 'url' => $medium->display($medium->get('extension') === 'svg' ? 'source' : 'thumbnail')->cropZoom(400, 300)->url(),
- 'size' => $medium->get('size'),
- 'metadata' => $medium->metadata() ?: [],
- 'original' => $medium->higherQualityAlternative()->get('filename')
- ];
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'results' => $media_list
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * Used by the filepicker field to get a list of files in a folder.
- *
- * @return ResponseInterface
- */
- protected function actionMediaPicker(): ResponseInterface
- {
- $this->checkAuthorization('media.list');
- /** @var FlexObject $object */
- $object = $this->getObject();
- if (!$object || !\is_callable([$object, 'getFieldSettings'])) {
- throw new RuntimeException('Not Found', 404);
- }
- // Get updated object from Form Flash.
- $flash = $this->getFormFlash($object);
- if ($flash->exists()) {
- $object = $flash->getObject() ?? $object;
- $object->update([], $flash->getFilesByFields());
- }
- $name = $this->getPost('name');
- $settings = $name ? $object->getFieldSettings($name) : null;
- if (empty($settings['media_picker_field'])) {
- throw new RuntimeException('Not Found', 404);
- }
- $media = $object->getMediaField($name);
- $available_files = [];
- $metadata = [];
- $thumbs = [];
- /**
- * @var string $name
- * @var Medium $medium
- */
- foreach ($media->all() as $name => $medium) {
- $available_files[] = $name;
- if (isset($settings['include_metadata'])) {
- $img_metadata = $medium->metadata();
- if ($img_metadata) {
- $metadata[$name] = $img_metadata;
- }
- }
- }
- // Peak in the flashObject for optimistic filepicker updates
- $pending_files = [];
- $sessionField = base64_encode($this->grav['uri']->url());
- $flash = $this->getSession()->getFlashObject('files-upload');
- $folder = $media->getPath() ?: null;
- if ($flash && isset($flash[$sessionField])) {
- foreach ($flash[$sessionField] as $field => $data) {
- foreach ($data as $file) {
- $test = \dirname($file['path']);
- if ($test === $folder) {
- $pending_files[] = $file['name'];
- }
- }
- }
- }
- $this->getSession()->setFlashObject('files-upload', $flash);
- // Handle Accepted file types
- // Accept can only be file extensions (.pdf|.jpg)
- if (isset($settings['accept'])) {
- $available_files = array_filter($available_files, function ($file) use ($settings) {
- return $this->filterAcceptedFiles($file, $settings);
- });
- $pending_files = array_filter($pending_files, function ($file) use ($settings) {
- return $this->filterAcceptedFiles($file, $settings);
- });
- }
- if (isset($settings['deny'])) {
- $available_files = array_filter($available_files, function ($file) use ($settings) {
- return $this->filterDeniedFiles($file, $settings);
- });
- $pending_files = array_filter($pending_files, function ($file) use ($settings) {
- return $this->filterDeniedFiles($file, $settings);
- });
- }
- // Generate thumbs if needed
- if (isset($settings['preview_images']) && $settings['preview_images'] === true) {
- foreach ($available_files as $filename) {
- $thumbs[$filename] = $media[$filename]->zoomCrop(100,100)->url();
- }
- }
- $response = [
- 'code' => 200,
- 'status' => 'success',
- 'files' => array_values($available_files),
- 'pending' => array_values($pending_files),
- 'folder' => $folder,
- 'metadata' => $metadata,
- 'thumbs' => $thumbs
- ];
- return $this->createJsonResponse($response);
- }
- /**
- * @param string $file
- * @param array $settings
- * @return false|int
- */
- protected function filterAcceptedFiles(string $file, array $settings)
- {
- $valid = false;
- foreach ((array)$settings['accept'] as $type) {
- $find = str_replace('*', '.*', $type);
- $valid |= preg_match('#' . $find . '$#i', $file);
- }
- return $valid;
- }
- /**
- * @param string $file
- * @param array $settings
- * @return false|int
- */
- protected function filterDeniedFiles(string $file, array $settings)
- {
- $valid = true;
- foreach ((array)$settings['deny'] as $type) {
- $find = str_replace('*', '.*', $type);
- $valid = !preg_match('#' . $find . '$#i', $file);
- }
- return $valid;
- }
- /**
- * @param string $action
- * @return void
- * @throws LogicException
- * @throws RuntimeException
- */
- protected function checkAuthorization(string $action): void
- {
- $object = $this->getObject();
- if (!$object) {
- throw new RuntimeException('Not Found', 404);
- }
- // If object does not have ACL support ignore ACL checks.
- if (!$object instanceof FlexAuthorizeInterface) {
- return;
- }
- switch ($action) {
- case 'media.list':
- $action = 'read';
- break;
- case 'media.create':
- case 'media.update':
- case 'media.delete':
- $action = $object->exists() ? 'update' : 'create';
- break;
- default:
- throw new LogicException(sprintf('Unsupported authorize action %s', $action), 500);
- }
- if (!$object->isAuthorized($action, null, $this->user)) {
- throw new RuntimeException('Forbidden', 403);
- }
- }
- }
|