在php/Symfony 4中将任何base64文件转换为文件并移动到目标路径

在php/Symfony 4中将任何base64文件转换为文件并移动到目标路径,php,symfony,php-7,symfony4,Php,Symfony,Php 7,Symfony4,提前感谢我在symfony 4中创建API,以便通过POSTMAN上传base64图像或任何文件,我必须将文件移动到目标目录。我已经在控制器中编写了以下代码。我试图通过控制器将文件移动到目录中,但是 我得到的错误如下: 未捕获的警告:文件内容(images/5c78de505abdd.svg):无法打开流:没有此类文件或目录{“异常”:“[object](ErrorException(代码:0):警告:文件内容(images/5c78de505abdd.svg):无法打开流:在/home/rag

提前感谢我在
symfony 4
中创建API,以便通过
POSTMAN
上传
base64图像
或任何文件,我必须将文件移动到目标目录。我已经在控制器中编写了以下代码。我试图通过控制器将文件移动到目录中,但是

我得到的错误如下:

未捕获的警告:文件内容(images/5c78de505abdd.svg):无法打开流:没有此类文件或目录{“异常”:“[object](ErrorException(代码:0):警告:文件内容(images/5c78de505abdd.svg):无法打开流:在/home/ragiththomas/Sites/asco forum/src/Modules/forum/ForumController.php:1107中没有此类文件或目录)}[]

svg文件的邮递员请求:

{"postFile":""}
控制器文件:

/**
* @route("/uploadForumFiles", name="upload_forum_files", methods="POST")
*
* @access public
*
* @return JsonResponse
*/
public function uploadForumFiles(Request $request, PostsInterceptor $apiInterceptor, ContainerInterface $container): JsonResponse {
try {
$fileContent = $requestDtoObj->postFile;
$target_dir = 'images/'; // add the specific path to save the file
$decoded_file = base64_decode($fileContent); // decode the file
$mime_type = finfo_buffer(finfo_open(), $decoded_file, FILEINFO_MIME_TYPE); // extract mime type
$extension = $this->mime2ext($mime_type); // extract extension from mime type
$file = uniqid() .'.'. $extension; // rename file as a unique name
$file_dir = $target_dir. uniqid().'.'.$extension;
file_put_contents($target_dir, $file);
echo $file_dir; die;
} catch (Exception $ex) {

        $this->logger->writeLog("ERROR", "Exception while uploading a file " . $ex->getMessage() . 'at line - ' . $ex->getLine() . ' in file' . $ex->getFile());
        return $this->responseHandler->getFailedResponse(array($ex->getMessage()), $apiInterceptor);
    }
}




public function mime2ext($mime){
$all_mimes = '{"png":["image/png","image/x-png"],"bmp":["image/bmp","image/x-bmp",
"image/x-bitmap","image/x-xbitmap","image/x-win-bitmap","image/x-windows-bmp",
"image/ms-bmp","image/x-ms-bmp","application/bmp","application/x-bmp",
"application/x-win-bitmap"],"gif":["image/gif"],"jpeg":["image/jpeg",
"image/pjpeg"],"xspf":["application/xspf+xml"],"vlc":["application/videolan"],
"wmv":["video/x-ms-wmv","video/x-ms-asf"],"au":["audio/x-au"],
"ac3":["audio/ac3"],"flac":["audio/x-flac"],"ogg":["audio/ogg",
"video/ogg","application/ogg"],"kmz":["application/vnd.google-earth.kmz"],
"kml":["application/vnd.google-earth.kml+xml"],"rtx":["text/richtext"],
"rtf":["text/rtf"],"jar":["application/java-archive","application/x-java-application",
"application/x-jar"],"zip":["application/x-zip","application/zip",
"application/x-zip-compressed","application/s-compressed","multipart/x-zip"],
"7zip":["application/x-compressed"],"xml":["application/xml","text/xml"],
"svg":["image/svg+xml","application/octet-stream"],"3g2":["video/3gpp2"],"3gp":["video/3gp","video/3gpp"],
"mp4":["video/mp4"],"m4a":["audio/x-m4a"],"f4v":["video/x-f4v"],"flv":["video/x-flv"],
"webm":["video/webm"],"aac":["audio/x-acc"],"m4u":["application/vnd.mpegurl"],
"pdf":["application/pdf"],
"pptx":["application/vnd.openxmlformats-officedocument.presentationml.presentation"],
"ppt":["application/powerpoint","application/vnd.ms-powerpoint","application/vnd.ms-office",
"application/msword"],"docx":["application/vnd.openxmlformats-officedocument.wordprocessingml.document"],
"xlsx":["application/vnd.openxmlformats-officedocument.spreadsheetml.sheet","application/vnd.ms-excel"],
"xl":["application/excel"],"xls":["application/msexcel","application/x-msexcel","application/x-ms-excel",
"application/x-excel","application/x-dos_ms_excel","application/xls","application/x-xls"],
"xsl":["text/xsl"],"mpeg":["video/mpeg"],"mov":["video/quicktime"],"avi":["video/x-msvideo",
"video/msvideo","video/avi","application/x-troff-msvideo"],"movie":["video/x-sgi-movie"],
"log":["text/x-log"],"txt":["text/plain"],"css":["text/css"],"html":["text/html"],
"wav":["audio/x-wav","audio/wave","audio/wav"],"xhtml":["application/xhtml+xml"],
"tar":["application/x-tar"],"tgz":["application/x-gzip-compressed"],"psd":["application/x-photoshop",
"image/vnd.adobe.photoshop"],"exe":["application/x-msdownload"],"js":["application/x-javascript"],
"mp3":["audio/mpeg","audio/mpg","audio/mpeg3","audio/mp3"],"rar":["application/x-rar","application/rar",
"application/x-rar-compressed"],"gzip":["application/x-gzip"],"hqx":["application/mac-binhex40",
"application/mac-binhex","application/x-binhex40","application/x-mac-binhex40"],
"cpt":["application/mac-compactpro"],"bin":["application/macbinary","application/mac-binary",
"application/x-binary","application/x-macbinary"],"oda":["application/oda"],
"ai":["application/postscript"],"smil":["application/smil"],"mif":["application/vnd.mif"],
"wbxml":["application/wbxml"],"wmlc":["application/wmlc"],"dcr":["application/x-director"],
"dvi":["application/x-dvi"],"gtar":["application/x-gtar"],"php":["application/x-httpd-php",
"application/php","application/x-php","text/php","text/x-php","application/x-httpd-php-source"],
"swf":["application/x-shockwave-flash"],"sit":["application/x-stuffit"],"z":["application/x-compress"],
"mid":["audio/midi"],"aif":["audio/x-aiff","audio/aiff"],"ram":["audio/x-pn-realaudio"],
"rpm":["audio/x-pn-realaudio-plugin"],"ra":["audio/x-realaudio"],"rv":["video/vnd.rn-realvideo"],
"jp2":["image/jp2","video/mj2","image/jpx","image/jpm"],"tiff":["image/tiff"],
"eml":["message/rfc822"],"pem":["application/x-x509-user-cert","application/x-pem-file"],
"p10":["application/x-pkcs10","application/pkcs10"],"p12":["application/x-pkcs12"],
"p7a":["application/x-pkcs7-signature"],"p7c":["application/pkcs7-mime","application/x-pkcs7-mime"],"p7r":["application/x-pkcs7-certreqresp"],"p7s":["application/pkcs7-signature"],"crt":["application/x-x509-ca-cert","application/pkix-cert"],"crl":["application/pkix-crl","application/pkcs-crl"],"pgp":["application/pgp"],"gpg":["application/gpg-keys"],"rsa":["application/x-pkcs7"],"ics":["text/calendar"],"zsh":["text/x-scriptzsh"],"cdr":["application/cdr","application/coreldraw","application/x-cdr","application/x-coreldraw","image/cdr","image/x-cdr","zz-application/zz-winassoc-cdr"],"wma":["audio/x-ms-wma"],"vcf":["text/x-vcard"],"srt":["text/srt"],"vtt":["text/vtt"],"ico":["image/x-icon","image/x-ico","image/vnd.microsoft.icon"],"csv":["text/x-comma-separated-values","text/comma-separated-values","application/vnd.msexcel"],"json":["application/json","text/json"]}';

$all_mimes = json_decode($all_mimes,true);
foreach ($all_mimes as $key => $value) {
    if(array_search($mime,$value) !== false) return $key;
}
return false;

}

file\u put\u contents
如果文件不存在,则创建该文件,但如果尝试将文件放入不存在的目录,则会失败。因此,您应该尝试以下方法:

  • 检查
    图像
    目录是否存在
  • 检查目录的写入权限
  • 尝试使用绝对路径,因此在您的情况下可能是
    $target_dir='/home/ragiththomas/Sites/asco forum/images/

file\u put\u contents
在文件不存在时创建该文件,但如果您尝试将该文件放在不存在的目录中,则会失败。因此,您应该尝试以下方法:

  • 检查
    图像
    目录是否存在
  • 检查目录的写入权限
  • 尝试使用绝对路径,因此在您的情况下可能是
    $target_dir='/home/ragiththomas/Sites/asco forum/images/

对于symfony4,另一种方法是使用vichuploadbundle。我们可以以图像为例

  • 根据本文档,安装并配置vichupload
在config/bundles.php文件中,不要忘记检查是否显示以下行:

Vich\UploaderBundle\VichUploaderBundle::class => ['all' => true],
在config/services.yaml中,添加此配置(这只是一个示例)

不要忘记在public目录中创建uploads目录,在uploads目录中创建images目录

您可以像下面这样配置文件config/packages/vich_uploader.yaml:

# config/packages/vich_uploader.yaml
vich_uploader:
    db_driver: orm

    mappings:
        object_image:
            uri_prefix: '%app.path.images%'
            upload_destination:   '%kernel.project_dir%/public%app.path.images%'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            delete_on_remove: true
            delete_on_update: true
  • 创建图像实体(带有验证断言),如下所示:

这是另一种解决方案的尝试。

对于symfony4,另一种方法是使用vichuploadbundle。我们可以以图像为例

  • 根据本文档,安装并配置vichupload
在config/bundles.php文件中,不要忘记检查是否显示以下行:

Vich\UploaderBundle\VichUploaderBundle::class => ['all' => true],
在config/services.yaml中,添加此配置(这只是一个示例)

不要忘记在public目录中创建uploads目录,在uploads目录中创建images目录

您可以像下面这样配置文件config/packages/vich_uploader.yaml:

# config/packages/vich_uploader.yaml
vich_uploader:
    db_driver: orm

    mappings:
        object_image:
            uri_prefix: '%app.path.images%'
            upload_destination:   '%kernel.project_dir%/public%app.path.images%'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            delete_on_remove: true
            delete_on_update: true
  • 创建图像实体(带有验证断言),如下所示:

这是另一种解决方案的尝试。

因此错误
警告:文件内容(images/5c78de505abdd.svg):无法打开流:没有这样的文件
是一个谜吗?在尝试放入任何内容之前,只需创建文件。我从这里创建文件$file=uniqid()$扩展;你可以在我的uploadForumFiles函数中看到我读了你的问题。它也是一种方法,而不是函数。不,您没有创建任何文件。通过创建一个文件,我的意思是物理地将其写入硬盘。这可以通过使用来完成。Symfony有一个内置的文件系统类,您可以使用它来创建文件/文件夹并向其中写入内容。@emix您在说什么,文件内容完全可以创建文件本身,如果它还不存在的话。它没有做的是自动创建包含在路径中的目录,如果这些目录实际上还不存在的话——这里似乎就是这样。因此错误
警告:文件内容(images/5c78de505abdd.svg):无法打开流:没有这样的文件
是一个谜吗?在尝试放入任何内容之前,只需创建文件。我从这里创建文件$file=uniqid()$扩展;你可以在我的uploadForumFiles函数中看到我读了你的问题。它也是一种方法,而不是函数。不,您没有创建任何文件。通过创建一个文件,我的意思是物理地将其写入硬盘。这可以通过使用来完成。Symfony有一个内置的文件系统类,您可以使用它来创建文件/文件夹并向其中写入内容。@emix您在说什么,文件内容完全可以创建文件本身,如果它还不存在的话。它没有做的是自动创建包含在路径中的目录,如果这些目录实际上还不存在的话——这里似乎就是这样path@Alexis你完全正确,硬编码路径从来不是个好主意,但在这种情况下,它只应该找到问题的原因。在Symfony中,您还可以使用
内核.project\u dir
getProjectDir()
。感谢Alexis和Pawel Napierala,使用dir后,它工作正常,您应该使用
\u dir\uuu
而不是硬编码path@Alexis你完全正确,硬编码路径从来不是个好主意,但在这种情况下,它只应该找到问题的原因。在Symfony中,您还可以使用
内核.project_dir
getProjectDir()
。感谢Alexis和Pawel Napierala,使用dir后,它工作正常

    <?php

    namespace App\Entity;

    use Doctrine\ORM\Mapping as ORM;
    use Symfony\Component\HttpFoundation\File\File;
    use Symfony\Component\HttpFoundation\File\UploadedFile;
    use Symfony\Component\Validator\Constraints as Assert;
    use Vich\UploaderBundle\Mapping\Annotation as Vich;

    /**
     * @ORM\Entity(repositoryClass="App\Repository\ImageRepository")
     * @Vich\Uploadable()
     * @ORM\Table(name="image")
     */
    class Image
    {
        /**
         * @ORM\Id()
         * @ORM\GeneratedValue()
         * @ORM\Column(type="integer")
         */
        private $id;

        /**
         * @var File|null
         * @Assert\NotBlank(message="Please upload a file.")
         * @Assert\File(
         *     maxSize = "1M",
         *     maxSizeMessage = "Maximum size allowed : {{ limit }} {{ suffix }}.",
         *     mimeTypes = {"image/png", "image/jpg", "image/jpeg"},
         *     mimeTypesMessage = "Allowed formats : png, jpg, jpeg."
         * )
         * @Vich\UploadableField(mapping="object_image", fileNameProperty="imageName")
         */
        private $imageFile;

        /**
         * @var string|null
         *
         * @ORM\Column(type="string", length=255)
         */
        private $imageName;

        /**
         * @var \DateTime
         * @ORM\Column(type="datetime")
         *
         */
        private $createdAt;

        /**
         * @var \DateTime
         * @ORM\Column(type="datetime")
         */
        private $updatedAt;

        public function __construct()
        {
            $this->createdAt = new \DateTime('now');
        }

        public function getId(): ?int
        {
            return $this->id;
        }

        /**
         * @return \DateTime
         */
        public function getCreatedAt(): \DateTime
        {
            return $this->createdAt;
        }

        /**
         * @param \DateTime $createdAt
         * @return Image
         */
        public function setCreatedAt(\DateTime $createdAt)
        {
            $this->createdAt = $createdAt;
            return $this;
        }

        /**
         * @return \DateTime
         */
        public function getUpdatedAt(): \DateTime
        {
            return $this->updatedAt;
        }

        /**
         * @param \DateTime $updatedAt
         * @return Image
         */
        public function setUpdatedAt(\DateTime $updatedAt)
        {
            $this->updatedAt = $updatedAt;

            return $this;
        }

        /**
         * @return null|File
         */
        public function getImageFile(): ?File
        {
            return $this->imageFile;
        }

        /**
         * @param null|File $imageFile
         * @return Image
         */
        public function setImageFile(?File $imageFile): Image
        {
            $this->imageFile = $imageFile;

            if($this->imageFile instanceof UploadedFile) {
                $this->updatedAt = new \DateTime('now');
            }

            return $this;
        }

        /**
         * @return null|string
         */
        public function getImageName(): ?string
        {
            return $this->imageName;
        }

        /**
         * @param null|string $imageName
         * @return Image
         */
        public function setImageName(?string $imageName): Image
        {
            $this->imageName = $imageName;
            return $this;
        }

    }


    <?php

    namespace App\Form;

    use App\Entity\Image;
    use Symfony\Component\Form\AbstractType;
    use Symfony\Component\Form\Extension\Core\Type\FileType;
    use Symfony\Component\Form\FormBuilderInterface;
    use Symfony\Component\OptionsResolver\OptionsResolver;

    class ImageType extends AbstractType
    {
        public function buildForm(FormBuilderInterface $builder, array $options)
        {
            $builder
                ->add('imageFile', FileType::class)
            ;
        }

        public function configureOptions(OptionsResolver $resolver)
        {
            $resolver->setDefaults([
                'data_class' => Image::class
            ]);
        }
    }


    <?php

    namespace App\Utils;

    use Symfony\Component\HttpFoundation\File\UploadedFile;

    class UploadedBase64File extends UploadedFile
    {

        public function __construct(string $base64Content, string $originalName)
        {
            $filePath = tempnam(sys_get_temp_dir(), 'UploadedFile');
            $data = base64_decode($this->getBase64String($base64Content));
            file_put_contents($filePath, $data);
            $error = null;
            $mimeType = null;
            $test = true;

            parent::__construct($filePath, $originalName, $mimeType, $error, $test);
        }

        private function getBase64String(string $base64Content)
        {

            $data = explode(';base64,', $base64Content);
            return $data[1];

        }

    }

    <?php

    namespace App\Controller\Api;

    use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
    use Symfony\Component\HttpFoundation\Request;
    use Symfony\Component\Routing\Annotation\Route;
    use App\Utils\UploadedBase64File;

    class CoreController extends AbstractController
    {
       /**
         * @Route("/api/images", methods={"POST"}, name="api_add_image")
         */
         public function addImage(Request $request) 
         {
             $data = json_decode($request->getContent(), true);
             if($data === null
                || !is_array($data)
                || count($data) !== 1
                || !isset($data['image']['name'], $data['image']['value'])
                || count($data['image']) !== 2
             ) {
                // Throw invalid format request for image
             }

             $imageFile = new UploadedBase64File($data['image']['value'], $data['image']['name']);
             $image = new Image();
             $form = $this->createForm(ImageType::class, $image, ['csrf_protection' => false]);
             $form->submit(['imageFile' => $imageFile]);  

             if(!($form->isSubmitted() && $form->isValid())) {
                 // Send json form error
             }

             // Persist, do thing you want to do and send json response 
         }
   }

    {
        "image": {
            "name": "originalfilename.png",
            "value": "base64 content"
        }
    }