From 66e84458f8193f8a60fbb8355b9e6d4618f0445e Mon Sep 17 00:00:00 2001 From: newtonjob Date: Wed, 14 Aug 2024 03:00:45 +0100 Subject: [PATCH 1/2] Properly handle cloudinary public IDs and extensions for media and raw resources --- src/CloudinaryAdapter.php | 64 +++++++++++++++++++--------- tests/Unit/CloudinaryAdapterTest.php | 13 ++++++ 2 files changed, 58 insertions(+), 19 deletions(-) diff --git a/src/CloudinaryAdapter.php b/src/CloudinaryAdapter.php index 600de57..5cb02b6 100644 --- a/src/CloudinaryAdapter.php +++ b/src/CloudinaryAdapter.php @@ -22,10 +22,18 @@ */ class CloudinaryAdapter implements FilesystemAdapter { - - /** Cloudinary\Cloudinary */ + /** + * @var Cloudinary + */ protected Cloudinary $cloudinary; + /** + * The media resource extensions supported by cloudinary. + */ + public $mediaExtensions = [ + 'jpg', 'jpeg', 'png', 'gif', 'pdf', 'bmp', 'tiff', 'svg', 'ico', 'eps', 'psd', 'webp', 'jxr', 'wdp', + 'mpeg', 'mp4', 'mkv', 'mov', 'flv', 'avi', '3gp', '3g2', 'wmv', 'webm', 'ogv', 'mxf', + ]; /** * Constructor @@ -70,28 +78,42 @@ public function write(string $path, string $contents, Config $config): void */ public function writeStream(string $path, $contents, Config $config): void { - $publicId = $config->get('public_id', $path); + $publicId = $this->preparePublicId($config->get('public_id', $path)); $resourceType = $config->get('resource_type', 'auto'); - $fileExtension = pathinfo($publicId, PATHINFO_EXTENSION); - - $newPublicId = $fileExtension ? substr($publicId, 0, - (strlen($fileExtension) + 1)) : $publicId; - $uploadOptions = [ - 'public_id' => $newPublicId, + 'public_id' => $publicId, 'resource_type' => $resourceType ]; $resourceMetadata = stream_get_meta_data($contents); - resolve(CloudinaryEngine::class)->upload($resourceMetadata['uri'], $uploadOptions); + $this->upload($resourceMetadata['uri'], $uploadOptions); + } + + /** + * Prepare the given public ID for cloudinary. + */ + public function preparePublicId($path): string + { + $extension = pathinfo($path, PATHINFO_EXTENSION); + + return str($path) + ->when($this->isMedia($extension)) + ->beforeLast('.'.$extension); } + /** + * Determine if the given extension is a media extension. + */ + public function isMedia($extension): bool + { + return in_array($extension, $this->mediaExtensions); + } /** * Rename a file. - * Paths without extensions. * * @param string $path * @param string $newpath @@ -107,14 +129,16 @@ public function rename(string $path, string $newpath): bool $remoteNewPath = ($pathInfo['dirname'] != '.') ? $newPathInfo['dirname'] . '/' . $newPathInfo['filename'] : $newPathInfo['filename']; - $result = $this->uploadApi()->rename($remotePath, $remoteNewPath); + $result = $this->uploadApi()->rename( + $this->preparePublicId($remotePath), + $this->preparePublicId($remoteNewPath) + ); return $result['public_id'] == $newPathInfo['filename']; } /** - * Expose the Cloudinary v2 Upload Functionality - * + * Expose the Cloudinary v2 Upload Functionality. */ protected function uploadApi(): UploadApi { @@ -147,6 +171,9 @@ protected function upload(string $file, array $options = []): void */ public function copy(string $source, string $destination, Config $config): void { + $source = $this->preparePublicId($source); + $destination = $this->preparePublicId($destination); + $this->uploadApi()->upload($source, ['public_id' => $destination]); } @@ -160,8 +187,7 @@ public function copy(string $source, string $destination, Config $config): void public function delete(string $path): void { try { - - $result = $this->uploadApi()->destroy($path); + $result = $this->uploadApi()->destroy($this->preparePublicId($path)); $finalResult = is_array($result) && $result['result'] == 'ok'; if ($finalResult != 'ok') { @@ -221,7 +247,7 @@ public function createDirectory(string $path, Config $config): void public function fileExists(string $path): bool { try { - $this->adminApi()->asset($path); + $this->adminApi()->asset($this->preparePublicId($path)); } catch (Exception) { return false; } @@ -250,7 +276,7 @@ public function directoryExists(string $path): bool */ public function read(string $path): string { - $resource = (array)$this->adminApi()->asset($path); + $resource = (array)$this->adminApi()->asset($this->preparePublicId($path)); return file_get_contents($resource['secure_url']); } @@ -264,7 +290,7 @@ public function read(string $path): string */ public function readStream(string $path): bool { - $resource = (array)$this->adminApi()->asset($path); + $resource = (array)$this->adminApi()->asset($this->preparePublicId($path)); return fopen($resource['secure_url'], 'rb'); } @@ -417,7 +443,7 @@ public function getMetadata(string $path): array */ public function getResource(string $path): array { - return (array)$this->adminApi()->asset($path); + return (array)$this->adminApi()->asset($this->preparePublicId($path)); } /** diff --git a/tests/Unit/CloudinaryAdapterTest.php b/tests/Unit/CloudinaryAdapterTest.php index 600e4c1..65d06c3 100644 --- a/tests/Unit/CloudinaryAdapterTest.php +++ b/tests/Unit/CloudinaryAdapterTest.php @@ -14,3 +14,16 @@ $result = Storage::disk('cloudinary')->url($file); expect($result)->toEqual($url); }); + +it('removes extensions from media resources but not raw resources', function ($actual, $expected) { + $adapter = Storage::disk('cloudinary')->getAdapter(); + + expect($adapter->preparePublicId($actual))->toBe($expected); +})->with([ + ['file.jpg', 'file'], + ['file.png', 'file'], + ['file.gif', 'file'], + ['file.xlsx', 'file.xlsx'], + ['file.zip', 'file.zip'], + ['file.csv', 'file.csv'], +]); From 724213bacfd52c1ffcc25561962320ae66fe8846 Mon Sep 17 00:00:00 2001 From: Josh Manders Date: Wed, 14 Aug 2024 11:02:48 -0500 Subject: [PATCH 2/2] add `avif` to media extensions --- src/CloudinaryAdapter.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/CloudinaryAdapter.php b/src/CloudinaryAdapter.php index 5cb02b6..b923886 100644 --- a/src/CloudinaryAdapter.php +++ b/src/CloudinaryAdapter.php @@ -32,7 +32,7 @@ class CloudinaryAdapter implements FilesystemAdapter */ public $mediaExtensions = [ 'jpg', 'jpeg', 'png', 'gif', 'pdf', 'bmp', 'tiff', 'svg', 'ico', 'eps', 'psd', 'webp', 'jxr', 'wdp', - 'mpeg', 'mp4', 'mkv', 'mov', 'flv', 'avi', '3gp', '3g2', 'wmv', 'webm', 'ogv', 'mxf', + 'mpeg', 'mp4', 'mkv', 'mov', 'flv', 'avi', '3gp', '3g2', 'wmv', 'webm', 'ogv', 'mxf', 'avif', ]; /**