vendor/contao/core-bundle/src/DependencyInjection/Configuration.php line 71

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\CoreBundle\DependencyInjection;
  11. use Contao\Config;
  12. use Contao\CoreBundle\Doctrine\Backup\RetentionPolicy;
  13. use Contao\CoreBundle\Util\LocaleUtil;
  14. use Contao\Image\ResizeConfiguration;
  15. use Imagine\Image\ImageInterface;
  16. use Symfony\Component\Config\Definition\Builder\NodeDefinition;
  17. use Symfony\Component\Config\Definition\Builder\TreeBuilder;
  18. use Symfony\Component\Config\Definition\ConfigurationInterface;
  19. use Symfony\Component\Filesystem\Filesystem;
  20. use Symfony\Component\Filesystem\Path;
  21. class Configuration implements ConfigurationInterface
  22. {
  23.     private string $projectDir;
  24.     public function __construct(string $projectDir)
  25.     {
  26.         $this->projectDir $projectDir;
  27.     }
  28.     public function getConfigTreeBuilder(): TreeBuilder
  29.     {
  30.         $treeBuilder = new TreeBuilder('contao');
  31.         $treeBuilder
  32.             ->getRootNode()
  33.             ->children()
  34.                 ->scalarNode('csrf_cookie_prefix')
  35.                     ->cannotBeEmpty()
  36.                     ->defaultValue('csrf_')
  37.                 ->end()
  38.                 ->scalarNode('csrf_token_name')
  39.                     ->cannotBeEmpty()
  40.                     ->defaultValue('contao_csrf_token')
  41.                 ->end()
  42.                 ->scalarNode('encryption_key')
  43.                     ->cannotBeEmpty()
  44.                     ->defaultValue('%kernel.secret%')
  45.                 ->end()
  46.                 ->integerNode('error_level')
  47.                     ->info('The error reporting level set when the framework is initialized. Defaults to E_ALL & ~E_NOTICE & ~E_DEPRECATED & ~E_USER_DEPRECATED.')
  48.                     ->min(-1)
  49.                     ->max(32767)
  50.                     ->defaultValue(E_ALL & ~E_NOTICE & ~E_DEPRECATED & ~E_USER_DEPRECATED)
  51.                 ->end()
  52.                 ->append($this->addIntlNode())
  53.                 ->booleanNode('legacy_routing')
  54.                     ->defaultTrue()
  55.                     ->info('Disabling legacy routing allows to configure the URL prefix and suffix per root page. However, it might not be compatible with third-party extensions.')
  56.                 ->end()
  57.                 ->variableNode('localconfig')
  58.                     ->info('Allows to set TL_CONFIG variables, overriding settings stored in localconfig.php. Changes in the Contao back end will not have any effect.')
  59.                     ->validate()
  60.                         ->always(
  61.                             static function (array $options): array {
  62.                                 foreach (array_keys($options) as $option) {
  63.                                     if ($newKey Config::getNewKey($option)) {
  64.                                         trigger_deprecation('contao/core-bundle''4.12''Setting "contao.localconfig.%s" has been deprecated. Use "%s" instead.'$option$newKey);
  65.                                     }
  66.                                 }
  67.                                 return $options;
  68.                             }
  69.                         )
  70.                     ->end()
  71.                 ->end()
  72.                 ->arrayNode('locales')
  73.                     ->info('Allows to configure which languages can be used in the Contao back end. Defaults to all languages for which a translation exists.')
  74.                     ->setDeprecated('contao/core-bundle''4.12''Using contao.locales is deprecated. Please use contao.intl.enabled_locales instead.')
  75.                     ->prototype('scalar')->end()
  76.                     ->defaultValue([])
  77.                 ->end()
  78.                 ->booleanNode('prepend_locale')
  79.                     ->info('Whether or not to add the page language to the URL.')
  80.                     ->setDeprecated('contao/core-bundle''4.10''The URL prefix is configured per root page since Contao 4.10. Using this option requires legacy routing.')
  81.                     ->defaultFalse()
  82.                 ->end()
  83.                 ->booleanNode('pretty_error_screens')
  84.                     ->info('Show customizable, pretty error screens instead of the default PHP error messages.')
  85.                     ->defaultValue(false)
  86.                 ->end()
  87.                 ->scalarNode('preview_script')
  88.                     ->info('An optional entry point script that bypasses the front end cache for previewing changes (e.g. "/preview.php").')
  89.                     ->cannotBeEmpty()
  90.                     ->defaultValue('')
  91.                     ->validate()
  92.                         ->always(static fn (string $value): string => Path::canonicalize($value))
  93.                     ->end()
  94.                 ->end()
  95.                 ->scalarNode('upload_path')
  96.                     ->info('The folder used by the file manager.')
  97.                     ->cannotBeEmpty()
  98.                     ->defaultValue('files')
  99.                     ->validate()
  100.                         ->ifTrue(static fn (string $v): int => preg_match('@^(app|assets|bin|config|contao|plugins|public|share|system|templates|var|vendor|web)(/|$)@'$v))
  101.                         ->thenInvalid('%s')
  102.                     ->end()
  103.                 ->end()
  104.                 ->scalarNode('editable_files')
  105.                     ->defaultValue('css,csv,html,ini,js,json,less,md,scss,svg,svgz,ts,txt,xliff,xml,yml,yaml')
  106.                 ->end()
  107.                 ->scalarNode('url_suffix')
  108.                     ->setDeprecated('contao/core-bundle''4.10''The URL suffix is configured per root page since Contao 4.10. Using this option requires legacy routing.')
  109.                     ->defaultValue('.html')
  110.                 ->end()
  111.                 ->scalarNode('web_dir')
  112.                     ->info('Absolute path to the web directory. Defaults to %kernel.project_dir%/public.')
  113.                     ->setDeprecated('contao/core-bundle''4.13''Setting the web directory in a config file is deprecated. Use the "extra.public-dir" config key in your root composer.json instead.')
  114.                     ->cannotBeEmpty()
  115.                     ->defaultValue($this->getDefaultWebDir())
  116.                     ->validate()
  117.                         ->always(static fn (string $value): string => Path::canonicalize($value))
  118.                     ->end()
  119.                 ->end()
  120.                 ->append($this->addImageNode())
  121.                 ->append($this->addSecurityNode())
  122.                 ->append($this->addSearchNode())
  123.                 ->append($this->addCrawlNode())
  124.                 ->append($this->addMailerNode())
  125.                 ->append($this->addBackendNode())
  126.                 ->append($this->addInsertTagsNode())
  127.                 ->append($this->addBackupNode())
  128.                 ->append($this->addSanitizerNode())
  129.             ->end()
  130.         ;
  131.         return $treeBuilder;
  132.     }
  133.     private function addImageNode(): NodeDefinition
  134.     {
  135.         return (new TreeBuilder('image'))
  136.             ->getRootNode()
  137.             ->addDefaultsIfNotSet()
  138.             ->children()
  139.                 ->booleanNode('bypass_cache')
  140.                     ->info('Bypass the image cache and always regenerate images when requested. This also disables deferred image resizing.')
  141.                     ->defaultValue(false)
  142.                 ->end()
  143.                 ->arrayNode('imagine_options')
  144.                     ->addDefaultsIfNotSet()
  145.                     ->children()
  146.                         ->integerNode('jpeg_quality')
  147.                             ->defaultValue(80)
  148.                         ->end()
  149.                         ->arrayNode('jpeg_sampling_factors')
  150.                             ->prototype('scalar')->end()
  151.                             ->defaultValue([211])
  152.                         ->end()
  153.                         ->integerNode('png_compression_level')
  154.                         ->end()
  155.                         ->integerNode('png_compression_filter')
  156.                         ->end()
  157.                         ->integerNode('webp_quality')
  158.                         ->end()
  159.                         ->booleanNode('webp_lossless')
  160.                         ->end()
  161.                         ->integerNode('avif_quality')
  162.                         ->end()
  163.                         ->booleanNode('avif_lossless')
  164.                         ->end()
  165.                         ->integerNode('heic_quality')
  166.                         ->end()
  167.                         ->booleanNode('heic_lossless')
  168.                         ->end()
  169.                         ->integerNode('jxl_quality')
  170.                         ->end()
  171.                         ->booleanNode('jxl_lossless')
  172.                         ->end()
  173.                         ->booleanNode('flatten')
  174.                             ->info('Allows to disable the layer flattening of animated images. Set this option to false to support animations. It has no effect with Gd as Imagine service.')
  175.                         ->end()
  176.                         ->scalarNode('interlace')
  177.                             ->defaultValue(ImageInterface::INTERLACE_PLANE)
  178.                         ->end()
  179.                     ->end()
  180.                 ->end()
  181.                 ->scalarNode('imagine_service')
  182.                     ->info('Contao automatically uses an Imagine service out of Gmagick, Imagick and Gd (in this order). Set a service ID here to override.')
  183.                     ->defaultNull()
  184.                 ->end()
  185.                 ->booleanNode('reject_large_uploads')
  186.                     ->info('Reject uploaded images exceeding the localconfig.gdMaxImgWidth and localconfig.gdMaxImgHeight dimensions.')
  187.                     ->defaultValue(false)
  188.                 ->end()
  189.                 ->arrayNode('sizes')
  190.                     ->info('Allows to define image sizes in the configuration file in addition to in the Contao back end. Use the special name "_defaults" to preset values for all sizes of the configuration file.')
  191.                     ->useAttributeAsKey('name')
  192.                     ->validate()
  193.                         ->always(
  194.                             static function (array $value): array {
  195.                                 static $reservedImageSizeNames = [
  196.                                     ResizeConfiguration::MODE_BOX,
  197.                                     ResizeConfiguration::MODE_PROPORTIONAL,
  198.                                     ResizeConfiguration::MODE_CROP,
  199.                                     'left_top',
  200.                                     'center_top',
  201.                                     'right_top',
  202.                                     'left_center',
  203.                                     'center_center',
  204.                                     'right_center',
  205.                                     'left_bottom',
  206.                                     'center_bottom',
  207.                                     'right_bottom',
  208.                                 ];
  209.                                 foreach (array_keys($value) as $name) {
  210.                                     if (preg_match('/^\d+$/', (string) $name)) {
  211.                                         throw new \InvalidArgumentException(sprintf('The image size name "%s" cannot contain only digits'$name));
  212.                                     }
  213.                                     if (\in_array($name$reservedImageSizeNamestrue)) {
  214.                                         throw new \InvalidArgumentException(sprintf('"%s" is a reserved image size name (reserved names: %s)'$nameimplode(', '$reservedImageSizeNames)));
  215.                                     }
  216.                                     if (preg_match('/[^a-z0-9_]/', (string) $name)) {
  217.                                         throw new \InvalidArgumentException(sprintf('The image size name "%s" must consist of lowercase letters, digits and underscores only'$name));
  218.                                     }
  219.                                 }
  220.                                 return $value;
  221.                             }
  222.                         )
  223.                     ->end()
  224.                     ->arrayPrototype()
  225.                         ->children()
  226.                             ->integerNode('width')
  227.                             ->end()
  228.                             ->integerNode('height')
  229.                             ->end()
  230.                             ->enumNode('resize_mode')
  231.                                 ->values([
  232.                                     ResizeConfiguration::MODE_CROP,
  233.                                     ResizeConfiguration::MODE_BOX,
  234.                                     ResizeConfiguration::MODE_PROPORTIONAL,
  235.                                 ])
  236.                             ->end()
  237.                             ->integerNode('zoom')
  238.                                 ->min(0)
  239.                                 ->max(100)
  240.                             ->end()
  241.                             ->scalarNode('css_class')
  242.                             ->end()
  243.                             ->booleanNode('lazy_loading')
  244.                             ->end()
  245.                             ->scalarNode('densities')
  246.                             ->end()
  247.                             ->scalarNode('sizes')
  248.                             ->end()
  249.                             ->booleanNode('skip_if_dimensions_match')
  250.                                 ->info('If the output dimensions match the source dimensions, the image will not be processed. Instead, the original file will be used.')
  251.                             ->end()
  252.                             ->arrayNode('formats')
  253.                                 ->info('Allows to convert one image format to another or to provide additional image formats for an image (e.g. WebP).')
  254.                                 ->example(['jpg' => ['jxl''webp''jpg'], 'gif' => ['avif''png']])
  255.                                 ->useAttributeAsKey('source')
  256.                                 ->arrayPrototype()
  257.                                     ->beforeNormalization()->castToArray()->end()
  258.                                     ->scalarPrototype()->end()
  259.                                 ->end()
  260.                             ->end()
  261.                             ->arrayNode('items')
  262.                                 ->arrayPrototype()
  263.                                     ->children()
  264.                                         ->integerNode('width')
  265.                                         ->end()
  266.                                         ->integerNode('height')
  267.                                         ->end()
  268.                                         ->enumNode('resize_mode')
  269.                                             ->values([
  270.                                                 ResizeConfiguration::MODE_CROP,
  271.                                                 ResizeConfiguration::MODE_BOX,
  272.                                                 ResizeConfiguration::MODE_PROPORTIONAL,
  273.                                             ])
  274.                                         ->end()
  275.                                         ->integerNode('zoom')
  276.                                             ->min(0)
  277.                                             ->max(100)
  278.                                         ->end()
  279.                                         ->scalarNode('media')
  280.                                         ->end()
  281.                                         ->scalarNode('densities')
  282.                                         ->end()
  283.                                         ->scalarNode('sizes')
  284.                                         ->end()
  285.                                         ->enumNode('resizeMode')
  286.                                             ->setDeprecated('contao/core-bundle''4.9''Using contao.image.sizes.*.items.resizeMode is deprecated. Please use contao.image.sizes.*.items.resize_mode instead.')
  287.                                             ->values([
  288.                                                 ResizeConfiguration::MODE_CROP,
  289.                                                 ResizeConfiguration::MODE_BOX,
  290.                                                 ResizeConfiguration::MODE_PROPORTIONAL,
  291.                                             ])
  292.                                         ->end()
  293.                                     ->end()
  294.                                 ->end()
  295.                             ->end()
  296.                             ->enumNode('resizeMode')
  297.                                 ->setDeprecated('contao/core-bundle''4.9''Using contao.image.sizes.*.resizeMode is deprecated. Please use contao.image.sizes.*.resize_mode instead.')
  298.                                 ->values([
  299.                                     ResizeConfiguration::MODE_CROP,
  300.                                     ResizeConfiguration::MODE_BOX,
  301.                                     ResizeConfiguration::MODE_PROPORTIONAL,
  302.                                 ])
  303.                             ->end()
  304.                             ->scalarNode('cssClass')
  305.                                 ->setDeprecated('contao/core-bundle''4.9''Using contao.image.sizes.*.cssClass is deprecated. Please use contao.image.sizes.*.css_class instead.')
  306.                             ->end()
  307.                             ->booleanNode('lazyLoading')
  308.                                 ->setDeprecated('contao/core-bundle''4.9''Using contao.image.sizes.*.lazyLoading is deprecated. Please use contao.image.sizes.*.lazy_loading instead.')
  309.                             ->end()
  310.                             ->booleanNode('skipIfDimensionsMatch')
  311.                                 ->setDeprecated('contao/core-bundle''4.9''Using contao.image.sizes.*.skipIfDimensionsMatch is deprecated. Please use contao.image.sizes.*.skip_if_dimensions_match instead.')
  312.                             ->end()
  313.                         ->end()
  314.                     ->end()
  315.                 ->end()
  316.                 ->scalarNode('target_dir')
  317.                     ->info('The target directory for the cached images processed by Contao.')
  318.                     ->example('%kernel.project_dir%/assets/images')
  319.                     ->cannotBeEmpty()
  320.                     ->defaultValue(Path::join($this->projectDir'assets/images'))
  321.                     ->validate()
  322.                         ->always(static fn (string $value): string => Path::canonicalize($value))
  323.                     ->end()
  324.                 ->end()
  325.                 ->scalarNode('target_path')
  326.                     ->setDeprecated('contao/core-bundle''4.9''Use the "contao.image.target_dir" parameter instead.')
  327.                     ->defaultNull()
  328.                 ->end()
  329.                 ->arrayNode('valid_extensions')
  330.                     ->prototype('scalar')->end()
  331.                     ->defaultValue(['jpg''jpeg''gif''png''tif''tiff''bmp''svg''svgz''webp'])
  332.                 ->end()
  333.                 ->arrayNode('preview')
  334.                     ->addDefaultsIfNotSet()
  335.                     ->children()
  336.                         ->scalarNode('target_dir')
  337.                             ->info('The target directory for the cached previews.')
  338.                             ->example('%kernel.project_dir%/assets/previews')
  339.                             ->cannotBeEmpty()
  340.                             ->defaultValue(Path::join($this->projectDir'assets/previews'))
  341.                             ->validate()
  342.                                 ->always(static fn (string $value): string => Path::canonicalize($value))
  343.                             ->end()
  344.                         ->end()
  345.                         ->integerNode('default_size')
  346.                             ->min(1)
  347.                             ->max(65535)
  348.                             ->defaultValue(512)
  349.                         ->end()
  350.                         ->integerNode('max_size')
  351.                             ->min(1)
  352.                             ->max(65535)
  353.                             ->defaultValue(1024)
  354.                         ->end()
  355.                         ->booleanNode('enable_fallback_images')
  356.                             ->info('Whether or not to generate previews for unsupported file types that show a file icon containing the file type.')
  357.                             ->defaultValue(true)
  358.                         ->end()
  359.                     ->end()
  360.                     ->validate()
  361.                         ->ifTrue(static fn (array $v) => $v['default_size'] > $v['max_size'])
  362.                         ->thenInvalid('The default_size must not be greater than the max_size: %s')
  363.                     ->end()
  364.                 ->end()
  365.             ->end()
  366.         ;
  367.     }
  368.     private function addIntlNode(): NodeDefinition
  369.     {
  370.         return (new TreeBuilder('intl'))
  371.             ->getRootNode()
  372.             ->addDefaultsIfNotSet()
  373.             ->children()
  374.                 ->arrayNode('locales')
  375.                     ->info('Adds, removes or overwrites the list of ICU locale IDs. Defaults to all locale IDs known to the system.')
  376.                     ->prototype('scalar')->end()
  377.                     ->defaultValue([])
  378.                     ->example(['+de''-de_AT''gsw_CH'])
  379.                     ->validate()
  380.                         ->ifTrue(
  381.                             static function (array $locales): bool {
  382.                                 foreach ($locales as $locale) {
  383.                                     if (!preg_match('/^[+-]?[a-z]{2}/'$locale)) {
  384.                                         return true;
  385.                                     }
  386.                                     $locale ltrim($locale'+-');
  387.                                     if (LocaleUtil::canonicalize($locale) !== $locale) {
  388.                                         return true;
  389.                                     }
  390.                                 }
  391.                                 return false;
  392.                             }
  393.                         )
  394.                         ->thenInvalid('All provided locales must be in the canonicalized ICU form and optionally start with +/- to add/remove the locale to/from the default list.')
  395.                     ->end()
  396.                 ->end()
  397.                 ->arrayNode('enabled_locales')
  398.                     ->info('Adds, removes or overwrites the list of enabled locale IDs that can be used in the Backend for example. Defaults to all languages for which a translation exists.')
  399.                     ->prototype('scalar')->end()
  400.                     ->defaultValue([])
  401.                     ->example(['+de''-de_AT''gsw_CH'])
  402.                     ->validate()
  403.                         ->ifTrue(
  404.                             static function (array $locales): bool {
  405.                                 foreach ($locales as $locale) {
  406.                                     if (!preg_match('/^[+-]?[a-z]{2}/'$locale)) {
  407.                                         return true;
  408.                                     }
  409.                                     $locale ltrim($locale'+-');
  410.                                     if (LocaleUtil::canonicalize($locale) !== $locale) {
  411.                                         return true;
  412.                                     }
  413.                                 }
  414.                                 return false;
  415.                             }
  416.                         )
  417.                         ->thenInvalid('All provided locales must be in the canonicalized ICU form and optionally start with +/- to add/remove the locale to/from the default list.')
  418.                     ->end()
  419.                 ->end()
  420.                 ->arrayNode('countries')
  421.                     ->info('Adds, removes or overwrites the list of ISO 3166-1 alpha-2 country codes.')
  422.                     ->prototype('scalar')->end()
  423.                     ->defaultValue([])
  424.                     ->example(['+DE''-AT''CH'])
  425.                     ->validate()
  426.                         ->ifTrue(
  427.                             static function (array $countries): bool {
  428.                                 foreach ($countries as $country) {
  429.                                     if (!preg_match('/^[+-]?[A-Z][A-Z0-9]$/'$country)) {
  430.                                         return true;
  431.                                     }
  432.                                 }
  433.                                 return false;
  434.                             }
  435.                         )
  436.                         ->thenInvalid('All provided countries must be two uppercase letters and optionally start with +/- to add/remove the country to/from the default list.')
  437.                     ->end()
  438.                 ->end()
  439.             ->end()
  440.         ;
  441.     }
  442.     private function addSecurityNode(): NodeDefinition
  443.     {
  444.         return (new TreeBuilder('security'))
  445.             ->getRootNode()
  446.             ->addDefaultsIfNotSet()
  447.             ->children()
  448.                 ->arrayNode('two_factor')
  449.                     ->addDefaultsIfNotSet()
  450.                     ->children()
  451.                         ->booleanNode('enforce_backend')
  452.                             ->defaultValue(false)
  453.                         ->end()
  454.                     ->end()
  455.                 ->end()
  456.             ->end()
  457.         ;
  458.     }
  459.     private function addSearchNode(): NodeDefinition
  460.     {
  461.         return (new TreeBuilder('search'))
  462.             ->getRootNode()
  463.             ->addDefaultsIfNotSet()
  464.             ->children()
  465.                 ->arrayNode('default_indexer')
  466.                     ->info('The default search indexer, which indexes pages in the database.')
  467.                     ->addDefaultsIfNotSet()
  468.                     ->children()
  469.                         ->booleanNode('enable')
  470.                             ->defaultTrue()
  471.                         ->end()
  472.                     ->end()
  473.                 ->end()
  474.                 ->booleanNode('index_protected')
  475.                     ->info('Enables indexing of protected pages.')
  476.                     ->defaultFalse()
  477.                 ->end()
  478.                 ->arrayNode('listener')
  479.                     ->info('The search index listener can index valid and delete invalid responses upon every request. You may limit it to one of the features or disable it completely.')
  480.                     ->addDefaultsIfNotSet()
  481.                     ->children()
  482.                         ->booleanNode('index')
  483.                             ->info('Enables indexing successful responses.')
  484.                             ->defaultTrue()
  485.                         ->end()
  486.                         ->booleanNode('delete')
  487.                             ->info('Enables deleting unsuccessful responses from the index.')
  488.                             ->defaultTrue()
  489.                         ->end()
  490.                     ->end()
  491.                 ->end()
  492.             ->end()
  493.         ;
  494.     }
  495.     private function addCrawlNode(): NodeDefinition
  496.     {
  497.         return (new TreeBuilder('crawl'))
  498.             ->getRootNode()
  499.             ->addDefaultsIfNotSet()
  500.             ->children()
  501.                 ->arrayNode('additional_uris')
  502.                     ->info('Additional URIs to crawl. By default, only the ones defined in the root pages are crawled.')
  503.                     ->validate()
  504.                     ->ifTrue(
  505.                         static function (array $uris): bool {
  506.                             foreach ($uris as $uri) {
  507.                                 if (!preg_match('@^https?://@'$uri)) {
  508.                                     return true;
  509.                                 }
  510.                             }
  511.                             return false;
  512.                         }
  513.                     )
  514.                     ->thenInvalid('All provided additional URIs must start with either http:// or https://.')
  515.                     ->end()
  516.                     ->prototype('scalar')->end()
  517.                     ->defaultValue([])
  518.                 ->end()
  519.                 ->arrayNode('default_http_client_options')
  520.                     ->info('Allows to configure the default HttpClient options (useful for proxy settings, SSL certificate validation and more).')
  521.                     ->prototype('variable')->end()
  522.                     ->defaultValue([])
  523.                 ->end()
  524.             ->end()
  525.         ;
  526.     }
  527.     private function addMailerNode(): NodeDefinition
  528.     {
  529.         return (new TreeBuilder('mailer'))
  530.             ->getRootNode()
  531.             ->addDefaultsIfNotSet()
  532.             ->children()
  533.                 ->arrayNode('transports')
  534.                     ->info('Specifies the mailer transports available for selection within Contao.')
  535.                     ->useAttributeAsKey('name')
  536.                     ->arrayPrototype()
  537.                         ->children()
  538.                             ->scalarNode('from')
  539.                                 ->info('Overrides the "From" address for any e-mails sent with this mailer transport.')
  540.                                 ->defaultNull()
  541.                             ->end()
  542.                         ->end()
  543.                     ->end()
  544.                 ->end()
  545.             ->end()
  546.         ;
  547.     }
  548.     private function addBackendNode(): NodeDefinition
  549.     {
  550.         return (new TreeBuilder('backend'))
  551.             ->getRootNode()
  552.             ->addDefaultsIfNotSet()
  553.             ->children()
  554.                 ->arrayNode('attributes')
  555.                     ->info('Adds HTML attributes to the <body> tag in the back end.')
  556.                     ->example(['app-name' => 'My App''app-version' => '1.2.3'])
  557.                     ->validate()
  558.                     ->always(
  559.                         static function (array $attributes): array {
  560.                             foreach (array_keys($attributes) as $name) {
  561.                                 if (preg_match('/[^a-z0-9\-.:_]/', (string) $name)) {
  562.                                     throw new \InvalidArgumentException(sprintf('The attribute name "%s" must be a valid HTML attribute name.'$name));
  563.                                 }
  564.                             }
  565.                             return $attributes;
  566.                         }
  567.                     )
  568.                     ->end()
  569.                     ->normalizeKeys(false)
  570.                     ->useAttributeAsKey('name')
  571.                     ->scalarPrototype()->end()
  572.                 ->end()
  573.                 ->arrayNode('custom_css')
  574.                     ->info('Adds custom style sheets to the back end.')
  575.                     ->example(['files/backend/custom.css'])
  576.                     ->cannotBeEmpty()
  577.                     ->scalarPrototype()->end()
  578.                 ->end()
  579.                 ->arrayNode('custom_js')
  580.                     ->info('Adds custom JavaScript files to the back end.')
  581.                     ->example(['files/backend/custom.js'])
  582.                     ->cannotBeEmpty()
  583.                     ->scalarPrototype()->end()
  584.                 ->end()
  585.                 ->scalarNode('badge_title')
  586.                     ->info('Configures the title of the badge in the back end.')
  587.                     ->example('develop')
  588.                     ->cannotBeEmpty()
  589.                     ->defaultValue('')
  590.                 ->end()
  591.                 ->scalarNode('route_prefix')
  592.                     ->info('Defines the path of the Contao backend.')
  593.                     ->validate()
  594.                         ->ifTrue(static fn (string $prefix) => !== preg_match('/^\/\S*[^\/]$/'$prefix))
  595.                         ->thenInvalid('The backend path must begin but not end with a slash. Invalid path configured: %s')
  596.                     ->end()
  597.                     ->example('/admin')
  598.                     ->defaultValue('/contao')
  599.                 ->end()
  600.             ->end()
  601.         ;
  602.     }
  603.     private function addInsertTagsNode(): NodeDefinition
  604.     {
  605.         return (new TreeBuilder('insert_tags'))
  606.             ->getRootNode()
  607.             ->addDefaultsIfNotSet()
  608.             ->children()
  609.                 ->arrayNode('allowed_tags')
  610.                     ->info('A list of allowed insert tags.')
  611.                     ->example(['*_url''request_token'])
  612.                     ->scalarPrototype()->end()
  613.                     ->defaultValue(['*'])
  614.                 ->end()
  615.             ->end()
  616.         ;
  617.     }
  618.     private function addBackupNode(): NodeDefinition
  619.     {
  620.         return (new TreeBuilder('backup'))
  621.             ->getRootNode()
  622.             ->addDefaultsIfNotSet()
  623.             ->children()
  624.                 ->arrayNode('ignore_tables')
  625.                     ->info('These tables are ignored by default when creating and restoring backups.')
  626.                     ->defaultValue(['tl_crawl_queue''tl_log''tl_search''tl_search_index''tl_search_term'])
  627.                     ->scalarPrototype()->end()
  628.                 ->end()
  629.                 ->integerNode('keep_max')
  630.                     ->info('The maximum number of backups to keep. Use 0 to keep all the backups forever.')
  631.                     ->defaultValue(5)
  632.                 ->end()
  633.                 ->arrayNode('keep_intervals')
  634.                     ->info('The latest backup plus the oldest of every configured interval will be kept. Intervals have to be specified as documented in https://www.php.net/manual/en/dateinterval.construct.php without the P prefix.')
  635.                     ->defaultValue(['1D''7D''14D''1M'])
  636.                     ->validate()
  637.                         ->ifTrue(
  638.                             static function (array $intervals) {
  639.                                 try {
  640.                                     RetentionPolicy::validateAndSortIntervals($intervals);
  641.                                 } catch (\Exception $e) {
  642.                                     return true;
  643.                                 }
  644.                                 return false;
  645.                             }
  646.                         )
  647.                     ->thenInvalid('%s')
  648.                     ->end()
  649.                     ->scalarPrototype()->end()
  650.                 ->end()
  651.             ->end()
  652.         ;
  653.     }
  654.     private function addSanitizerNode(): NodeDefinition
  655.     {
  656.         return (new TreeBuilder('sanitizer'))
  657.             ->getRootNode()
  658.             ->addDefaultsIfNotSet()
  659.             ->children()
  660.                 ->arrayNode('allowed_url_protocols')
  661.                     ->prototype('scalar')->end()
  662.                     ->defaultValue(['http''https''ftp''mailto''tel''data''skype''whatsapp'])
  663.                     ->validate()
  664.                         ->always(
  665.                             static function (array $protocols): array {
  666.                                 foreach ($protocols as $protocol) {
  667.                                     if (!preg_match('/^[a-z][a-z0-9\-+.]*$/i', (string) $protocol)) {
  668.                                         throw new \InvalidArgumentException(sprintf('The protocol name "%s" must be a valid URI scheme.'$protocol));
  669.                                     }
  670.                                 }
  671.                                 return $protocols;
  672.                             }
  673.                         )
  674.                     ->end()
  675.                 ->end()
  676.             ->end()
  677.         ;
  678.     }
  679.     private function getDefaultWebDir(): string
  680.     {
  681.         $webDir Path::join($this->projectDir'web');
  682.         if ((new Filesystem())->exists($webDir)) {
  683.             return $webDir;
  684.         }
  685.         return Path::join($this->projectDir'public');
  686.     }
  687. }