vendor/pimcore/pimcore/lib/Kernel.php line 280

Open in your IDE?
  1. <?php
  2. /**
  3.  * Pimcore
  4.  *
  5.  * This source file is available under two different licenses:
  6.  * - GNU General Public License version 3 (GPLv3)
  7.  * - Pimcore Commercial License (PCL)
  8.  * Full copyright and license information is available in
  9.  * LICENSE.md which is distributed with this source code.
  10.  *
  11.  *  @copyright  Copyright (c) Pimcore GmbH (http://www.pimcore.org)
  12.  *  @license    http://www.pimcore.org/license     GPLv3 and PCL
  13.  */
  14. namespace Pimcore;
  15. use Doctrine\Bundle\DoctrineBundle\DoctrineBundle;
  16. use Doctrine\Bundle\MigrationsBundle\DoctrineMigrationsBundle;
  17. use FOS\JsRoutingBundle\FOSJsRoutingBundle;
  18. use League\FlysystemBundle\FlysystemBundle;
  19. use Pimcore\Bundle\AdminBundle\PimcoreAdminBundle;
  20. use Pimcore\Bundle\CoreBundle\PimcoreCoreBundle;
  21. use Pimcore\Cache\Runtime;
  22. use Pimcore\Config\BundleConfigLocator;
  23. use Pimcore\Event\SystemEvents;
  24. use Pimcore\Extension\Bundle\Config\StateConfig;
  25. use Pimcore\HttpKernel\BundleCollection\BundleCollection;
  26. use Pimcore\HttpKernel\BundleCollection\ItemInterface;
  27. use Pimcore\HttpKernel\BundleCollection\LazyLoadedItem;
  28. use Presta\SitemapBundle\PrestaSitemapBundle;
  29. use Scheb\TwoFactorBundle\SchebTwoFactorBundle;
  30. use Sensio\Bundle\FrameworkExtraBundle\SensioFrameworkExtraBundle;
  31. use Symfony\Bundle\DebugBundle\DebugBundle;
  32. use Symfony\Bundle\FrameworkBundle\FrameworkBundle;
  33. use Symfony\Bundle\FrameworkBundle\Kernel\MicroKernelTrait;
  34. use Symfony\Bundle\MonologBundle\MonologBundle;
  35. use Symfony\Bundle\SecurityBundle\SecurityBundle;
  36. use Symfony\Bundle\TwigBundle\TwigBundle;
  37. use Symfony\Bundle\WebProfilerBundle\WebProfilerBundle;
  38. use Symfony\Cmf\Bundle\RoutingBundle\CmfRoutingBundle;
  39. use Symfony\Component\Config\Loader\LoaderInterface;
  40. use Symfony\Component\Config\Resource\FileExistenceResource;
  41. use Symfony\Component\Config\Resource\FileResource;
  42. use Symfony\Component\DependencyInjection\ContainerBuilder;
  43. use Symfony\Component\DependencyInjection\ContainerInterface;
  44. use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
  45. use Symfony\Component\EventDispatcher\GenericEvent;
  46. use Symfony\Component\HttpKernel\Bundle\BundleInterface;
  47. use Symfony\Component\HttpKernel\Kernel as SymfonyKernel;
  48. use Symfony\Component\Routing\Loader\Configurator\RoutingConfigurator;
  49. abstract class Kernel extends SymfonyKernel
  50. {
  51.     use MicroKernelTrait {
  52.         registerContainerConfiguration as microKernelRegisterContainerConfiguration;
  53.         registerBundles as microKernelRegisterBundles;
  54.     }
  55.     /**
  56.      * @var Extension\Config
  57.      */
  58.     protected $extensionConfig;
  59.     /**
  60.      * @var BundleCollection
  61.      */
  62.     private $bundleCollection;
  63.     /**
  64.      * {@inheritdoc}
  65.      */
  66.     public function getRootDir()
  67.     {
  68.         return PIMCORE_PROJECT_ROOT;
  69.     }
  70.     /**
  71.      * {@inheritdoc}
  72.      */
  73.     public function getProjectDir()
  74.     {
  75.         return PIMCORE_PROJECT_ROOT;
  76.     }
  77.     /**
  78.      * {@inheritdoc}
  79.      */
  80.     public function getCacheDir()
  81.     {
  82.         if (isset($_SERVER['APP_CACHE_DIR'])) {
  83.             return $_SERVER['APP_CACHE_DIR'].'/'.$this->environment;
  84.         }
  85.         return PIMCORE_SYMFONY_CACHE_DIRECTORY '/' $this->environment;
  86.     }
  87.     /**
  88.      * {@inheritdoc}
  89.      */
  90.     public function getLogDir()
  91.     {
  92.         return PIMCORE_LOG_DIRECTORY;
  93.     }
  94.     /**
  95.      * {@inheritdoc}
  96.      */
  97.     protected function configureContainer(ContainerConfigurator $container): void
  98.     {
  99.         $projectDir realpath($this->getProjectDir());
  100.         $container->import($projectDir '/config/{packages}/*.yaml');
  101.         $container->import($projectDir '/config/{packages}/'.$this->environment.'/*.yaml');
  102.         if (is_file($projectDir '/config/services.yaml')) {
  103.             $container->import($projectDir '/config/services.yaml');
  104.             $container->import($projectDir '/config/{services}_'.$this->environment.'.yaml');
  105.         } elseif (is_file($path $projectDir '/config/services.php')) {
  106.             (require $path)($container->withPath($path), $this);
  107.         }
  108.     }
  109.     /**
  110.      * {@inheritdoc}
  111.      */
  112.     protected function configureRoutes(RoutingConfigurator $routes): void
  113.     {
  114.         $projectDir realpath($this->getProjectDir());
  115.         $routes->import($projectDir '/config/{routes}/'.$this->environment.'/*.yaml');
  116.         $routes->import($projectDir '/config/{routes}/*.yaml');
  117.         if (is_file($projectDir '/config/routes.yaml')) {
  118.             $routes->import($projectDir '/config/routes.yaml');
  119.         } elseif (is_file($path $projectDir '/config/routes.php')) {
  120.             (require $path)($routes->withPath($path), $this);
  121.         }
  122.     }
  123.     /**
  124.      * {@inheritdoc}
  125.      */
  126.     public function registerContainerConfiguration(LoaderInterface $loader)
  127.     {
  128.         $loader->load(function (ContainerBuilder $container) {
  129.             $this->registerExtensionConfigFileResources($container);
  130.         });
  131.         $bundleConfigLocator = new BundleConfigLocator($this);
  132.         foreach ($bundleConfigLocator->locate('config') as $bundleConfig) {
  133.             $loader->load($bundleConfig);
  134.         }
  135.         $this->microKernelRegisterContainerConfiguration($loader);
  136.         //load system configuration
  137.         $systemConfigFile Config::locateConfigFile('system.yml');
  138.         if (file_exists($systemConfigFile)) {
  139.             $loader->load($systemConfigFile);
  140.         }
  141.         foreach (['image-thumbnails''video-thumbnails''custom-reports''document-types''web-to-print''predefined-properties''predefined-asset-metadata''staticroutes''perspectives''custom-views'] as $configDir) {
  142.             $configDir PIMCORE_CONFIGURATION_DIRECTORY "/$configDir/";
  143.             if (is_dir($configDir)) {
  144.                 // @phpstan-ignore-next-line
  145.                 $loader->import($configDir);
  146.             }
  147.         }
  148.     }
  149.     private function registerExtensionConfigFileResources(ContainerBuilder $container)
  150.     {
  151.         $filenames = [
  152.             'extensions.php',
  153.             sprintf('extensions_%s.php'$this->getEnvironment()),
  154.         ];
  155.         $directories = [
  156.             PIMCORE_CUSTOM_CONFIGURATION_DIRECTORY,
  157.             PIMCORE_CONFIGURATION_DIRECTORY,
  158.         ];
  159.         // add possible extensions.php files as file existence resources (only for the current env)
  160.         foreach ($directories as $directory) {
  161.             foreach ($filenames as $filename) {
  162.                 $container->addResource(new FileExistenceResource($directory '/' $filename));
  163.             }
  164.         }
  165.         // add extensions.php as container resource
  166.         if ($this->extensionConfig->configFileExists()) {
  167.             $container->addResource(new FileResource($this->extensionConfig->locateConfigFile()));
  168.         }
  169.     }
  170.     /**
  171.      * {@inheritdoc}
  172.      */
  173.     public function boot()
  174.     {
  175.         if (true === $this->booted) {
  176.             // make sure container reset is handled properly
  177.             parent::boot();
  178.             return;
  179.         }
  180.         // handle system requirements
  181.         $this->setSystemRequirements();
  182.         // initialize extension manager config
  183.         $this->extensionConfig = new Extension\Config();
  184.         parent::boot();
  185.     }
  186.     /**
  187.      * {@inheritdoc}
  188.      */
  189.     public function shutdown()
  190.     {
  191.         if (true === $this->booted) {
  192.             // cleanup runtime cache, doctrine, monolog ... to free some memory and avoid locking issues
  193.             $this->container->get(\Pimcore\Helper\LongRunningHelper::class)->cleanUp();
  194.         }
  195.         parent::shutdown();
  196.     }
  197.     /**
  198.      * {@inheritdoc}
  199.      */
  200.     protected function initializeContainer()
  201.     {
  202.         parent::initializeContainer();
  203.         // initialize runtime cache (defined as synthetic service)
  204.         Runtime::getInstance();
  205.         // set the extension config on the container
  206.         $this->getContainer()->set(Extension\Config::class, $this->extensionConfig);
  207.         \Pimcore::initLogger();
  208.         \Pimcore\Cache::init();
  209.         // on pimcore shutdown
  210.         register_shutdown_function(function () {
  211.             // check if container still exists at this point as it could already
  212.             // be cleared (e.g. when running tests which boot multiple containers)
  213.             try {
  214.                 $container $this->getContainer();
  215.             } catch (\LogicException) {
  216.                 // Container is cleared. Allow tests to finish.
  217.             }
  218.             if (isset($container) && $container instanceof ContainerInterface) {
  219.                 $container->get('event_dispatcher')->dispatch(new GenericEvent(), SystemEvents::SHUTDOWN);
  220.             }
  221.             \Pimcore::shutdown();
  222.         });
  223.     }
  224.     /**
  225.      * Returns an array of bundles to register.
  226.      *
  227.      * @return BundleInterface[] An array of bundle instances
  228.      */
  229.     public function registerBundles(): array
  230.     {
  231.         $collection $this->createBundleCollection();
  232.         if (is_file($this->getProjectDir().'/config/bundles.php')) {
  233.             $flexBundles = [];
  234.             array_push($flexBundles, ...$this->microKernelRegisterBundles());
  235.             $collection->addBundles($flexBundles);
  236.         }
  237.         // core bundles (Symfony, Pimcore)
  238.         $this->registerCoreBundlesToCollection($collection);
  239.         // custom bundles
  240.         $this->registerBundlesToCollection($collection);
  241.         // bundles registered in extensions.php
  242.         $this->registerExtensionManagerBundles($collection);
  243.         $bundles $collection->getBundles($this->getEnvironment());
  244.         $this->bundleCollection $collection;
  245.         return $bundles;
  246.     }
  247.     /**
  248.      * Creates bundle collection. Use this method to set bundles on the collection
  249.      * early.
  250.      *
  251.      * @return BundleCollection
  252.      */
  253.     protected function createBundleCollection(): BundleCollection
  254.     {
  255.         return new BundleCollection();
  256.     }
  257.     /**
  258.      * Returns the bundle collection which was used to build the set of used bundles
  259.      *
  260.      * @return BundleCollection
  261.      */
  262.     public function getBundleCollection(): BundleCollection
  263.     {
  264.         return $this->bundleCollection;
  265.     }
  266.     /**
  267.      * Registers "core" bundles
  268.      *
  269.      * @param BundleCollection $collection
  270.      */
  271.     protected function registerCoreBundlesToCollection(BundleCollection $collection)
  272.     {
  273.         $collection->addBundles([
  274.             // symfony "core"/standard
  275.             new FrameworkBundle(),
  276.             new SecurityBundle(),
  277.             new TwigBundle(),
  278.             new MonologBundle(),
  279.             new DoctrineBundle(),
  280.             new DoctrineMigrationsBundle(),
  281.             new SensioFrameworkExtraBundle(),
  282.             new CmfRoutingBundle(),
  283.             new PrestaSitemapBundle(),
  284.             new SchebTwoFactorBundle(),
  285.             new FOSJsRoutingBundle(),
  286.             new FlysystemBundle(),
  287.         ], 100);
  288.         // pimcore bundles
  289.         $collection->addBundles([
  290.             new PimcoreCoreBundle(),
  291.             new PimcoreAdminBundle(),
  292.         ], 60);
  293.         // load development bundles only in matching environments
  294.         if (in_array($this->getEnvironment(), $this->getEnvironmentsForDevBundles(), true)) {
  295.             $collection->addBundles([
  296.                 new DebugBundle(),
  297.                 new WebProfilerBundle(),
  298.             ], 80);
  299.         }
  300.     }
  301.     protected function getEnvironmentsForDevBundles(): array
  302.     {
  303.         return ['dev''test'];
  304.     }
  305.     /**
  306.      * Registers bundles enabled via extension manager
  307.      *
  308.      * @param BundleCollection $collection
  309.      */
  310.     protected function registerExtensionManagerBundles(BundleCollection $collection)
  311.     {
  312.         $stateConfig = new StateConfig($this->extensionConfig);
  313.         foreach ($stateConfig->getEnabledBundles() as $className => $options) {
  314.             if (!class_exists($className)) {
  315.                 continue;
  316.             }
  317.             // do not register bundles twice - skip if it was already loaded manually
  318.             if ($collection->hasItem($className)) {
  319.                 continue;
  320.             }
  321.             // use lazy loaded item to instantiate the bundle only if environment matches
  322.             $collection->add(new LazyLoadedItem(
  323.                 $className,
  324.                 $options['priority'],
  325.                 $options['environments'],
  326.                 ItemInterface::SOURCE_EXTENSION_MANAGER_CONFIG
  327.             ));
  328.         }
  329.     }
  330.     /**
  331.      * Adds bundles to register to the bundle collection. The collection is able
  332.      * to handle priorities and environment specific bundles.
  333.      *
  334.      * To be implemented in child classes
  335.      *
  336.      * @param BundleCollection $collection
  337.      */
  338.     public function registerBundlesToCollection(BundleCollection $collection)
  339.     {
  340.     }
  341.     /**
  342.      * Handle system settings and requirements
  343.      */
  344.     protected function setSystemRequirements()
  345.     {
  346.         // try to set system-internal variables
  347.         $maxExecutionTime 240;
  348.         if (php_sapi_name() === 'cli') {
  349.             $maxExecutionTime 0;
  350.         }
  351.         //@ini_set("memory_limit", "1024M");
  352.         @ini_set('max_execution_time'$maxExecutionTime);
  353.         @set_time_limit($maxExecutionTime);
  354.         ini_set('default_charset''UTF-8');
  355.         // set internal character encoding to UTF-8
  356.         mb_internal_encoding('UTF-8');
  357.         // this is for simple_dom_html
  358.         ini_set('pcre.recursion-limit'100000);
  359.         // zlib.output_compression conflicts with while (@ob_end_flush()) ;
  360.         // see also: https://github.com/pimcore/pimcore/issues/291
  361.         if (ini_get('zlib.output_compression')) {
  362.             @ini_set('zlib.output_compression''Off');
  363.         }
  364.         // set dummy timezone if no tz is specified / required for example by the logger, ...
  365.         $defaultTimezone = @date_default_timezone_get();
  366.         if (!$defaultTimezone) {
  367.             date_default_timezone_set('UTC'); // UTC -> default timezone
  368.         }
  369.     }
  370. }