GOOD SHELL MAS BOY
Server: Apache/2.4.52 (Ubuntu)
System: Linux vmi1836763.contaboserver.net 5.15.0-130-generic #140-Ubuntu SMP Wed Dec 18 17:59:53 UTC 2024 x86_64
User: www-data (33)
PHP: 8.4.10
Disabled: NONE
Upload Files
File: /var/www/html/vendor/cloudinary-labs/cloudinary-laravel/src/CloudinaryAdapter.php
<?php

namespace CloudinaryLabs\CloudinaryLaravel;

use Cloudinary\Api\Admin\AdminApi;
use Cloudinary\Api\Exception\ApiError;
use Cloudinary\Api\Upload\UploadApi;
use Cloudinary\Cloudinary;
use Cloudinary\Configuration\Configuration;
use Exception;
use Illuminate\Support\Str;
use League\Flysystem\Config;
use League\Flysystem\FileAttributes;
use League\Flysystem\FilesystemAdapter;
use League\Flysystem\StorageAttributes;
use League\Flysystem\UnableToDeleteFile;
use League\Flysystem\UnableToSetVisibility;
use Throwable;

/**
 * Class CloudinaryAdapter
 */
class CloudinaryAdapter implements FilesystemAdapter
{
    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',
        'avif',
    ];

    /**
     * Constructor
     * Sets configuration for Cloudinary Api.
     *
     * @param  Configuration|string|array|null  $config  Cloudinary configuration
     */
    public function __construct($config = null)
    {
        $this->cloudinary = new Cloudinary($config);
    }

    /**
     * Write a new file.
     * Create temporary stream with content.
     * Pass to writeStream.
     *
     * @param  Config  $config  Config object
     * @return void false on failure file meta data on success
     *
     * @throws ApiError
     */
    public function write(string $path, string $contents, Config $config): void
    {
        $tempFile = tmpfile();

        fwrite($tempFile, $contents);

        $this->writeStream($path, $tempFile, $config);
    }

    /**
     * Write a new file using a stream
     *
     * @param  resource  $contents
     * @param  Config  $config  Config object
     * @return void false on failure file meta data on success
     *
     * @throws ApiError
     */
    public function writeStream(string $path, $contents, Config $config): void
    {
        $publicId = $this->preparePublicId($config->get('public_id', $path));

        $resourceType = $config->get('resource_type', 'auto');

        $uploadOptions = [
            'public_id' => $publicId,
            'resource_type' => $resourceType,
        ];

        $resourceMetadata = stream_get_meta_data($contents);

        $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)->toString();
    }

    /**
     * Determine if the given extension is a media extension.
     */
    public function isMedia($extension): bool
    {
        return in_array($extension, $this->mediaExtensions);
    }

    /**
     * Rename a file.
     */
    public function rename(string $path, string $newpath): bool
    {
        $pathInfo = pathinfo($path);
        $newPathInfo = pathinfo($newpath);

        $remotePath = ($pathInfo['dirname'] != '.') ? $pathInfo['dirname'].'/'.$pathInfo['filename'] : $pathInfo['filename'];

        $remoteNewPath = ($pathInfo['dirname'] != '.') ? $newPathInfo['dirname'].'/'.$newPathInfo['filename'] : $newPathInfo['filename'];

        $result = $this->uploadApi()->rename(
            $this->preparePublicId($remotePath),
            $this->preparePublicId($remoteNewPath)
        );

        return $result['public_id'] == $newPathInfo['filename'];
    }

    /**
     * Expose the Cloudinary v2 Upload Functionality.
     */
    protected function uploadApi(): UploadApi
    {
        return $this->cloudinary->uploadApi();
    }

    /**
     * Upload a file
     *
     *
     * @throws ApiError
     */
    protected function upload(string $file, array $options = []): void
    {
        $this->uploadApi()->upload($file, $options);
    }

    /**
     * Copy a file.
     * Copy content from existing url.
     *
     * @throws ApiError
     */
    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]);
    }

    /**
     * Delete a file.
     */
    public function delete(string $path): void
    {
        try {
            $result = $this->uploadApi()->destroy($this->preparePublicId($path));
            $finalResult = is_array($result) && $result['result'] == 'ok';

            if ($finalResult != 'ok') {
                throw new UnableToDeleteFile('file not found');
            }
        } catch (Throwable $exception) {
            throw UnableToDeleteFile::atLocation($path, '', $exception);
        }
    }

    /**
     * Delete a directory.
     * Delete Files using directory as a prefix.
     *
     *
     *
     * @throws ApiError
     */
    public function deleteDirectory(string $path): void
    {
        $this->adminApi()->deleteAssetsByPrefix($path);
    }

    /**
     * Expose the Cloudinary v2 Upload Functionality
     */
    protected function adminApi(): AdminApi
    {
        return $this->cloudinary->adminApi();
    }

    /**
     * Create a directory.
     *
     * @param  string  $path  directory name
     *
     * @throws ApiError
     */
    public function createDirectory(string $path, Config $config): void
    {
        $this->adminApi()->createFolder($path);
    }

    /**
     * Check whether a file exists.
     */
    public function fileExists(string $path): bool
    {
        try {
            $this->adminApi()->asset($this->preparePublicId($path));
        } catch (Exception) {
            return false;
        }

        return true;
    }

    /**
     * Check whether a directory exists.
     */
    public function directoryExists(string $path): bool
    {
        return $this->fileExists($path);
    }

    /**
     * Read a file.
     */
    public function read(string $path): string
    {
        try {
            $resource = (array) $this->adminApi()->asset($this->preparePublicId($path));
        } catch (Exception) {
            $resource = (array) $this->adminApi()->asset(
                $this->preparePublicId($path), ['resource_type' => 'video']
            );
        }

        return file_get_contents($resource['secure_url']);
    }

    /**
     * Read a file as a stream.
     *
     *
     * @return resource
     */
    public function readStream(string $path)
    {
        $resource = (array) $this->adminApi()->asset($this->preparePublicId($path));

        return fopen($resource['secure_url'], 'rb');
    }

    /**
     * Set visibility for the file
     *
     * @param  mixed  $visibility
     *
     * @throws UnableToSetVisibility
     */
    public function setVisibility(string $path, $visibility): void
    {
        throw UnableToSetVisibility::atLocation($path, 'Cloudinary API does not support visibility.');
    }

    /**
     * Check visibility of the file
     *
     * @throws UnableToSetVisibility
     */
    public function visibility(string $path): FileAttributes
    {
        throw UnableToSetVisibility::atLocation($path, 'Cloudinary API does not support visibility.');
    }

    /**
     * List contents of a directory.
     *
     * @return iterable<StorageAttributes>
     */
    public function listContents(string $path = '', bool $deep = false): iterable
    {
        $resources = [];

        // get resources array
        $response = null;
        do {
            $response = (array) $this->adminApi()->assets(
                [
                    'type' => 'upload',
                    'prefix' => $path,
                    'max_results' => 500,
                    'next_cursor' => $response['next_cursor'] ?? null,
                ]
            );
            $resources = array_merge($resources, $response['resources']);
        } while (array_key_exists('next_cursor', $response));

        // parse resourses
        foreach ($resources as $i => $resource) {
            $resources[$i] = $this->prepareFileAttributes($this->prepareResourceMetadata($resource));
        }

        return $resources;
    }

    /**
     * Transform array of resource metadata into a {@link FileAttributes} instance.
     */
    protected function prepareFileAttributes(array $metadata): FileAttributes
    {
        return new FileAttributes(
            $metadata['path'],
            $metadata['size'],
            null,
            $metadata['timestamp'],
            $metadata['mimetype'],
            $metadata
        );
    }

    /**
     * Prepare apropriate metadata for resource metadata given from cloudinary.
     */
    protected function prepareResourceMetadata(array $resource): array
    {
        $resource['type'] = 'file';
        $resource['path'] = $resource['public_id'];
        $resource = array_merge($resource, $this->prepareSize($resource));
        $resource = array_merge($resource, $this->prepareTimestamp($resource));

        return array_merge($resource, $this->prepareMimetype($resource));
    }

    /**
     * prepare size response
     */
    protected function prepareSize(array $resource): array
    {
        $size = $resource['bytes'];

        return compact('size');
    }

    /**
     * prepare timestamp response
     */
    protected function prepareTimestamp(array $resource): array
    {
        $timestamp = strtotime($resource['created_at']);

        return compact('timestamp');
    }

    /**
     * prepare mimetype response
     */
    protected function prepareMimetype(array $resource): array
    {
        $mimetype = $resource['resource_type'];

        return compact('mimetype');
    }

    /**
     * Get all the meta data of a file or directory.
     */
    public function getMetadata(string $path): array
    {
        return $this->prepareResourceMetadata($this->getResource($path));
    }

    /**
     * Get Resource data
     */
    public function getResource(string $path): array
    {
        return (array) $this->adminApi()->asset($this->preparePublicId($path));
    }

    /**
     * Get all the meta data of a file or directory.
     */
    public function getSize(string $path): array
    {
        return $this->prepareSize($this->getResource($path));
    }

    /**
     * Get the mimetype of a file.
     */
    public function getMimetype(string $path): array
    {
        return $this->prepareMimetype($this->getResource($path));
    }

    /**
     * Get the mimetype of a file.
     */
    public function mimeType(string $path): FileAttributes
    {
        $mimeType = $this->getMimetype($path);

        return new FileAttributes($path, null, null, null, $mimeType['mimetype']);
    }

    /**
     * Get the timestamp of a file.
     */
    public function lastModified(string $path): FileAttributes
    {
        $timeStamp = $this->getTimestamp($path);

        return new FileAttributes($path, null, null, $timeStamp['timestamp']);
    }

    /**
     * Get the filesize of a file
     */
    public function fileSize(string $path): FileAttributes
    {
        $fileSize = $this->getSize($path);

        return new FileAttributes($path, $fileSize['size']);
    }

    /**
     * Move a file to another location
     *
     *
     * @throws ApiError
     */
    public function move(string $source, string $destination, Config $config): void
    {
        $this->copy($source, $destination, $config);
        $this->delete($source);
    }

    /**
     * Get the timestamp of a file.
     */
    public function getTimestamp(string $path): array
    {
        return $this->prepareTimestamp($this->getResource($path));
    }

    /**
     * Get the url of a file
     */
    public function getUrl(string $path): string
    {
        if ($path == '/') {
            return $path;
        }
        try {
            $resource = $this->getResource(Str::beforeLast($path, '.'));

            return $resource['secure_url'] ?? '';
        } catch (Exception) {
            return '';
        }
    }
}