我想使用Voter只允许所有者在我的应用程序中编辑项目对象.
我有一个route/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;
}
}
Run Code Online (Sandbox Code Playgroud)
我在configure.yml中注册它,以便它获取服务容器作为参数:
services:
fuxcon2013.security.owner_voter:
class: FUxCon2013\ProjectsBundle\Security\OwnerVoter
public: false
arguments: [ @service_container ]
tags:
- { name: security.voter }
Run Code Online (Sandbox Code Playgroud)
最后一个块是将security.yml中的访问决策管理器配置为一致:
security:
access_decision_manager:
# strategy can be: affirmative, unanimous or consensus
strategy: unanimous
allow_if_all_abstain: true
Run Code Online (Sandbox Code Playgroud)