6

我想使用 Voter 仅允许所有者在我的应用程序中编辑项目对象。

我有一个路由 /project/42/edit 调用我的动作 ProjectController.editAction(Project $project)。我使用类型提示 (Project $project) 自动调用 ParamConverter 将 ID 42 从 URI 转换为项目对象。这对控制器动作很有效,但是对于选民来说似乎调用得太晚了。它的 vote() 方法被调用,请求作为第二个参数,而不是我的项目。

有没有办法将项目传递给选民,而不必再次从数据库中检索它?

更新:了解到我必须在编辑方法的安全上下文中手动调用 isGranted() 。这与此答案的方法非常相似

这是我的选民:

namespace FUxCon2013\ProjectsBundle\Security;

use FUxCon2013\ProjectsBundle\Entity\Project;
use Symfony\Component\BrowserKit\Request;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\Security\Core\Authorization\Voter\VoterInterface;
use Symfony\Component\Security\Core\Authentication\Token\TokenInterface;

class OwnerVoter implements VoterInterface
{
    public function __construct(ContainerInterface $container)
    {
        $this->container     = $container;
    }

    public function supportsAttribute($attribute)
    {
        return $attribute == 'MAY_EDIT';
    }

    public function supportsClass($class)
    {
        // your voter supports all type of token classes, so return true
        return true;
    }

    function vote(TokenInterface $token, $object, array $attributes)
    {
        if (!in_array('MAY_EDIT', $attributes)) {
            return self::ACCESS_ABSTAIN;
        }
        if (!($object instanceof Project)) {
            return self::ACCESS_ABSTAIN;
        }

        $user = $token->getUser();
        $securityContext = $this->container->get('security.context');

        return $securityContext->isGranted('IS_AUTHENTICATED_FULLY')
            && $user->getId() == $object->getUser()->getId()
            ? self::ACCESS_GRANTED
            : self::ACCESS_DENIED;
    }
}

我在 configure.yml 中注册它,以便它获取服务容器作为参数:

services:
    fuxcon2013.security.owner_voter:
        class:      FUxCon2013\ProjectsBundle\Security\OwnerVoter
        public:     false
        arguments: [ @service_container ]
        tags:
            - { name: security.voter }

最后一个块是将 security.yml 中的访问决策管理器配置为一致:

security:
    access_decision_manager:
        # strategy can be: affirmative, unanimous or consensus
        strategy: unanimous
        allow_if_all_abstain: true
4

2 回答 2

1

请看一下我昨天写的这个答案。

您可以通过检查对象的所有者轻松地使其适应您的需求。

于 2013-06-13T16:27:54.797 回答
0

如果您使用角色安全处理程序,则当前对象不会在投票者中传递。

我不得不扩展后者以获得前者。

不要犹豫,评论详情。

于 2013-08-05T15:22:52.703 回答