vendor/contao/manager-bundle/src/ContaoManager/Plugin.php line 235

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. /*
  4.  * This file is part of Contao.
  5.  *
  6.  * (c) Leo Feyer
  7.  *
  8.  * @license LGPL-3.0-or-later
  9.  */
  10. namespace Contao\ManagerBundle\ContaoManager;
  11. use Contao\CoreBundle\ContaoCoreBundle;
  12. use Contao\ManagerBundle\ContaoManager\ApiCommand\GenerateJwtCookieCommand;
  13. use Contao\ManagerBundle\ContaoManager\ApiCommand\GetConfigCommand;
  14. use Contao\ManagerBundle\ContaoManager\ApiCommand\GetDotEnvCommand;
  15. use Contao\ManagerBundle\ContaoManager\ApiCommand\ParseJwtCookieCommand;
  16. use Contao\ManagerBundle\ContaoManager\ApiCommand\RemoveDotEnvCommand;
  17. use Contao\ManagerBundle\ContaoManager\ApiCommand\SetConfigCommand;
  18. use Contao\ManagerBundle\ContaoManager\ApiCommand\SetDotEnvCommand;
  19. use Contao\ManagerBundle\ContaoManagerBundle;
  20. use Contao\ManagerPlugin\Api\ApiPluginInterface;
  21. use Contao\ManagerPlugin\Bundle\BundlePluginInterface;
  22. use Contao\ManagerPlugin\Bundle\Config\BundleConfig;
  23. use Contao\ManagerPlugin\Bundle\Parser\ParserInterface;
  24. use Contao\ManagerPlugin\Config\ConfigPluginInterface;
  25. use Contao\ManagerPlugin\Config\ContainerBuilder as PluginContainerBuilder;
  26. use Contao\ManagerPlugin\Config\ExtensionPluginInterface;
  27. use Contao\ManagerPlugin\Dependency\DependentPluginInterface;
  28. use Contao\ManagerPlugin\Routing\RoutingPluginInterface;
  29. use Doctrine\Bundle\DoctrineBundle\DoctrineBundle;
  30. use FOS\HttpCacheBundle\FOSHttpCacheBundle;
  31. use League\FlysystemBundle\FlysystemBundle;
  32. use Nelmio\CorsBundle\NelmioCorsBundle;
  33. use Nelmio\SecurityBundle\NelmioSecurityBundle;
  34. use Symfony\Bundle\DebugBundle\DebugBundle;
  35. use Symfony\Bundle\FrameworkBundle\FrameworkBundle;
  36. use Symfony\Bundle\MonologBundle\MonologBundle;
  37. use Symfony\Bundle\SecurityBundle\SecurityBundle;
  38. use Symfony\Bundle\TwigBundle\TwigBundle;
  39. use Symfony\Bundle\WebProfilerBundle\WebProfilerBundle;
  40. use Symfony\Component\Config\Loader\LoaderInterface;
  41. use Symfony\Component\Config\Loader\LoaderResolverInterface;
  42. use Symfony\Component\DependencyInjection\ContainerBuilder;
  43. use Symfony\Component\Filesystem\Path;
  44. use Symfony\Component\Finder\Finder;
  45. use Symfony\Component\Finder\SplFileInfo;
  46. use Symfony\Component\HttpKernel\KernelInterface;
  47. use Symfony\Component\Mailer\Transport\NativeTransportFactory;
  48. use Symfony\Component\Routing\Route;
  49. use Symfony\Component\Routing\RouteCollection;
  50. use Twig\Extra\TwigExtraBundle\TwigExtraBundle;
  51. /**
  52.  * @internal
  53.  */
  54. class Plugin implements BundlePluginInterfaceConfigPluginInterfaceRoutingPluginInterfaceExtensionPluginInterfaceDependentPluginInterfaceApiPluginInterface
  55. {
  56.     private static ?string $autoloadModules null;
  57.     /**
  58.      * Sets the path to enable autoloading of legacy Contao modules.
  59.      */
  60.     public static function autoloadModules(string $modulePath): void
  61.     {
  62.         static::$autoloadModules $modulePath;
  63.     }
  64.     public function getPackageDependencies(): array
  65.     {
  66.         return ['contao/core-bundle'];
  67.     }
  68.     public function getBundles(ParserInterface $parser): array
  69.     {
  70.         $configs = [
  71.             BundleConfig::create(FrameworkBundle::class),
  72.             BundleConfig::create(SecurityBundle::class)->setLoadAfter([FrameworkBundle::class]),
  73.             BundleConfig::create(TwigBundle::class),
  74.             BundleConfig::create(TwigExtraBundle::class),
  75.             BundleConfig::create(MonologBundle::class),
  76.             BundleConfig::create(DoctrineBundle::class),
  77.             BundleConfig::create(NelmioCorsBundle::class),
  78.             BundleConfig::create(NelmioSecurityBundle::class),
  79.             BundleConfig::create(FOSHttpCacheBundle::class),
  80.             BundleConfig::create(ContaoManagerBundle::class)->setLoadAfter([ContaoCoreBundle::class]),
  81.             BundleConfig::create(DebugBundle::class)->setLoadInProduction(false),
  82.             BundleConfig::create(WebProfilerBundle::class)->setLoadInProduction(false),
  83.             BundleConfig::create(FlysystemBundle::class)->setLoadAfter([ContaoCoreBundle::class]),
  84.         ];
  85.         // Autoload the legacy modules
  86.         if (null !== static::$autoloadModules && file_exists(static::$autoloadModules)) {
  87.             /** @var array<SplFileInfo> $modules */
  88.             $modules Finder::create()
  89.                 ->directories()
  90.                 ->depth(0)
  91.                 ->in(static::$autoloadModules)
  92.             ;
  93.             $iniConfigs = [];
  94.             foreach ($modules as $module) {
  95.                 if (!file_exists(Path::join($module->getPathname(), '.skip'))) {
  96.                     $iniConfigs[] = $parser->parse($module->getFilename(), 'ini');
  97.                 }
  98.             }
  99.             if (!empty($iniConfigs)) {
  100.                 $configs array_merge($configs, ...$iniConfigs);
  101.             }
  102.         }
  103.         return $configs;
  104.     }
  105.     public function registerContainerConfiguration(LoaderInterface $loader, array $managerConfig): void
  106.     {
  107.         $loader->load(
  108.             static function (ContainerBuilder $container) use ($loader): void {
  109.                 if ('dev' === $container->getParameter('kernel.environment')) {
  110.                     $loader->load('@ContaoManagerBundle/Resources/skeleton/config/config_dev.yml');
  111.                 } else {
  112.                     $loader->load('@ContaoManagerBundle/Resources/skeleton/config/config_prod.yml');
  113.                 }
  114.                 $container->setParameter('container.dumper.inline_class_loader'true);
  115.             }
  116.         );
  117.     }
  118.     public function getRouteCollection(LoaderResolverInterface $resolverKernelInterface $kernel): ?RouteCollection
  119.     {
  120.         if ('dev' !== $kernel->getEnvironment()) {
  121.             return null;
  122.         }
  123.         $collections = [];
  124.         $files = [
  125.             '_wdt' => '@WebProfilerBundle/Resources/config/routing/wdt.xml',
  126.             '_profiler' => '@WebProfilerBundle/Resources/config/routing/profiler.xml',
  127.         ];
  128.         foreach ($files as $prefix => $file) {
  129.             /** @var RouteCollection $collection */
  130.             $collection $resolver->resolve($file)->load($file);
  131.             $collection->addPrefix($prefix);
  132.             $collections[] = $collection;
  133.         }
  134.         $collection array_reduce(
  135.             $collections,
  136.             static function (RouteCollection $carryRouteCollection $item): RouteCollection {
  137.                 $carry->addCollection($item);
  138.                 return $carry;
  139.             },
  140.             new RouteCollection()
  141.         );
  142.         // Redirect the deprecated install.php file
  143.         $collection->add(
  144.             'contao_install_redirect',
  145.             new Route(
  146.                 '/install.php',
  147.                 [
  148.                     '_scope' => 'backend',
  149.                     '_controller' => 'Symfony\Bundle\FrameworkBundle\Controller\RedirectController::redirectAction',
  150.                     'route' => 'contao_install',
  151.                     'permanent' => true,
  152.                 ]
  153.             )
  154.         );
  155.         return $collection;
  156.     }
  157.     public function getApiFeatures(): array
  158.     {
  159.         return [
  160.             'dot-env' => [
  161.                 'APP_SECRET',
  162.                 'APP_ENV',
  163.                 'COOKIE_WHITELIST',
  164.                 'DATABASE_URL',
  165.                 'DISABLE_HTTP_CACHE',
  166.                 'MAILER_URL',
  167.                 'MAILER_DSN',
  168.                 'TRACE_LEVEL',
  169.                 'TRUSTED_PROXIES',
  170.                 'TRUSTED_HOSTS',
  171.             ],
  172.             'config' => [
  173.                 'disable-packages',
  174.             ],
  175.             'jwt-cookie' => [
  176.                 'debug',
  177.             ],
  178.         ];
  179.     }
  180.     public function getApiCommands(): array
  181.     {
  182.         return [
  183.             GetConfigCommand::class,
  184.             SetConfigCommand::class,
  185.             GetDotEnvCommand::class,
  186.             SetDotEnvCommand::class,
  187.             RemoveDotEnvCommand::class,
  188.             GenerateJwtCookieCommand::class,
  189.             ParseJwtCookieCommand::class,
  190.         ];
  191.     }
  192.     public function getExtensionConfig($extensionName, array $extensionConfigsPluginContainerBuilder $container): array
  193.     {
  194.         switch ($extensionName) {
  195.             case 'contao':
  196.                 return $this->handlePrependLocale($extensionConfigs$container);
  197.             case 'framework':
  198.                 $extensionConfigs $this->checkMailerTransport($extensionConfigs$container);
  199.                 $extensionConfigs $this->addDefaultMailer($extensionConfigs$container);
  200.                 if (!isset($_SERVER['APP_SECRET'])) {
  201.                     $container->setParameter('env(APP_SECRET)'$container->getParameter('secret'));
  202.                 }
  203.                 if (!isset($_SERVER['MAILER_DSN'])) {
  204.                     if (isset($_SERVER['MAILER_URL'])) {
  205.                         trigger_deprecation('contao/manager-bundle''4.13''Using the "MAILER_URL" environment variable has been deprecated and will no longer work in Contao 5.0. Use the "MAILER_DSN" environment variable instead.');
  206.                         $container->setParameter('env(MAILER_DSN)'$this->getMailerDsnFromMailerUrl($_SERVER['MAILER_URL']));
  207.                     } else {
  208.                         $container->setParameter('env(MAILER_DSN)'$this->getMailerDsn($container));
  209.                     }
  210.                 }
  211.                 return $extensionConfigs;
  212.             case 'doctrine':
  213.                 if (!isset($_SERVER['DATABASE_URL'])) {
  214.                     $container->setParameter('env(DATABASE_URL)'$this->getDatabaseUrl($container$extensionConfigs));
  215.                 }
  216.                 $extensionConfigs $this->addDefaultPdoDriverOptions($extensionConfigs$container);
  217.                 $extensionConfigs $this->addDefaultDoctrineMapping($extensionConfigs$container);
  218.                 $extensionConfigs $this->enableStrictMode($extensionConfigs$container);
  219.                 $extensionConfigs $this->setDefaultCollation($extensionConfigs);
  220.                 return $extensionConfigs;
  221.             case 'nelmio_security':
  222.                 return $this->checkClickjackingPaths($extensionConfigs);
  223.         }
  224.         return $extensionConfigs;
  225.     }
  226.     /**
  227.      * Adds backwards compatibility for the %prepend_locale% parameter.
  228.      *
  229.      * @return array<string,array<string,mixed>>
  230.      */
  231.     private function handlePrependLocale(array $extensionConfigsContainerBuilder $container): array
  232.     {
  233.         if (!$container->hasParameter('prepend_locale')) {
  234.             return $extensionConfigs;
  235.         }
  236.         foreach ($extensionConfigs as $extensionConfig) {
  237.             if (isset($extensionConfig['prepend_locale'])) {
  238.                 return $extensionConfigs;
  239.             }
  240.         }
  241.         trigger_deprecation('contao/manager-bundle''4.6''Defining the "prepend_locale" parameter in the parameters.yml file has been deprecated and will no longer work in Contao 5.0. Define the "contao.prepend_locale" parameter in the config.yml file instead.');
  242.         $extensionConfigs[] = [
  243.             'prepend_locale' => '%prepend_locale%',
  244.         ];
  245.         return $extensionConfigs;
  246.     }
  247.     /**
  248.      * Sets the PDO driver options if applicable (#2459).
  249.      *
  250.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  251.      */
  252.     private function addDefaultPdoDriverOptions(array $extensionConfigsContainerBuilder $container): array
  253.     {
  254.         // Do not add PDO options if the constant does not exist
  255.         if (!\defined('PDO::MYSQL_ATTR_MULTI_STATEMENTS')) {
  256.             return $extensionConfigs;
  257.         }
  258.         [$driver$options] = $this->parseDbalDriverAndOptions($extensionConfigs$container);
  259.         // Do not add PDO options if custom options have been defined
  260.         if (isset($options[\PDO::MYSQL_ATTR_MULTI_STATEMENTS])) {
  261.             return $extensionConfigs;
  262.         }
  263.         // Do not add PDO options if the selected driver is not mysql
  264.         if (null !== $driver && 'mysql' !== $driver) {
  265.             return $extensionConfigs;
  266.         }
  267.         $extensionConfigs[] = [
  268.             'dbal' => [
  269.                 'connections' => [
  270.                     'default' => [
  271.                         'options' => [
  272.                             \PDO::MYSQL_ATTR_MULTI_STATEMENTS => false,
  273.                         ],
  274.                     ],
  275.                 ],
  276.             ],
  277.         ];
  278.         return $extensionConfigs;
  279.     }
  280.     /**
  281.      * Adds a default ORM mapping for the App namespace if none is configured.
  282.      *
  283.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  284.      */
  285.     private function addDefaultDoctrineMapping(array $extensionConfigsContainerBuilder $container): array
  286.     {
  287.         $defaultEntityManager 'default';
  288.         foreach ($extensionConfigs as $config) {
  289.             if (null !== $em $config['orm']['default_entity_manager'] ?? null) {
  290.                 $defaultEntityManager $em;
  291.             }
  292.         }
  293.         $mappings = [];
  294.         $autoMappingEnabled false;
  295.         foreach ($extensionConfigs as $config) {
  296.             $mappings[] = $config['orm']['mappings'] ?? [];
  297.             foreach ($config['orm']['entity_managers'] ?? [] as $em) {
  298.                 $mappings[] = $em['mappings'] ?? [];
  299.             }
  300.             $autoMappingEnabled |= ($config['orm']['auto_mapping'] ?? false)
  301.                 || ($config['orm']['entity_managers'][$defaultEntityManager]['auto_mapping'] ?? false);
  302.         }
  303.         // Skip if auto mapping is not enabled for the default entity manager.
  304.         if (!$autoMappingEnabled) {
  305.             return $extensionConfigs;
  306.         }
  307.         // Skip if a mapping with the name or alias "App" already exists or any
  308.         // mapping already targets "%kernel.project_dir%/src/Entity".
  309.         foreach (array_replace(...$mappings) as $name => $values) {
  310.             if (
  311.                 'App' === $name
  312.                 || 'App' === ($values['alias'] ?? '')
  313.                 || '%kernel.project_dir%/src/Entity' === ($values['dir'] ?? '')
  314.             ) {
  315.                 return $extensionConfigs;
  316.             }
  317.         }
  318.         // Skip if the "%kernel.project_dir%/src/Entity" directory does not exist.
  319.         if (!$container->fileExists(Path::join($container->getParameter('kernel.project_dir'), 'src/Entity'))) {
  320.             return $extensionConfigs;
  321.         }
  322.         $extensionConfigs[] = [
  323.             'orm' => [
  324.                 'entity_managers' => [
  325.                     $defaultEntityManager => [
  326.                         'mappings' => [
  327.                             'App' => [
  328.                                 'dir' => '%kernel.project_dir%/src/Entity',
  329.                                 'is_bundle' => false,
  330.                                 'prefix' => 'App\Entity',
  331.                                 'alias' => 'App',
  332.                             ],
  333.                         ],
  334.                     ],
  335.                 ],
  336.             ],
  337.         ];
  338.         return $extensionConfigs;
  339.     }
  340.     /**
  341.      * Enables the SQL strict mode for PDO and MySQL drivers.
  342.      *
  343.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  344.      */
  345.     private function enableStrictMode(array $extensionConfigsContainerBuilder $container): array
  346.     {
  347.         [$driver$options] = $this->parseDbalDriverAndOptions($extensionConfigs$container);
  348.         // Skip if driver is not supported
  349.         if (null === ($key = ['mysql' => 1002'mysqli' => 3][$driver] ?? null)) {
  350.             return $extensionConfigs;
  351.         }
  352.         // Skip if init command is already configured
  353.         if (isset($options[$key])) {
  354.             return $extensionConfigs;
  355.         }
  356.         // Enable strict mode
  357.         $extensionConfigs[] = [
  358.             'dbal' => [
  359.                 'connections' => [
  360.                     'default' => [
  361.                         'options' => [
  362.                             $key => "SET SESSION sql_mode=CONCAT(@@sql_mode, IF(INSTR(@@sql_mode, 'STRICT_'), '', ',TRADITIONAL'))",
  363.                         ],
  364.                     ],
  365.                 ],
  366.             ],
  367.         ];
  368.         return $extensionConfigs;
  369.     }
  370.     /**
  371.      * Sets the "collate" and "collation" options to the same value (see #4798).
  372.      *
  373.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  374.      */
  375.     private function setDefaultCollation(array $extensionConfigs): array
  376.     {
  377.         $defaultCollation null;
  378.         foreach ($extensionConfigs as $config) {
  379.             $collation $config['dbal']['connections']['default']['default_table_options']['collation'] ?? $config['dbal']['connections']['default']['default_table_options']['collate'] ?? null;
  380.             if (null !== $collation) {
  381.                 $defaultCollation $collation;
  382.             }
  383.         }
  384.         if (null !== $defaultCollation) {
  385.             $extensionConfigs[] = [
  386.                 'dbal' => [
  387.                     'connections' => [
  388.                         'default' => [
  389.                             'default_table_options' => [
  390.                                 'collate' => $defaultCollation,
  391.                                 'collation' => $defaultCollation,
  392.                             ],
  393.                         ],
  394.                     ],
  395.                 ],
  396.             ];
  397.         }
  398.         return $extensionConfigs;
  399.     }
  400.     /**
  401.      * Changes the mail transport from "mail" to "sendmail".
  402.      *
  403.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  404.      */
  405.     private function checkMailerTransport(array $extensionConfigsContainerBuilder $container): array
  406.     {
  407.         if ('mail' === $container->getParameter('mailer_transport')) {
  408.             $container->setParameter('mailer_transport''sendmail');
  409.         }
  410.         return $extensionConfigs;
  411.     }
  412.     /**
  413.      * Dynamically adds a default mailer to the config, if no mailer is defined.
  414.      *
  415.      * We cannot add a default mailer configuration to the skeleton config.yml,
  416.      * since different types of configurations are not allowed.
  417.      *
  418.      * For example, if the Manager Bundle defined
  419.      *
  420.      *     framework:
  421.      *         mailer:
  422.      *             dsn: '%env(MAILER_DSN)%'
  423.      *
  424.      * in the skeleton config.yml and the user adds
  425.      *
  426.      *     framework:
  427.      *         mailer:
  428.      *             transports:
  429.      *                 foobar: 'smtps://smtp.example.com'
  430.      *
  431.      * to their config.yml, the merged configuration will lead to an error, since
  432.      * you cannot use "framework.mailer.dsn" together with "framework.mailer.transports".
  433.      * Thus, the default mailer configuration needs to be added dynamically if
  434.      * not already present.
  435.      *
  436.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  437.      */
  438.     private function addDefaultMailer(array $extensionConfigsContainerBuilder $container): array
  439.     {
  440.         foreach ($extensionConfigs as $config) {
  441.             if (isset($config['mailer']) && (isset($config['mailer']['transports']) || isset($config['mailer']['dsn']))) {
  442.                 return $extensionConfigs;
  443.             }
  444.         }
  445.         $extensionConfigs[] = [
  446.             'mailer' => [
  447.                 'dsn' => '%env(MAILER_DSN)%',
  448.             ],
  449.         ];
  450.         return $extensionConfigs;
  451.     }
  452.     /**
  453.      * @return array{0: string|null, 1: array<string, mixed>}
  454.      */
  455.     private function parseDbalDriverAndOptions(array $extensionConfigsContainerBuilder $container): array
  456.     {
  457.         $driver null;
  458.         $url null;
  459.         $options = [];
  460.         foreach ($extensionConfigs as $config) {
  461.             if (null !== ($driverConfig $config['dbal']['connections']['default']['driver'] ?? null)) {
  462.                 $driver $driverConfig;
  463.             }
  464.             if (null !== ($urlConfig $config['dbal']['connections']['default']['url'] ?? null)) {
  465.                 $url $container->resolveEnvPlaceholders($urlConfigtrue);
  466.             }
  467.             if (null !== ($optionsConfig $config['dbal']['connections']['default']['options'] ?? null)) {
  468.                 $options[] = $optionsConfig;
  469.             }
  470.         }
  471.         // If URL is set, it overrides the driver option
  472.         if (!empty($url)) {
  473.             $driver str_replace('-''_'parse_url($urlPHP_URL_SCHEME));
  474.         }
  475.         // Normalize the driver name
  476.         if (\in_array($driver, ['pdo_mysql''mysql2'], true)) {
  477.             $driver 'mysql';
  478.         }
  479.         return [$driverarray_replace([], ...$options)];
  480.     }
  481.     /**
  482.      * Adds a clickjacking configuration for "^/.*" if not already defined.
  483.      *
  484.      * @return array<string,array<string,array<string,array<string,mixed>>>>
  485.      */
  486.     private function checkClickjackingPaths(array $extensionConfigs): array
  487.     {
  488.         foreach ($extensionConfigs as $extensionConfig) {
  489.             if (isset($extensionConfig['clickjacking']['paths']['^/.*'])) {
  490.                 return $extensionConfigs;
  491.             }
  492.         }
  493.         $extensionConfigs[] = [
  494.             'clickjacking' => [
  495.                 'paths' => [
  496.                     '^/.*' => 'SAMEORIGIN',
  497.                 ],
  498.             ],
  499.         ];
  500.         return $extensionConfigs;
  501.     }
  502.     private function getDatabaseUrl(ContainerBuilder $container, array $extensionConfigs): string
  503.     {
  504.         $driver 'mysql';
  505.         foreach ($extensionConfigs as $extensionConfig) {
  506.             // Loop over all configs so the last one wins
  507.             $driver $extensionConfig['dbal']['connections']['default']['driver'] ?? $driver;
  508.         }
  509.         $userPassword '';
  510.         if ($user $container->getParameter('database_user')) {
  511.             $userPassword $this->encodeUrlParameter((string) $user);
  512.             if ($password $container->getParameter('database_password')) {
  513.                 $userPassword .= ':'.$this->encodeUrlParameter((string) $password);
  514.             }
  515.             $userPassword .= '@';
  516.         }
  517.         $dbName '';
  518.         if ($name $container->getParameter('database_name')) {
  519.             $dbName .= '/'.$this->encodeUrlParameter((string) $name);
  520.         }
  521.         if ($container->hasParameter('database_version') && $version $container->getParameter('database_version')) {
  522.             $dbName .= '?serverVersion='.$this->encodeUrlParameter((string) $version);
  523.         }
  524.         return sprintf(
  525.             '%s://%s%s:%s%s',
  526.             str_replace('_''-'$driver),
  527.             $userPassword,
  528.             $container->getParameter('database_host'),
  529.             (int) $container->getParameter('database_port'),
  530.             $dbName
  531.         );
  532.     }
  533.     private function getMailerDsnFromMailerUrl(string $mailerUrl): string
  534.     {
  535.         if (false === $parts parse_url($mailerUrl)) {
  536.             throw new \InvalidArgumentException(sprintf('The MAILER_URL "%s" is not valid.'$mailerUrl));
  537.         }
  538.         $options = [
  539.             'transport' => null,
  540.             'username' => null,
  541.             'password' => null,
  542.             'host' => null,
  543.             'port' => null,
  544.             'encryption' => null,
  545.         ];
  546.         $queryOptions = [];
  547.         if (isset($parts['scheme'])) {
  548.             $options['transport'] = $parts['scheme'];
  549.         }
  550.         if (isset($parts['user'])) {
  551.             $options['username'] = rawurldecode($parts['user']);
  552.         }
  553.         if (isset($parts['pass'])) {
  554.             $options['password'] = rawurldecode($parts['pass']);
  555.         }
  556.         if (isset($parts['host'])) {
  557.             $options['host'] = rawurldecode($parts['host']);
  558.         }
  559.         if (isset($parts['port'])) {
  560.             $options['port'] = $parts['port'];
  561.         }
  562.         if (isset($parts['query'])) {
  563.             parse_str($parts['query'], $query);
  564.             foreach ($query as $key => $value) {
  565.                 if (empty($key)) {
  566.                     continue;
  567.                 }
  568.                 if (\array_key_exists($key$options)) {
  569.                     $options[$key] = $value;
  570.                 } else {
  571.                     $queryOptions[$key] = $value;
  572.                 }
  573.             }
  574.         }
  575.         if (empty($options['transport'])) {
  576.             throw new \InvalidArgumentException(sprintf('The MAILER_URL "%s" is not valid.'$mailerUrl));
  577.         }
  578.         if (\in_array($options['transport'], ['mail''sendmail'], true)) {
  579.             return 'sendmail://default';
  580.         }
  581.         /*
  582.          * Check for gmail transport.
  583.          *
  584.          * With Swiftmailer a DSN like "gmail://username:password@localhost" was
  585.          * supported out-of-the-box. See https://symfony.com/doc/4.4/email.html#using-gmail-to-send-emails
  586.          * Symfony Mailer supports something similar, but only with an additional
  587.          * dependency. See https://symfony.com/doc/4.4/components/mailer.html#transport
  588.          *
  589.          * Thus we add backwards compatibility for the "gmail" transport here.
  590.          */
  591.         if ('gmail' === $options['transport']) {
  592.             $options['host'] = 'smtp.gmail.com';
  593.             $options['transport'] = 'smtps';
  594.         }
  595.         if (empty($options['host']) || !\in_array($options['transport'], ['smtp''smtps'], true)) {
  596.             throw new \InvalidArgumentException(sprintf('The MAILER_URL "%s" is not valid.'$mailerUrl));
  597.         }
  598.         $transport $options['transport'];
  599.         $credentials '';
  600.         $port '';
  601.         if (!empty($options['encryption']) && 'ssl' === $options['encryption']) {
  602.             $transport 'smtps';
  603.         }
  604.         if (!empty($options['username'])) {
  605.             $credentials .= $this->encodeUrlParameter((string) $options['username']);
  606.             if (!empty($options['password'])) {
  607.                 $credentials .= ':'.$this->encodeUrlParameter((string) $options['password']);
  608.             }
  609.             $credentials .= '@';
  610.         }
  611.         if (!empty($options['port'])) {
  612.             $port ':'.$options['port'];
  613.         }
  614.         return sprintf(
  615.             '%s://%s%s%s%s',
  616.             $transport,
  617.             $credentials,
  618.             $options['host'],
  619.             $port,
  620.             !empty($queryOptions) ? '?'.http_build_query($queryOptions) : ''
  621.         );
  622.     }
  623.     private function getMailerDsn(ContainerBuilder $container): string
  624.     {
  625.         if (!$container->hasParameter('mailer_transport') || 'sendmail' === $container->getParameter('mailer_transport')) {
  626.             return class_exists(NativeTransportFactory::class) ? 'native://default' 'sendmail://default';
  627.         }
  628.         $transport 'smtp';
  629.         $credentials '';
  630.         $portSuffix '';
  631.         if (($encryption $container->getParameter('mailer_encryption')) && 'ssl' === $encryption) {
  632.             $transport 'smtps';
  633.         }
  634.         if ($user $container->getParameter('mailer_user')) {
  635.             $credentials .= $this->encodeUrlParameter((string) $user);
  636.             if ($password $container->getParameter('mailer_password')) {
  637.                 $credentials .= ':'.$this->encodeUrlParameter((string) $password);
  638.             }
  639.             $credentials .= '@';
  640.         }
  641.         if ($port $container->getParameter('mailer_port')) {
  642.             $portSuffix ':'.$port;
  643.         }
  644.         return sprintf(
  645.             '%s://%s%s%s',
  646.             $transport,
  647.             $credentials,
  648.             $container->getParameter('mailer_host'),
  649.             $portSuffix
  650.         );
  651.     }
  652.     private function encodeUrlParameter(string $parameter): string
  653.     {
  654.         return str_replace('%''%%'rawurlencode($parameter));
  655.     }
  656. }