Symfony / API 平台 - 自定义操作 - 使用 VichUploaderBundle 进行文件替换的 PUT 方法
我正在使用 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 请求? )
您为什么要查看
$request->attributes
?
您应该查看
$request->files
。
请参阅 https://symfony.com/doc/current/components/http_foundation.html#accessing-request-data
您需要发送一个带有额外参数的 POST 请求,
_method = PUT
,它会将 POST 请求更改为 PUT 请求。
我最近遇到了一个非常相似的问题,但是,我没有使用整个 symfony 库,而是使用了
HttpFoundation/Request
和
HttpFoundation/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
)。
这似乎对带有和不带有文件的
PUT
和
PATCH
请求都有效(使用多部分表单)。
如果您使用的是 Docker,则需要重新启动容器。