src/Core/Framework/Routing/ApiRequestContextResolver.php line 342
<?php declare(strict_types=1);
namespace Shopware\Core\Framework\Routing;
use Doctrine\DBAL\Connection;
use Shopware\Core\Checkout\Cart\Price\Struct\CartPrice;
use Shopware\Core\Defaults;
use Shopware\Core\Framework\Api\Context\AdminApiSource;
use Shopware\Core\Framework\Api\Context\ContextSource;
use Shopware\Core\Framework\Api\Context\SalesChannelApiSource;
use Shopware\Core\Framework\Api\Context\SystemSource;
use Shopware\Core\Framework\Api\Exception\MissingPrivilegeException;
use Shopware\Core\Framework\Api\Util\AccessKeyHelper;
use Shopware\Core\Framework\App\Exception\AppNotFoundException;
use Shopware\Core\Framework\Context;
use Shopware\Core\Framework\DataAbstractionLayer\Pricing\CashRoundingConfig;
use Shopware\Core\Framework\Log\Package;
use Shopware\Core\Framework\Routing\Exception\LanguageNotFoundException;
use Shopware\Core\Framework\Uuid\Uuid;
use Shopware\Core\PlatformRequest;
use Symfony\Component\HttpFoundation\Request;
#[Package('core')]
class ApiRequestContextResolver implements RequestContextResolverInterface
{
use RouteScopeCheckTrait;
/**
* @internal
*/
public function __construct(private readonly Connection $connection, private readonly RouteScopeRegistry $routeScopeRegistry)
{
}
public function resolve(Request $request): void
{
if ($request->attributes->has(PlatformRequest::ATTRIBUTE_CONTEXT_OBJECT)) {
return;
}
if (!$this->isRequestScoped($request, ApiContextRouteScopeDependant::class)) {
return;
}
$params = $this->getContextParameters($request);
$languageIdChain = $this->getLanguageIdChain($params);
$rounding = $this->getCashRounding($params['currencyId']);
$context = new Context(
$this->resolveContextSource($request),
[],
$params['currencyId'],
$languageIdChain,
$params['versionId'] ?? Defaults::LIVE_VERSION,
$params['currencyFactory'],
$params['considerInheritance'],
CartPrice::TAX_STATE_GROSS,
$rounding
);
if ($request->headers->has(PlatformRequest::HEADER_SKIP_TRIGGER_FLOW)) {
$skipTriggerFlow = filter_var($request->headers->get(PlatformRequest::HEADER_SKIP_TRIGGER_FLOW, 'false'), \FILTER_VALIDATE_BOOLEAN);
if ($skipTriggerFlow) {
$context->addState(Context::SKIP_TRIGGER_FLOW);
}
}
$request->attributes->set(PlatformRequest::ATTRIBUTE_CONTEXT_OBJECT, $context);
}
protected function getScopeRegistry(): RouteScopeRegistry
{
return $this->routeScopeRegistry;
}
/**
* @return array{currencyId: string, languageId: string, systemFallbackLanguageId: string, currencyFactory: float, currencyPrecision: int, versionId: ?string, considerInheritance: bool}
*/
private function getContextParameters(Request $request): array
{
$params = [
'currencyId' => Defaults::CURRENCY,
'languageId' => Defaults::LANGUAGE_SYSTEM,
'systemFallbackLanguageId' => Defaults::LANGUAGE_SYSTEM,
'currencyFactory' => 1.0,
'currencyPrecision' => 2,
'versionId' => $request->headers->get(PlatformRequest::HEADER_VERSION_ID),
'considerInheritance' => false,
];
$runtimeParams = $this->getRuntimeParameters($request);
/** @var array{currencyId: string, languageId: string, systemFallbackLanguageId: string, currencyFactory: float, currencyPrecision: int, versionId: ?string, considerInheritance: bool} $params */
$params = array_replace_recursive($params, $runtimeParams);
return $params;
}
private function getRuntimeParameters(Request $request): array
{
$parameters = [];
if ($request->headers->has(PlatformRequest::HEADER_LANGUAGE_ID)) {
$langHeader = $request->headers->get(PlatformRequest::HEADER_LANGUAGE_ID);
if ($langHeader !== null) {
$parameters['languageId'] = $langHeader;
}
}
if ($request->headers->has(PlatformRequest::HEADER_CURRENCY_ID)) {
$currencyHeader = $request->headers->get(PlatformRequest::HEADER_CURRENCY_ID);
if ($currencyHeader !== null) {
$parameters['currencyId'] = $currencyHeader;
}
}
if ($request->headers->has(PlatformRequest::HEADER_INHERITANCE)) {
$parameters['considerInheritance'] = true;
}
return $parameters;
}
private function resolveContextSource(Request $request): ContextSource
{
if ($userId = $request->attributes->get(PlatformRequest::ATTRIBUTE_OAUTH_USER_ID)) {
$appIntegrationId = $request->headers->get(PlatformRequest::HEADER_APP_INTEGRATION_ID);
// The app integration id header is only to be used by a privileged user
if ($this->userAppIntegrationHeaderPrivileged($userId, $appIntegrationId)) {
$userId = null;
} else {
$appIntegrationId = null;
}
return $this->getAdminApiSource($userId, $appIntegrationId);
}
if (!$request->attributes->has(PlatformRequest::ATTRIBUTE_OAUTH_ACCESS_TOKEN_ID)) {
return new SystemSource();
}
$clientId = $request->attributes->get(PlatformRequest::ATTRIBUTE_OAUTH_CLIENT_ID);
$keyOrigin = AccessKeyHelper::getOrigin($clientId);
if ($keyOrigin === 'user') {
$userId = $this->getUserIdByAccessKey($clientId);
return $this->getAdminApiSource($userId);
}
if ($keyOrigin === 'integration') {
$integrationId = $this->getIntegrationIdByAccessKey($clientId);
return $this->getAdminApiSource(null, $integrationId);
}
if ($keyOrigin === 'sales-channel') {
$salesChannelId = $this->getSalesChannelIdByAccessKey($clientId);
return new SalesChannelApiSource($salesChannelId);
}
return new SystemSource();
}
/**
* @param array{languageId: string, systemFallbackLanguageId: string} $params
*
* @return non-empty-array<string>
*/
private function getLanguageIdChain(array $params): array
{
$chain = [$params['languageId']];
if ($chain[0] === Defaults::LANGUAGE_SYSTEM) {
return $chain; // no query needed
}
// `Context` ignores nulls and duplicates
$chain[] = $this->getParentLanguageId($chain[0]);
$chain[] = $params['systemFallbackLanguageId'];
/** @var non-empty-array<string> $filtered */
$filtered = array_filter($chain);
return $filtered;
}
private function getParentLanguageId(?string $languageId): ?string
{
if ($languageId === null || !Uuid::isValid($languageId)) {
throw new LanguageNotFoundException($languageId);
}
$data = $this->connection->createQueryBuilder()
->select(['LOWER(HEX(language.parent_id))'])
->from('language')
->where('language.id = :id')
->setParameter('id', Uuid::fromHexToBytes($languageId))
->executeQuery()
->fetchFirstColumn();
if (empty($data)) {
throw new LanguageNotFoundException($languageId);
}
return $data[0];
}
private function getUserIdByAccessKey(string $clientId): string
{
$id = $this->connection->createQueryBuilder()
->select(['user_id'])
->from('user_access_key')
->where('access_key = :accessKey')
->setParameter('accessKey', $clientId)
->executeQuery()
->fetchOne();
return Uuid::fromBytesToHex($id);
}
private function getSalesChannelIdByAccessKey(string $clientId): string
{
$id = $this->connection->createQueryBuilder()
->select(['id'])
->from('sales_channel')
->where('access_key = :accessKey')
->setParameter('accessKey', $clientId)
->executeQuery()
->fetchOne();
return Uuid::fromBytesToHex($id);
}
private function getIntegrationIdByAccessKey(string $clientId): string
{
$id = $this->connection->createQueryBuilder()
->select(['id'])
->from('integration')
->where('access_key = :accessKey')
->setParameter('accessKey', $clientId)
->executeQuery()
->fetchOne();
return Uuid::fromBytesToHex($id);
}
private function getAdminApiSource(?string $userId, ?string $integrationId = null): AdminApiSource
{
$source = new AdminApiSource($userId, $integrationId);
// Use the permissions associated to that app, if the request is made by an integration associated to an app
$appPermissions = $this->fetchPermissionsIntegrationByApp($integrationId);
if ($appPermissions !== null) {
$source->setIsAdmin(false);
$source->setPermissions($appPermissions);
return $source;
}
if ($userId !== null) {
$source->setPermissions($this->fetchPermissions($userId));
$source->setIsAdmin($this->isAdmin($userId));
return $source;
}
if ($integrationId !== null) {
$source->setIsAdmin($this->isAdminIntegration($integrationId));
$source->setPermissions($this->fetchIntegrationPermissions($integrationId));
return $source;
}
return $source;
}
private function isAdmin(string $userId): bool
{
return (bool) $this->connection->fetchOne(
'SELECT admin FROM `user` WHERE id = :id',
['id' => Uuid::fromHexToBytes($userId)]
);
}
private function isAdminIntegration(string $integrationId): bool
{
return (bool) $this->connection->fetchOne(
'SELECT admin FROM `integration` WHERE id = :id',
['id' => Uuid::fromHexToBytes($integrationId)]
);
}
private function fetchPermissions(string $userId): array
{
$permissions = $this->connection->createQueryBuilder()
->select(['role.privileges'])
->from('acl_user_role', 'mapping')
->innerJoin('mapping', 'acl_role', 'role', 'mapping.acl_role_id = role.id')
->where('mapping.user_id = :userId')
->setParameter('userId', Uuid::fromHexToBytes($userId))
->executeQuery()
->fetchFirstColumn();
$list = [];
foreach ($permissions as $privileges) {
$privileges = json_decode((string) $privileges, true, 512, \JSON_THROW_ON_ERROR);
$list = array_merge($list, $privileges);
}
return array_unique(array_filter($list));
}
private function getCashRounding(string $currencyId): CashRoundingConfig
{
$rounding = $this->connection->fetchAssociative(
'SELECT item_rounding FROM currency WHERE id = :id',
['id' => Uuid::fromHexToBytes($currencyId)]
);
if ($rounding === false) {
throw new \RuntimeException(sprintf('No cash rounding for currency "%s" found', $currencyId));
}
$rounding = json_decode((string) $rounding['item_rounding'], true, 512, \JSON_THROW_ON_ERROR);
return new CashRoundingConfig(
(int) $rounding['decimals'],
(float) $rounding['interval'],
(bool) $rounding['roundForNet']
);
}
private function fetchPermissionsIntegrationByApp(?string $integrationId): ?array
{
if (!$integrationId) {
return null;
}
$privileges = $this->connection->fetchOne('
SELECT `acl_role`.`privileges`
FROM `acl_role`
INNER JOIN `app` ON `app`.`acl_role_id` = `acl_role`.`id`
WHERE `app`.`integration_id` = :integrationId
', ['integrationId' => Uuid::fromHexToBytes($integrationId)]);
if ($privileges === false) {
return null;
}
return json_decode((string) $privileges, true, 512, \JSON_THROW_ON_ERROR);
}
private function fetchIntegrationPermissions(string $integrationId): array
{
$permissions = $this->connection->createQueryBuilder()
->select(['role.privileges'])
->from('integration_role', 'mapping')
->innerJoin('mapping', 'acl_role', 'role', 'mapping.acl_role_id = role.id')
->where('mapping.integration_id = :integrationId')
->setParameter('integrationId', Uuid::fromHexToBytes($integrationId))
->executeQuery()
->fetchFirstColumn();
$list = [];
foreach ($permissions as $privileges) {
$privileges = json_decode((string) $privileges, true, 512, \JSON_THROW_ON_ERROR);
$list = array_merge($list, $privileges);
}
return array_unique(array_filter($list));
}
private function fetchAppNameByIntegrationId(string $integrationId): ?string
{
$name = $this->connection->createQueryBuilder()
->select(['app.name'])
->from('app', 'app')
->innerJoin('app', 'integration', 'integration', 'integration.id = app.integration_id')
->where('integration.id = :integrationId')
->andWhere('app.active = 1')
->setParameter('integrationId', Uuid::fromHexToBytes($integrationId))
->executeQuery()
->fetchOne();
if ($name === false) {
return null;
}
return $name;
}
/**
* @throws MissingPrivilegeException
* @throws AppNotFoundException
*/
private function userAppIntegrationHeaderPrivileged(string $userId, ?string $appIntegrationId): bool
{
if ($appIntegrationId === null) {
return false;
}
$appName = $this->fetchAppNameByIntegrationId($appIntegrationId);
if ($appName === null) {
throw new AppNotFoundException($appIntegrationId);
}
$isAdmin = $this->isAdmin($userId);
if ($isAdmin) {
return true;
}
$permissions = $this->fetchPermissions($userId);
$allAppsPrivileged = \in_array('app.all', $permissions, true);
$appPrivilegeName = \sprintf('app.%s', $appName);
$specificAppPrivileged = \in_array($appPrivilegeName, $permissions, true);
if (!($specificAppPrivileged || $allAppsPrivileged)) {
throw new MissingPrivilegeException([$appPrivilegeName]);
}
return true;
}
}