local/modules/Front/Controller/FeedController.php line 58

  1. <?php
  2. /*
  3.  * This file is part of the Thelia package.
  4.  * http://www.thelia.net
  5.  *
  6.  * (c) OpenStudio <info@thelia.net>
  7.  *
  8.  * For the full copyright and license information, please view the LICENSE
  9.  * file that was distributed with this source code.
  10.  */
  11. namespace Front\Controller;
  12. use Symfony\Component\Cache\Adapter\AdapterInterface;
  13. use Thelia\Controller\Front\BaseFrontController;
  14. use Thelia\Core\HttpFoundation\Request;
  15. use Thelia\Core\HttpFoundation\Response;
  16. use Thelia\Model\BrandQuery;
  17. use Thelia\Model\CategoryQuery;
  18. use Thelia\Model\ConfigQuery;
  19. use Thelia\Model\FolderQuery;
  20. use Thelia\Model\Lang;
  21. use Thelia\Model\LangQuery;
  22. /**
  23.  * Controller uses to generate RSS Feeds.
  24.  *
  25.  * A default cache of 2 hours is used to avoid attack. You can flush cache if you have `ADMIN` role and pass flush=1 in
  26.  * query string parameter.
  27.  *
  28.  * @author Julien Chanséaume <jchanseaume@openstudio.fr>
  29.  */
  30. class FeedController extends BaseFrontController
  31. {
  32.     /**
  33.      * Folder name for feeds cache.
  34.      */
  35.     public const FEED_CACHE_DIR 'feeds';
  36.     /**
  37.      * Key prefix for feed cache.
  38.      */
  39.     public const FEED_CACHE_KEY 'feed';
  40.     /**
  41.      * render the RSS feed.
  42.      *
  43.      * @param $context string   The context of the feed : catalog, content. default: catalog
  44.      * @param $lang string      The lang of the feed : fr_FR, en_US, ... default: default language of the site
  45.      * @param $id string        The id of the parent element. The id of the main parent category for catalog context.
  46.      *                          The id of the content folder for content context
  47.      *
  48.      * @throws \RuntimeException
  49.      *
  50.      * @return Response
  51.      */
  52.     public function generateAction($context$lang$id)
  53.     {
  54.         /** @var Request $request */
  55.         $request $this->getRequest();
  56.         // context
  57.         if ('' === $context) {
  58.             $context 'catalog';
  59.         } elseif (!\in_array($context, ['catalog''content''brand'])) {
  60.             $this->pageNotFound();
  61.         }
  62.         // the locale : fr_FR, en_US,
  63.         if ('' !== $lang) {
  64.             if (!$this->checkLang($lang)) {
  65.                 $this->pageNotFound();
  66.             }
  67.         } else {
  68.             try {
  69.                 $lang Lang::getDefaultLanguage();
  70.                 $lang $lang->getLocale();
  71.             } catch (\RuntimeException $ex) {
  72.                 // @todo generate error page
  73.                 throw new \RuntimeException('No default language is defined. Please define one.');
  74.             }
  75.         }
  76.         if (null === $lang LangQuery::create()->findOneByLocale($lang)) {
  77.             $this->pageNotFound();
  78.         }
  79.         $lang $lang->getId();
  80.         // check if element exists and is visible
  81.         if ('' !== $id) {
  82.             if (false === $this->checkId($context$id)) {
  83.                 $this->pageNotFound();
  84.             }
  85.         }
  86.         $flush $request->query->get('flush''');
  87.         /** @var AdapterInterface $cacheAdapter */
  88.         $cacheAdapter $this->container->get('thelia.cache');
  89.         $cacheKey self::FEED_CACHE_KEY.$lang.$context;
  90.         $cacheItem $cacheAdapter->getItem($cacheKey);
  91.         if (!$cacheItem->isHit() || $flush) {
  92.             $cacheExpire = (int) (ConfigQuery::read('feed_ttl''7200')) ?: 7200;
  93.             // render the view
  94.             $cacheContent $this->renderRaw(
  95.                 'feed',
  96.                 [
  97.                     '_context_' => $context,
  98.                     '_lang_' => $lang,
  99.                     '_id_' => $id,
  100.                 ]
  101.             );
  102.             $cacheItem->expiresAfter($cacheExpire);
  103.             $cacheItem->set($cacheContent);
  104.             $cacheAdapter->save($cacheItem);
  105.         }
  106.         $response = new Response();
  107.         $response->setContent($cacheItem->get());
  108.         $response->headers->set('Content-Type''application/rss+xml');
  109.         return $response;
  110.     }
  111.     /**
  112.      * get the cache directory for feeds.
  113.      *
  114.      * @return mixed|string
  115.      */
  116.     protected function getCacheDir()
  117.     {
  118.         $cacheDir $this->container->getParameter('kernel.cache_dir');
  119.         $cacheDir rtrim($cacheDir'/');
  120.         $cacheDir .= '/'.self::FEED_CACHE_DIR.'/';
  121.         return $cacheDir;
  122.     }
  123.     /**
  124.      * Check if current user has ADMIN role.
  125.      *
  126.      * @return bool
  127.      */
  128.     protected function checkAdmin()
  129.     {
  130.         return $this->getSecurityContext()->hasAdminUser();
  131.     }
  132.     /**
  133.      * Check if a lang is used.
  134.      *
  135.      * @param $lang string  The lang code. e.g.: fr
  136.      *
  137.      * @return bool true if the language is used, otherwise false
  138.      */
  139.     private function checkLang($lang)
  140.     {
  141.         // load locals
  142.         $lang LangQuery::create()
  143.             ->findOneByLocale($lang);
  144.         return null !== $lang;
  145.     }
  146.     /**
  147.      * Check if the element exists and is visible.
  148.      *
  149.      * @param $context string   catalog or content
  150.      * @param $id string        id of the element
  151.      *
  152.      * @return bool
  153.      */
  154.     private function checkId($context$id)
  155.     {
  156.         $ret false;
  157.         if (is_numeric($id)) {
  158.             if ('catalog' === $context) {
  159.                 $cat CategoryQuery::create()->findPk($id);
  160.                 $ret = (null !== $cat && $cat->getVisible());
  161.             } elseif ('brand' === $context) {
  162.                 $brand BrandQuery::create()->findPk($id);
  163.                 $ret = (null !== $brand && $brand->getVisible());
  164.             } else {
  165.                 $folder FolderQuery::create()->findPk($id);
  166.                 $ret = (null !== $folder && $folder->getVisible());
  167.             }
  168.         }
  169.         return $ret;
  170.     }
  171. }