123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Security\Core\Authorization;
- use Symfony\Component\Security\Core\Authentication\Token\TokenInterface;
- use Symfony\Component\Security\Core\Authorization\Voter\VoterInterface;
- /**
- * Decorates the original AccessDecisionManager class to log information
- * about the security voters and the decisions made by them.
- *
- * @author Javier Eguiluz <javier.eguiluz@gmail.com>
- *
- * @internal
- */
- class TraceableAccessDecisionManager implements AccessDecisionManagerInterface
- {
- private $manager;
- private $strategy;
- private $voters = [];
- private $decisionLog = []; // All decision logs
- private $currentLog = []; // Logs being filled in
- public function __construct(AccessDecisionManagerInterface $manager)
- {
- $this->manager = $manager;
- if ($this->manager instanceof AccessDecisionManager) {
- // The strategy and voters are stored in a private properties of the decorated service
- $reflection = new \ReflectionProperty(AccessDecisionManager::class, 'strategy');
- $reflection->setAccessible(true);
- $this->strategy = $reflection->getValue($manager);
- $reflection = new \ReflectionProperty(AccessDecisionManager::class, 'voters');
- $reflection->setAccessible(true);
- $this->voters = $reflection->getValue($manager);
- }
- }
- /**
- * {@inheritdoc}
- *
- * @param bool $allowMultipleAttributes Whether to allow passing multiple values to the $attributes array
- */
- public function decide(TokenInterface $token, array $attributes, $object = null/*, bool $allowMultipleAttributes = false*/): bool
- {
- $currentDecisionLog = [
- 'attributes' => $attributes,
- 'object' => $object,
- 'voterDetails' => [],
- ];
- $this->currentLog[] = &$currentDecisionLog;
- $result = $this->manager->decide($token, $attributes, $object, 3 < \func_num_args() && func_get_arg(3));
- $currentDecisionLog['result'] = $result;
- $this->decisionLog[] = array_pop($this->currentLog); // Using a stack since decide can be called by voters
- return $result;
- }
- /**
- * Adds voter vote and class to the voter details.
- *
- * @param array $attributes attributes used for the vote
- * @param int $vote vote of the voter
- */
- public function addVoterVote(VoterInterface $voter, array $attributes, int $vote)
- {
- $currentLogIndex = \count($this->currentLog) - 1;
- $this->currentLog[$currentLogIndex]['voterDetails'][] = [
- 'voter' => $voter,
- 'attributes' => $attributes,
- 'vote' => $vote,
- ];
- }
- public function getStrategy(): string
- {
- // The $strategy property is misleading because it stores the name of its
- // method (e.g. 'decideAffirmative') instead of the original strategy name
- // (e.g. 'affirmative')
- return null === $this->strategy ? '-' : strtolower(substr($this->strategy, 6));
- }
- /**
- * @return iterable|VoterInterface[]
- */
- public function getVoters(): iterable
- {
- return $this->voters;
- }
- public function getDecisionLog(): array
- {
- return $this->decisionLog;
- }
- }
- class_alias(TraceableAccessDecisionManager::class, DebugAccessDecisionManager::class);
|