vendor/shopware/core/Content/Category/SalesChannel/CachedCategoryRoute.php line 119

Open in your IDE?
  1. <?php declare(strict_types=1);
  2. namespace Shopware\Core\Content\Category\SalesChannel;
  3. use OpenApi\Annotations as OA;
  4. use Shopware\Core\Content\Category\Event\CategoryRouteCacheKeyEvent;
  5. use Shopware\Core\Content\Category\Event\CategoryRouteCacheTagsEvent;
  6. use Shopware\Core\Content\Cms\Aggregate\CmsSlot\CmsSlotEntity;
  7. use Shopware\Core\Content\Cms\SalesChannel\Struct\ProductBoxStruct;
  8. use Shopware\Core\Content\Cms\SalesChannel\Struct\ProductSliderStruct;
  9. use Shopware\Core\Framework\Adapter\Cache\AbstractCacheTracer;
  10. use Shopware\Core\Framework\Adapter\Cache\CacheValueCompressor;
  11. use Shopware\Core\Framework\DataAbstractionLayer\Cache\EntityCacheKeyGenerator;
  12. use Shopware\Core\Framework\DataAbstractionLayer\FieldSerializer\JsonFieldSerializer;
  13. use Shopware\Core\Framework\Routing\Annotation\RouteScope;
  14. use Shopware\Core\Framework\Routing\Annotation\Since;
  15. use Shopware\Core\Profiling\Profiler;
  16. use Shopware\Core\System\SalesChannel\SalesChannelContext;
  17. use Symfony\Component\HttpFoundation\Request;
  18. use Symfony\Component\Routing\Annotation\Route;
  19. use Symfony\Contracts\Cache\CacheInterface;
  20. use Symfony\Contracts\Cache\ItemInterface;
  21. use Symfony\Contracts\EventDispatcher\EventDispatcherInterface;
  22. /**
  23.  * @Route(defaults={"_routeScope"={"store-api"}})
  24.  */
  25. class CachedCategoryRoute extends AbstractCategoryRoute
  26. {
  27.     private AbstractCategoryRoute $decorated;
  28.     private CacheInterface $cache;
  29.     private EntityCacheKeyGenerator $generator;
  30.     /**
  31.      * @var AbstractCacheTracer<CategoryRouteResponse>
  32.      */
  33.     private AbstractCacheTracer $tracer;
  34.     private array $states;
  35.     private EventDispatcherInterface $dispatcher;
  36.     /**
  37.      * @param AbstractCacheTracer<CategoryRouteResponse> $tracer
  38.      */
  39.     public function __construct(
  40.         AbstractCategoryRoute $decorated,
  41.         CacheInterface $cache,
  42.         EntityCacheKeyGenerator $generator,
  43.         AbstractCacheTracer $tracer,
  44.         EventDispatcherInterface $dispatcher,
  45.         array $states
  46.     ) {
  47.         $this->decorated $decorated;
  48.         $this->cache $cache;
  49.         $this->generator $generator;
  50.         $this->tracer $tracer;
  51.         $this->states $states;
  52.         $this->dispatcher $dispatcher;
  53.     }
  54.     public static function buildName(string $id): string
  55.     {
  56.         return 'category-route-' $id;
  57.     }
  58.     public function getDecorated(): AbstractCategoryRoute
  59.     {
  60.         return $this->decorated;
  61.     }
  62.     /**
  63.      * @Since("6.2.0.0")
  64.      * @OA\Post(
  65.      *     path="/category/{categoryId}",
  66.      *     summary="Fetch a single category",
  67.      *     description="This endpoint returns information about the category, as well as a fully resolved (hydrated with mapping values) CMS page, if one is assigned to the category. You can pass slots which should be resolved exclusively.",
  68.      *     operationId="readCategory",
  69.      *     tags={"Store API", "Category"},
  70.      *     @OA\Parameter(
  71.      *         name="categoryId",
  72.      *         description="Identifier of the category to be fetched",
  73.      *         @OA\Schema(type="string", pattern="^[0-9a-f]{32}$"),
  74.      *         in="path",
  75.      *         required=true
  76.      *     ),
  77.      *     @OA\Parameter(
  78.      *         name="slots",
  79.      *         description="Resolves only the given slot identifiers. The identifiers have to be seperated by a '|' character",
  80.      *         @OA\Schema(type="string"),
  81.      *         in="query",
  82.      *     ),
  83.      *     @OA\Parameter(name="Api-Basic-Parameters"),
  84.      *     @OA\Response(
  85.      *          response="200",
  86.      *          description="The loaded category with cms page",
  87.      *          @OA\JsonContent(ref="#/components/schemas/Category")
  88.      *     )
  89.      * )
  90.      *
  91.      * @Route("/store-api/category/{navigationId}", name="store-api.category.detail", methods={"GET","POST"})
  92.      */
  93.     public function load(string $navigationIdRequest $requestSalesChannelContext $context): CategoryRouteResponse
  94.     {
  95.         return Profiler::trace('category-route', function () use ($navigationId$request$context) {
  96.             if ($context->hasState(...$this->states)) {
  97.                 return $this->getDecorated()->load($navigationId$request$context);
  98.             }
  99.             $key $this->generateKey($navigationId$request$context);
  100.             $value $this->cache->get($key, function (ItemInterface $item) use ($navigationId$request$context) {
  101.                 $name self::buildName($navigationId);
  102.                 $response $this->tracer->trace($name, function () use ($navigationId$request$context) {
  103.                     return $this->getDecorated()->load($navigationId$request$context);
  104.                 });
  105.                 $item->tag($this->generateTags($navigationId$response$request$context));
  106.                 return CacheValueCompressor::compress($response);
  107.             });
  108.             return CacheValueCompressor::uncompress($value);
  109.         });
  110.     }
  111.     private function generateKey(string $navigationIdRequest $requestSalesChannelContext $context): string
  112.     {
  113.         $parts array_merge(
  114.             $request->query->all(),
  115.             $request->request->all(),
  116.             [$this->generator->getSalesChannelContextHash($context)]
  117.         );
  118.         $event = new CategoryRouteCacheKeyEvent($navigationId$parts$request$contextnull);
  119.         $this->dispatcher->dispatch($event);
  120.         return self::buildName($navigationId) . '-' md5(JsonFieldSerializer::encodeJson($event->getParts()));
  121.     }
  122.     private function generateTags(string $navigationIdCategoryRouteResponse $responseRequest $requestSalesChannelContext $context): array
  123.     {
  124.         $tags array_merge(
  125.             $this->tracer->get(self::buildName($navigationId)),
  126.             $this->extractProductIds($response),
  127.             [self::buildName($navigationId)]
  128.         );
  129.         $event = new CategoryRouteCacheTagsEvent($navigationId$tags$request$response$contextnull);
  130.         $this->dispatcher->dispatch($event);
  131.         return array_unique(array_filter($event->getTags()));
  132.     }
  133.     private function extractProductIds(CategoryRouteResponse $response): array
  134.     {
  135.         $page $response->getCategory()->getCmsPage();
  136.         if ($page === null) {
  137.             return [];
  138.         }
  139.         $ids = [];
  140.         $slots $page->getElementsOfType('product-slider');
  141.         /** @var CmsSlotEntity $slot */
  142.         foreach ($slots as $slot) {
  143.             $slider $slot->getData();
  144.             if (!$slider instanceof ProductSliderStruct) {
  145.                 continue;
  146.             }
  147.             if ($slider->getProducts() === null) {
  148.                 continue;
  149.             }
  150.             foreach ($slider->getProducts() as $product) {
  151.                 $ids[] = $product->getId();
  152.                 $ids[] = $product->getParentId();
  153.             }
  154.         }
  155.         $slots $page->getElementsOfType('product-box');
  156.         /** @var CmsSlotEntity $slot */
  157.         foreach ($slots as $slot) {
  158.             $box $slot->getData();
  159.             if (!$box instanceof ProductBoxStruct) {
  160.                 continue;
  161.             }
  162.             if ($box->getProduct() === null) {
  163.                 continue;
  164.             }
  165.             $ids[] = $box->getProduct()->getId();
  166.             $ids[] = $box->getProduct()->getParentId();
  167.         }
  168.         $ids array_values(array_unique(array_filter($ids)));
  169.         return array_merge(
  170.             array_map([EntityCacheKeyGenerator::class, 'buildProductTag'], $ids),
  171.             [EntityCacheKeyGenerator::buildCmsTag($page->getId())]
  172.         );
  173.     }
  174. }