components-web-app/api-components-bundle

View on GitHub
src/Serializer/ContextBuilder/CwaResourceContextBuilder.php

Summary

Maintainability
A
1 hr
Test Coverage
<?php

/*
 * This file is part of the Silverback API Components Bundle Project
 *
 * (c) Daniel West <daniel@silverback.is>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

declare(strict_types=1);

namespace Silverback\ApiComponentsBundle\Serializer\ContextBuilder;

use ApiPlatform\Serializer\SerializerContextBuilderInterface;
use Silverback\ApiComponentsBundle\Entity\Core\AbstractComponent;
use Silverback\ApiComponentsBundle\Entity\Core\AbstractPageData;
use Silverback\ApiComponentsBundle\Serializer\MappingLoader\CwaResourceLoader;
use Symfony\Bundle\SecurityBundle\Security;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\Security\Core\Role\RoleHierarchyInterface;

/**
 * @author Daniel West <daniel@silverback.is>
 */
final class CwaResourceContextBuilder implements SerializerContextBuilderInterface
{
    public function __construct(
        private SerializerContextBuilderInterface $decorated,
        private RoleHierarchyInterface $roleHierarchy,
        private Security $security,
    ) {
    }

    public function createFromRequest(Request $request, bool $normalization, ?array $extractedAttributes = null): array
    {
        $context = $this->decorated->createFromRequest($request, $normalization, $extractedAttributes);

        if (
            !is_a($resourceClass = $context['resource_class'], AbstractComponent::class, true)
            && !is_a($resourceClass, AbstractPageData::class, true)
        ) {
            return $context;
        }

        $reflectionClass = new \ReflectionClass($resourceClass);
        $shortName = $reflectionClass->getShortName();
        $componentNames = [$shortName];
        while ($parent = $reflectionClass->getParentClass()) {
            $componentNames[] = $parent->getShortName();
            $reflectionClass = $parent;
        }
        $rw = $normalization ? 'read' : 'write';
        foreach ($componentNames as $componentName) {
            $context['groups'][] = sprintf('%s:%s:%s', $componentName, CwaResourceLoader::GROUP_NAME, $rw);
        }

        $user = $this->security->getUser();
        if ($user) {
            $reachableRoles = $this->roleHierarchy->getReachableRoleNames($user->getRoles());
            foreach ($reachableRoles as $reachableRole) {
                $context['groups'][] = sprintf('%s:%s:%s:%s', $shortName, CwaResourceLoader::GROUP_NAME, $rw, strtolower($reachableRole));
            }
        }

        return $context;
    }
}