使用Api-Platform,如何在创建时自动将授权用户添加为资源所有者?

2022-08-12 00:00:00 php symfony symfony4 api-platform.com

如何在创建时自动将当前授权用户添加到资源(POST)。

我使用的是JWT身份验证,并且/api/routes受到保护,不会被未经授权的用户访问。我想对其进行设置,以便在经过身份验证的用户创建新资源(即通过向/api/articles发送POST请求)时,新创建的Article资源与经过身份验证的用户相关。

我当前使用每个资源类型的自定义EventSubscriber从令牌存储中添加用户。

以下是订户基类的要点: https://gist.github.com/dsuurlant/5988f90e757b41454ce52050fd502273

及其扩展的实体订阅者: https://gist.github.com/dsuurlant/a8af7e6922679f45b818ec4ddad36286

但是,如果实体构造函数需要用户作为参数,则这不起作用。

例如

class Book {

    public User $owner;
    public string $name;

    public class __construct(User $user, string $name) {
        $this->owner = $user;
        $this->name  = $name;
    }
}

实体创建时如何自动注入授权用户?


解决方案

我暂时使用DTOs and data transformers。

主要缺点是必须在需要此行为的地方为每个资源创建新的DTO。

作为一个简单的例子,我这样做:

class BootDtoTransformer implements DataTransformerInterface
{

    private Security $security;

    public function __construct(Security $security)
    {
        $this->security = $security;
    }

    public function transform($data, string $to, array $context = [])
    {

        $owner = $this->security->getUser();

        return $new Book($owner, $data->name);;
    }

    public function supportsTransformation($data, string $to, array $context = []): bool
    {

        if ($data instanceof Book) {
            return false;
        }

        return Book::class === $to && null !== ($context['input']['class'] ?? null);
    }

}

这在逻辑上只对单个资源有效。为了拥有多个资源的泛型转换器,我最终使用了一些接口来区分"可拥有的"资源,并使用一些反射来实例化每个类。

我认为这在反规范化阶段是可行的,但我无法使其工作。

相关文章