vendor/shopware/core/Framework/Routing/ApiRequestContextResolver.php line 323

  1. <?php declare(strict_types=1);
  2. namespace Shopware\Core\Framework\Routing;
  3. use Doctrine\DBAL\Connection;
  4. use Shopware\Core\Checkout\Cart\Price\Struct\CartPrice;
  5. use Shopware\Core\Defaults;
  6. use Shopware\Core\Framework\Api\Context\AdminApiSource;
  7. use Shopware\Core\Framework\Api\Context\ContextSource;
  8. use Shopware\Core\Framework\Api\Context\SalesChannelApiSource;
  9. use Shopware\Core\Framework\Api\Context\SystemSource;
  10. use Shopware\Core\Framework\Api\Exception\MissingPrivilegeException;
  11. use Shopware\Core\Framework\Api\Util\AccessKeyHelper;
  12. use Shopware\Core\Framework\App\Exception\AppNotFoundException;
  13. use Shopware\Core\Framework\Context;
  14. use Shopware\Core\Framework\DataAbstractionLayer\Pricing\CashRoundingConfig;
  15. use Shopware\Core\Framework\Log\Package;
  16. use Shopware\Core\Framework\Routing\Exception\LanguageNotFoundException;
  17. use Shopware\Core\Framework\Uuid\Uuid;
  18. use Shopware\Core\PlatformRequest;
  19. use Symfony\Component\HttpFoundation\Request;
  20. #[Package('core')]
  21. class ApiRequestContextResolver implements RequestContextResolverInterface
  22. {
  23.     use RouteScopeCheckTrait;
  24.     /**
  25.      * @internal
  26.      */
  27.     public function __construct(
  28.         private readonly Connection $connection,
  29.         private readonly RouteScopeRegistry $routeScopeRegistry
  30.     ) {
  31.     }
  32.     public function resolve(Request $request): void
  33.     {
  34.         if ($request->attributes->has(PlatformRequest::ATTRIBUTE_CONTEXT_OBJECT)) {
  35.             return;
  36.         }
  37.         if (!$this->isRequestScoped($requestApiContextRouteScopeDependant::class)) {
  38.             return;
  39.         }
  40.         $params $this->getContextParameters($request);
  41.         $languageIdChain $this->getLanguageIdChain($params);
  42.         $rounding $this->getCashRounding($params['currencyId']);
  43.         $context = new Context(
  44.             $this->resolveContextSource($request),
  45.             [],
  46.             $params['currencyId'],
  47.             $languageIdChain,
  48.             $params['versionId'] ?? Defaults::LIVE_VERSION,
  49.             $params['currencyFactory'],
  50.             $params['considerInheritance'],
  51.             CartPrice::TAX_STATE_GROSS,
  52.             $rounding
  53.         );
  54.         if ($request->headers->has(PlatformRequest::HEADER_SKIP_TRIGGER_FLOW)) {
  55.             $skipTriggerFlow filter_var($request->headers->get(PlatformRequest::HEADER_SKIP_TRIGGER_FLOW'false'), \FILTER_VALIDATE_BOOLEAN);
  56.             if ($skipTriggerFlow) {
  57.                 $context->addState(Context::SKIP_TRIGGER_FLOW);
  58.             }
  59.         }
  60.         $request->attributes->set(PlatformRequest::ATTRIBUTE_CONTEXT_OBJECT$context);
  61.     }
  62.     protected function getScopeRegistry(): RouteScopeRegistry
  63.     {
  64.         return $this->routeScopeRegistry;
  65.     }
  66.     /**
  67.      * @return array{currencyId: string, languageId: string, systemFallbackLanguageId: string, currencyFactory: float, currencyPrecision: int, versionId: ?string, considerInheritance: bool}
  68.      */
  69.     private function getContextParameters(Request $request): array
  70.     {
  71.         $params = [
  72.             'currencyId' => Defaults::CURRENCY,
  73.             'languageId' => Defaults::LANGUAGE_SYSTEM,
  74.             'systemFallbackLanguageId' => Defaults::LANGUAGE_SYSTEM,
  75.             'currencyFactory' => 1.0,
  76.             'currencyPrecision' => 2,
  77.             'versionId' => $request->headers->get(PlatformRequest::HEADER_VERSION_ID),
  78.             'considerInheritance' => false,
  79.         ];
  80.         $runtimeParams $this->getRuntimeParameters($request);
  81.         /** @var array{currencyId: string, languageId: string, systemFallbackLanguageId: string, currencyFactory: float, currencyPrecision: int, versionId: ?string, considerInheritance: bool} $params */
  82.         $params array_replace_recursive($params$runtimeParams);
  83.         return $params;
  84.     }
  85.     private function getRuntimeParameters(Request $request): array
  86.     {
  87.         $parameters = [];
  88.         if ($request->headers->has(PlatformRequest::HEADER_LANGUAGE_ID)) {
  89.             $langHeader $request->headers->get(PlatformRequest::HEADER_LANGUAGE_ID);
  90.             if ($langHeader !== null) {
  91.                 $parameters['languageId'] = $langHeader;
  92.             }
  93.         }
  94.         if ($request->headers->has(PlatformRequest::HEADER_CURRENCY_ID)) {
  95.             $currencyHeader $request->headers->get(PlatformRequest::HEADER_CURRENCY_ID);
  96.             if ($currencyHeader !== null) {
  97.                 $parameters['currencyId'] = $currencyHeader;
  98.             }
  99.         }
  100.         if ($request->headers->has(PlatformRequest::HEADER_INHERITANCE)) {
  101.             $parameters['considerInheritance'] = true;
  102.         }
  103.         return $parameters;
  104.     }
  105.     private function resolveContextSource(Request $request): ContextSource
  106.     {
  107.         if ($userId $request->attributes->get(PlatformRequest::ATTRIBUTE_OAUTH_USER_ID)) {
  108.             $appIntegrationId $request->headers->get(PlatformRequest::HEADER_APP_INTEGRATION_ID);
  109.             // The app integration id header is only to be used by a privileged user
  110.             if ($this->userAppIntegrationHeaderPrivileged($userId$appIntegrationId)) {
  111.                 $userId null;
  112.             } else {
  113.                 $appIntegrationId null;
  114.             }
  115.             return $this->getAdminApiSource($userId$appIntegrationId);
  116.         }
  117.         if (!$request->attributes->has(PlatformRequest::ATTRIBUTE_OAUTH_ACCESS_TOKEN_ID)) {
  118.             return new SystemSource();
  119.         }
  120.         $clientId $request->attributes->get(PlatformRequest::ATTRIBUTE_OAUTH_CLIENT_ID);
  121.         $keyOrigin AccessKeyHelper::getOrigin($clientId);
  122.         if ($keyOrigin === 'user') {
  123.             $userId $this->getUserIdByAccessKey($clientId);
  124.             return $this->getAdminApiSource($userId);
  125.         }
  126.         if ($keyOrigin === 'integration') {
  127.             $integrationId $this->getIntegrationIdByAccessKey($clientId);
  128.             return $this->getAdminApiSource(null$integrationId);
  129.         }
  130.         if ($keyOrigin === 'sales-channel') {
  131.             $salesChannelId $this->getSalesChannelIdByAccessKey($clientId);
  132.             return new SalesChannelApiSource($salesChannelId);
  133.         }
  134.         return new SystemSource();
  135.     }
  136.     /**
  137.      * @param array{languageId: string, systemFallbackLanguageId: string} $params
  138.      *
  139.      * @return non-empty-array<string>
  140.      */
  141.     private function getLanguageIdChain(array $params): array
  142.     {
  143.         $chain = [$params['languageId']];
  144.         if ($chain[0] === Defaults::LANGUAGE_SYSTEM) {
  145.             return $chain// no query needed
  146.         }
  147.         // `Context` ignores nulls and duplicates
  148.         $chain[] = $this->getParentLanguageId($chain[0]);
  149.         $chain[] = $params['systemFallbackLanguageId'];
  150.         /** @var non-empty-array<string> $filtered */
  151.         $filtered array_filter($chain);
  152.         return $filtered;
  153.     }
  154.     private function getParentLanguageId(?string $languageId): ?string
  155.     {
  156.         if ($languageId === null || !Uuid::isValid($languageId)) {
  157.             throw new LanguageNotFoundException($languageId);
  158.         }
  159.         $data $this->connection->createQueryBuilder()
  160.             ->select(['LOWER(HEX(language.parent_id))'])
  161.             ->from('language')
  162.             ->where('language.id = :id')
  163.             ->setParameter('id'Uuid::fromHexToBytes($languageId))
  164.             ->executeQuery()
  165.             ->fetchFirstColumn();
  166.         if (empty($data)) {
  167.             throw new LanguageNotFoundException($languageId);
  168.         }
  169.         return $data[0];
  170.     }
  171.     private function getUserIdByAccessKey(string $clientId): string
  172.     {
  173.         $id $this->connection->createQueryBuilder()
  174.             ->select(['user_id'])
  175.             ->from('user_access_key')
  176.             ->where('access_key = :accessKey')
  177.             ->setParameter('accessKey'$clientId)
  178.             ->executeQuery()
  179.             ->fetchOne();
  180.         return Uuid::fromBytesToHex($id);
  181.     }
  182.     private function getSalesChannelIdByAccessKey(string $clientId): string
  183.     {
  184.         $id $this->connection->createQueryBuilder()
  185.             ->select(['id'])
  186.             ->from('sales_channel')
  187.             ->where('access_key = :accessKey')
  188.             ->setParameter('accessKey'$clientId)
  189.             ->executeQuery()
  190.             ->fetchOne();
  191.         return Uuid::fromBytesToHex($id);
  192.     }
  193.     private function getIntegrationIdByAccessKey(string $clientId): string
  194.     {
  195.         $id $this->connection->createQueryBuilder()
  196.             ->select(['id'])
  197.             ->from('integration')
  198.             ->where('access_key = :accessKey')
  199.             ->setParameter('accessKey'$clientId)
  200.             ->executeQuery()
  201.             ->fetchOne();
  202.         return Uuid::fromBytesToHex($id);
  203.     }
  204.     private function getAdminApiSource(?string $userId, ?string $integrationId null): AdminApiSource
  205.     {
  206.         $source = new AdminApiSource($userId$integrationId);
  207.         // Use the permissions associated to that app, if the request is made by an integration associated to an app
  208.         $appPermissions $this->fetchPermissionsIntegrationByApp($integrationId);
  209.         if ($appPermissions !== null) {
  210.             $source->setIsAdmin(false);
  211.             $source->setPermissions($appPermissions);
  212.             return $source;
  213.         }
  214.         if ($userId !== null) {
  215.             $source->setPermissions($this->fetchPermissions($userId));
  216.             $source->setIsAdmin($this->isAdmin($userId));
  217.             return $source;
  218.         }
  219.         if ($integrationId !== null) {
  220.             $source->setIsAdmin($this->isAdminIntegration($integrationId));
  221.             $source->setPermissions($this->fetchIntegrationPermissions($integrationId));
  222.             return $source;
  223.         }
  224.         return $source;
  225.     }
  226.     private function isAdmin(string $userId): bool
  227.     {
  228.         return (bool) $this->connection->fetchOne(
  229.             'SELECT admin FROM `user` WHERE id = :id',
  230.             ['id' => Uuid::fromHexToBytes($userId)]
  231.         );
  232.     }
  233.     private function isAdminIntegration(string $integrationId): bool
  234.     {
  235.         return (bool) $this->connection->fetchOne(
  236.             'SELECT admin FROM `integration` WHERE id = :id',
  237.             ['id' => Uuid::fromHexToBytes($integrationId)]
  238.         );
  239.     }
  240.     private function fetchPermissions(string $userId): array
  241.     {
  242.         $permissions $this->connection->createQueryBuilder()
  243.             ->select(['role.privileges'])
  244.             ->from('acl_user_role''mapping')
  245.             ->innerJoin('mapping''acl_role''role''mapping.acl_role_id = role.id')
  246.             ->where('mapping.user_id = :userId')
  247.             ->setParameter('userId'Uuid::fromHexToBytes($userId))
  248.             ->executeQuery()
  249.             ->fetchFirstColumn();
  250.         $list = [];
  251.         foreach ($permissions as $privileges) {
  252.             $privileges json_decode((string) $privilegestrue512\JSON_THROW_ON_ERROR);
  253.             $list array_merge($list$privileges);
  254.         }
  255.         return array_unique(array_filter($list));
  256.     }
  257.     private function getCashRounding(string $currencyId): CashRoundingConfig
  258.     {
  259.         $rounding $this->connection->fetchAssociative(
  260.             'SELECT item_rounding FROM currency WHERE id = :id',
  261.             ['id' => Uuid::fromHexToBytes($currencyId)]
  262.         );
  263.         if ($rounding === false) {
  264.             throw new \RuntimeException(sprintf('No cash rounding for currency "%s" found'$currencyId));
  265.         }
  266.         $rounding json_decode((string) $rounding['item_rounding'], true512\JSON_THROW_ON_ERROR);
  267.         return new CashRoundingConfig(
  268.             (int) $rounding['decimals'],
  269.             (float) $rounding['interval'],
  270.             (bool) $rounding['roundForNet']
  271.         );
  272.     }
  273.     private function fetchPermissionsIntegrationByApp(?string $integrationId): ?array
  274.     {
  275.         if (!$integrationId) {
  276.             return null;
  277.         }
  278.         $privileges $this->connection->fetchOne('
  279.             SELECT `acl_role`.`privileges`
  280.             FROM `acl_role`
  281.             INNER JOIN `app` ON `app`.`acl_role_id` = `acl_role`.`id`
  282.             WHERE `app`.`integration_id` = :integrationId
  283.         ', ['integrationId' => Uuid::fromHexToBytes($integrationId)]);
  284.         if ($privileges === false) {
  285.             return null;
  286.         }
  287.         return json_decode((string) $privilegestrue512\JSON_THROW_ON_ERROR);
  288.     }
  289.     private function fetchIntegrationPermissions(string $integrationId): array
  290.     {
  291.         $permissions $this->connection->createQueryBuilder()
  292.             ->select(['role.privileges'])
  293.             ->from('integration_role''mapping')
  294.             ->innerJoin('mapping''acl_role''role''mapping.acl_role_id = role.id')
  295.             ->where('mapping.integration_id = :integrationId')
  296.             ->setParameter('integrationId'Uuid::fromHexToBytes($integrationId))
  297.             ->executeQuery()
  298.             ->fetchFirstColumn();
  299.         $list = [];
  300.         foreach ($permissions as $privileges) {
  301.             $privileges json_decode((string) $privilegestrue512\JSON_THROW_ON_ERROR);
  302.             $list array_merge($list$privileges);
  303.         }
  304.         return array_unique(array_filter($list));
  305.     }
  306.     private function fetchAppNameByIntegrationId(string $integrationId): ?string
  307.     {
  308.         $name $this->connection->createQueryBuilder()
  309.             ->select(['app.name'])
  310.             ->from('app''app')
  311.             ->innerJoin('app''integration''integration''integration.id = app.integration_id')
  312.             ->where('integration.id = :integrationId')
  313.             ->andWhere('app.active = 1')
  314.             ->setParameter('integrationId'Uuid::fromHexToBytes($integrationId))
  315.             ->executeQuery()
  316.             ->fetchOne();
  317.         if ($name === false) {
  318.             return null;
  319.         }
  320.         return $name;
  321.     }
  322.     /**
  323.      * @throws MissingPrivilegeException
  324.      * @throws AppNotFoundException
  325.      */
  326.     private function userAppIntegrationHeaderPrivileged(string $userId, ?string $appIntegrationId): bool
  327.     {
  328.         if ($appIntegrationId === null) {
  329.             return false;
  330.         }
  331.         $appName $this->fetchAppNameByIntegrationId($appIntegrationId);
  332.         if ($appName === null) {
  333.             throw new AppNotFoundException($appIntegrationId);
  334.         }
  335.         $isAdmin $this->isAdmin($userId);
  336.         if ($isAdmin) {
  337.             return true;
  338.         }
  339.         $permissions $this->fetchPermissions($userId);
  340.         $allAppsPrivileged \in_array('app.all'$permissionstrue);
  341.         $appPrivilegeName \sprintf('app.%s'$appName);
  342.         $specificAppPrivileged \in_array($appPrivilegeName$permissionstrue);
  343.         if (!($specificAppPrivileged || $allAppsPrivileged)) {
  344.             throw new MissingPrivilegeException([$appPrivilegeName]);
  345.         }
  346.         return true;
  347.     }
  348. }