开发者问题收集

Symfony / API 平台 - 自定义操作 - 使用 VichUploaderBundle 进行文件替换的 PUT 方法

2019-09-03
2569

我正在使用 Vich Uploader Bundle 与 API Platform 结合来存储文件。我已遵循官方文档中的说明 https://api-platform.com/docs/core/file-upload/#handling-file-upload

一切运行良好,只是我想包含 PUT 调用来替换现有的 MediaObject 实体。我的尝试如下:

我创建了一个自定义操作并将其映射到实体中的 PUT 方法,如下所示:

<?php
// api/src/Entity/MediaObject.php

namespace App\Entity;

use ApiPlatform\Core\Annotation\ApiProperty;
use ApiPlatform\Core\Annotation\ApiResource;
use App\Controller\CreateMediaObject;
use App\Controller\EditMediaObject;
use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\HttpFoundation\File\File;
use Symfony\Component\Serializer\Annotation\Groups;
use Symfony\Component\Validator\Constraints as Assert;
use Vich\UploaderBundle\Mapping\Annotation as Vich;

/**
 * @ORM\Entity
 * @ApiResource(
 * 
 *     ...
 * 
 *     itemOperations={
 *         "get",
 *         "put"={
 *             "controller"=EditMediaObject::class,
 *             "deserialize"=false,
 *             "validation_groups"={"Default", "media_object_create"},
 *             "swagger_context"={
 *                 "consumes"={
 *                     "multipart/form-data",
 *                 },
 *                 "parameters"={
 *                     {
 *                         "in"="formData",
 *                         "name"="file",
 *                         "type"="file",
 *                         "description"="The file to upload",
 *                     },
 *                 },
 *             },
 *         },
 *         "delete"
 *     },
 * )
 * @Vich\Uploadable
 */
class MediaObject
{
    /**
     * @var int|null
     *
     * @ORM\Column(type="integer")
     * @ORM\GeneratedValue
     * @ORM\Id
     */
    protected $id;

    /**
     * @var string|null
     *
     * @ApiProperty(iri="http://schema.org/contentUrl")
     * @Groups({"media_object_read"})
     */
    public $contentUrl;

    /**
     * @var File|null
     *
     * @Assert\NotNull(groups={"media_object_create"})
     * @Vich\UploadableField(mapping="media_object", fileNameProperty="filePath")
     */
    public $file;

    /**
     * @var string|null
     *
     * @ORM\Column(nullable=true)
     */
    public $filePath;

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

相应的控制器:

<?php
// api/src/Controller/EditMediaObject.php

namespace App\Controller;

use App\Entity\MediaObject;
use Symfony\Component\HttpFoundation\RequestStack;
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;

final class EditMediaObject
{
    /**
    * @var RequestStack
    */
    private $requestStack;


    public function __construct(RequestStack $requestStack)
    {
        $this->requestStack = $requestStack;
    }
    public function __invoke(MediaObject $data): MediaObject
    {
        $request = $this->requestStack->getCurrentRequest();
        $uploadedFile = $request->attributes->get('file');
        if (!$uploadedFile) {
            throw new BadRequestHttpException ('"file" is required');
        }
        $data->file = $uploadedFile;

        return $data;
    }
}

$data 对象已成功填充。但是我无法访问初始请求参数(特别是“文件”)。

这是因为初步的 OPTIONS 调用吗?如果是这样,我该如何转发/保存文件以将其附加到 MediaObject?

提前致谢

编辑:

如果有人感兴趣,在 PHP 中不可能通过 PUT 请求发送文件参数。(来源: PHP 多部分表单数据 PUT 请求?

3个回答

您为什么要查看 $request->attributes ? 您应该查看 $request->files

请参阅 https://symfony.com/doc/current/components/http_foundation.html#accessing-request-data

Tristan
2019-09-04

您需要发送一个带有额外参数的 POST 请求, _method = PUT ,它会将 POST 请求更改为 PUT 请求。

user15873
2019-11-28

我最近遇到了一个非常相似的问题,但是,我没有使用整个 symfony 库,而是使用了 HttpFoundation/RequestHttpFoundation/Response 对象。

起初,我对请求似乎完全为空感到困惑。

在花费了比我预期更多的时间研究它之后,我发现这个问题可以追溯到 PHP 的内部。

我设法找到了 此 PECL 扩展

我对 pecl 并不是很熟悉,似乎无法使用 pear 使其工作。但我使用的是 CentOS 和 Remi PHP,它有一个 yum 包。

我运行了 yum install php-pecl-apfd ,它立即修复了这个问题。我相信在各种版本的 Linux 中都有其他软件包,并且我确信任何对 pear/pecl/general php 扩展有更多了解的人都可以在 Windows 或 Mac 上毫无问题地运行它。

通过修复,我的意思是我的请求对象已按预期填充(例如 request->all()request->files )。

这似乎对带有和不带有文件的 PUTPATCH 请求都有效(使用多部分表单)。

如果您使用的是 Docker,则需要重新启动容器。

DazBaldwin
2020-06-26