vendor/pimcore/pimcore/bundles/CoreBundle/DependencyInjection/Configuration.php line 1525

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\Bundle\CoreBundle\DependencyInjection;
  15. use Pimcore\Bundle\CoreBundle\DependencyInjection\Config\Processor\PlaceholderProcessor;
  16. use Pimcore\Targeting\Storage\CookieStorage;
  17. use Pimcore\Targeting\Storage\TargetingStorageInterface;
  18. use Pimcore\Workflow\EventSubscriber\ChangePublishedStateSubscriber;
  19. use Pimcore\Workflow\EventSubscriber\NotificationSubscriber;
  20. use Pimcore\Workflow\Notification\NotificationEmailService;
  21. use Symfony\Component\Config\Definition\Builder\ArrayNodeDefinition;
  22. use Symfony\Component\Config\Definition\Builder\NodeDefinition;
  23. use Symfony\Component\Config\Definition\Builder\TreeBuilder;
  24. use Symfony\Component\Config\Definition\ConfigurationInterface;
  25. /**
  26.  * @internal
  27.  */
  28. final class Configuration implements ConfigurationInterface
  29. {
  30.     /**
  31.      * @var PlaceholderProcessor
  32.      */
  33.     private $placeholderProcessor;
  34.     private $placeholders = [];
  35.     public function __construct()
  36.     {
  37.         $this->placeholderProcessor = new PlaceholderProcessor();
  38.         $this->placeholders = [];
  39.     }
  40.     /**
  41.      * Generates the configuration tree builder.
  42.      *
  43.      * @return \Symfony\Component\Config\Definition\Builder\TreeBuilder The tree builder
  44.      */
  45.     public function getConfigTreeBuilder()
  46.     {
  47.         $treeBuilder = new TreeBuilder('pimcore');
  48.         /** @var ArrayNodeDefinition $rootNode */
  49.         $rootNode $treeBuilder->getRootNode();
  50.         $rootNode->addDefaultsIfNotSet();
  51.         $rootNode
  52.             ->children()
  53.                 ->arrayNode('error_handling')
  54.                     ->addDefaultsIfNotSet()
  55.                     ->children()
  56.                         ->booleanNode('render_error_document')
  57.                             ->info('Render error document in case of an error instead of showing Symfony\'s error page')
  58.                             ->defaultTrue()
  59.                             ->beforeNormalization()
  60.                                 ->ifString()
  61.                                 ->then(function ($v) {
  62.                                     return (bool)$v;
  63.                                 })
  64.                             ->end()
  65.                         ->end()
  66.                     ->end()
  67.                     ->setDeprecated(
  68.                         'pimcore/pimcore',
  69.                         '10.1',
  70.                         'The "%node%" option is deprecated since Pimcore 10.1, it will be removed in Pimcore 11.'
  71.                     )
  72.                 ->end()
  73.                 ->arrayNode('bundles')
  74.                     ->addDefaultsIfNotSet()
  75.                     ->children()
  76.                         ->arrayNode('search_paths')
  77.                             ->prototype('scalar')->end()
  78.                         ->end()
  79.                         ->booleanNode('handle_composer')
  80.                             ->defaultTrue()
  81.                         ->end()
  82.                     ->end()
  83.                 ->end()
  84.                 ->arrayNode('flags')
  85.                     ->info('Generic map for feature flags')
  86.                     ->prototype('scalar')->end()
  87.                 ->end()
  88.                 ->arrayNode('translations')
  89.                     ->addDefaultsIfNotSet()
  90.                     ->children()
  91.                         ->arrayNode('admin_translation_mapping')
  92.                             ->useAttributeAsKey('locale')
  93.                             ->prototype('scalar')->end()
  94.                         ->end()
  95.                         ->arrayNode('debugging')
  96.                             ->info('If debugging is enabled, the translator will return the plain translation key instead of the translated message.')
  97.                             ->addDefaultsIfNotSet()
  98.                             ->canBeDisabled()
  99.                             ->children()
  100.                                 ->scalarNode('parameter')
  101.                                     ->defaultValue('pimcore_debug_translations')
  102.                                 ->end()
  103.                             ->end()
  104.                         ->end()
  105.                         ->arrayNode('data_object')
  106.                             ->addDefaultsIfNotSet()
  107.                             ->children()
  108.                                 ->arrayNode('translation_extractor')
  109.                                     ->children()
  110.                                         ->arrayNode('attributes')
  111.                                             ->info('Can be used to restrict the extracted localized fields (e.g. used by XLIFF exporter in the Pimcore backend)')
  112.                                             ->prototype('array')
  113.                                                 ->prototype('scalar')->end()
  114.                                             ->end()
  115.                                             ->example(
  116.                                                 [
  117.                                                     'Product' => ['name''description'],
  118.                                                     'Brand' => ['name'],
  119.                                                 ]
  120.                                             )
  121.                                         ->end()
  122.                                     ->end()
  123.                                 ->end()
  124.                             ->end()
  125.                         ->end()
  126.                     ->end()
  127.                 ->end()
  128.                 ->arrayNode('maps')
  129.                     ->addDefaultsIfNotSet()
  130.                     ->children()
  131.                         ->scalarNode('tile_layer_url_template')
  132.                             ->defaultValue('https://a.tile.openstreetmap.org/{z}/{x}/{y}.png')
  133.                         ->end()
  134.                         ->scalarNode('geocoding_url_template')
  135.                             ->defaultValue('https://nominatim.openstreetmap.org/search?q={q}&addressdetails=1&format=json&limit=1')
  136.                         ->end()
  137.                         ->scalarNode('reverse_geocoding_url_template')
  138.                             ->defaultValue('https://nominatim.openstreetmap.org/reverse?format=json&lat={lat}&lon={lon}&addressdetails=1')
  139.                         ->end()
  140.                     ->end()
  141.                 ->end()
  142.             ->end();
  143.         $this->addGeneralNode($rootNode);
  144.         $this->addMaintenanceNode($rootNode);
  145.         $this->addServicesNode($rootNode);
  146.         $this->addObjectsNode($rootNode);
  147.         $this->addAssetNode($rootNode);
  148.         $this->addDocumentsNode($rootNode);
  149.         $this->addEncryptionNode($rootNode);
  150.         $this->addModelsNode($rootNode);
  151.         $this->addRoutingNode($rootNode);
  152.         $this->addCacheNode($rootNode);
  153.         $this->addContextNode($rootNode);
  154.         $this->addAdminNode($rootNode);
  155.         $this->addWebProfilerNode($rootNode);
  156.         $this->addSecurityNode($rootNode);
  157.         $this->addEmailNode($rootNode);
  158.         $this->addNewsletterNode($rootNode);
  159.         $this->addCustomReportsNode($rootNode);
  160.         $this->addTargetingNode($rootNode);
  161.         $this->addSitemapsNode($rootNode);
  162.         $this->addWorkflowNode($rootNode);
  163.         $this->addHttpClientNode($rootNode);
  164.         $this->addApplicationLogNode($rootNode);
  165.         $this->addPredefinedPropertiesNode($rootNode);
  166.         $this->addStaticRoutesNode($rootNode);
  167.         $this->addPerspectivesNode($rootNode);
  168.         $this->addCustomViewsNode($rootNode);
  169.         return $treeBuilder;
  170.     }
  171.     /**
  172.      * Add maintenance config
  173.      *
  174.      * @param ArrayNodeDefinition $rootNode
  175.      */
  176.     private function addMaintenanceNode(ArrayNodeDefinition $rootNode)
  177.     {
  178.         $rootNode
  179.             ->children()
  180.             ->arrayNode('maintenance')
  181.             ->addDefaultsIfNotSet()
  182.             ->children()
  183.                 ->arrayNode('housekeeping')
  184.                 ->addDefaultsIfNotSet()
  185.                 ->children()
  186.                     ->integerNode('cleanup_tmp_files_atime_older_than')
  187.                         ->defaultValue(7776000// 90 days
  188.                     ->end()
  189.                     ->integerNode('cleanup_profiler_files_atime_older_than')
  190.                         ->defaultValue(1800)
  191.                     ->end()
  192.         ;
  193.     }
  194.     /**
  195.      * Add general config
  196.      *
  197.      * @param ArrayNodeDefinition $rootNode
  198.      */
  199.     private function addGeneralNode(ArrayNodeDefinition $rootNode)
  200.     {
  201.         $rootNode
  202.             ->children()
  203.             ->arrayNode('general')
  204.             ->addDefaultsIfNotSet()
  205.             ->children()
  206.                 ->scalarNode('timezone')
  207.                     ->defaultValue('')
  208.                 ->end()
  209.                 ->scalarNode('path_variable')
  210.                     ->info('Additional $PATH variable (: separated) (/x/y:/foo/bar):')
  211.                     ->defaultNull()
  212.                 ->end()
  213.                 ->scalarNode('domain')
  214.                     ->defaultValue('')
  215.                 ->end()
  216.                 ->booleanNode('redirect_to_maindomain')
  217.                     ->beforeNormalization()
  218.                         ->ifString()
  219.                         ->then(function ($v) {
  220.                             return (bool)$v;
  221.                         })
  222.                     ->end()
  223.                     ->defaultFalse()
  224.                 ->end()
  225.                 ->scalarNode('language')
  226.                     ->defaultValue('en')
  227.                 ->end()
  228.                 ->scalarNode('valid_languages')
  229.                     ->defaultValue('en')
  230.                 ->end()
  231.                 ->arrayNode('fallback_languages')
  232.                     ->performNoDeepMerging()
  233.                     ->beforeNormalization()
  234.                     ->ifArray()
  235.                         ->then(function ($v) {
  236.                             return $v;
  237.                         })
  238.                     ->end()
  239.                     ->prototype('scalar')
  240.                     ->end()
  241.                 ->end()
  242.                 ->scalarNode('default_language')
  243.                     ->defaultValue('en')
  244.                 ->end()
  245.                 ->booleanNode('disable_usage_statistics')
  246.                     ->beforeNormalization()
  247.                         ->ifString()
  248.                         ->then(function ($v) {
  249.                             return (bool)$v;
  250.                         })
  251.                     ->end()
  252.                     ->defaultFalse()
  253.                 ->end()
  254.                 ->booleanNode('debug_admin_translations')
  255.                     ->info('Debug Admin-Translations (displayed wrapped in +)')
  256.                     ->beforeNormalization()
  257.                         ->ifString()
  258.                         ->then(function ($v) {
  259.                             return (bool)$v;
  260.                         })
  261.                     ->end()
  262.                     ->defaultFalse()
  263.                 ->end()
  264.                 ->scalarNode('instance_identifier')
  265.                     ->defaultNull()
  266.                     ->info('UUID instance identifier. Has to be unique throughout multiple Pimcore instances. UUID generation will be automatically enabled if a Instance identifier is provided (do not change the instance identifier afterwards - this will cause invalid UUIDs)')
  267.                     ->end()
  268.                 ->end()
  269.             ->end();
  270.     }
  271.     /**
  272.      * @param ArrayNodeDefinition $rootNode
  273.      */
  274.     private function addServicesNode(ArrayNodeDefinition $rootNode)
  275.     {
  276.         $rootNode
  277.             ->children()
  278.             ->arrayNode('services')
  279.                 ->addDefaultsIfNotSet()
  280.                 ->children()
  281.                     ->arrayNode('google')
  282.                     ->addDefaultsIfNotSet()
  283.                     ->children()
  284.                         ->scalarNode('client_id')
  285.                             ->info('This is required for the Google API integrations. Only use a `Service Account´ from the Google Cloud Console.')
  286.                             ->defaultNull()
  287.                         ->end()
  288.                         ->scalarNode('email')
  289.                             ->info('Email address of the Google service account')
  290.                             ->defaultNull()
  291.                         ->end()
  292.                         ->scalarNode('simple_api_key')
  293.                             ->info('Server API key')
  294.                             ->defaultNull()
  295.                         ->end()
  296.                         ->scalarNode('browser_api_key')
  297.                             ->info('Browser API key')
  298.                             ->defaultNull()
  299.                         ->end()
  300.                     ->end()
  301.                     ->end()
  302.                 ->end()
  303.             ->end();
  304.     }
  305.     /**
  306.      * @param ArrayNodeDefinition $rootNode
  307.      */
  308.     private function addModelsNode(ArrayNodeDefinition $rootNode)
  309.     {
  310.         $rootNode
  311.             ->children()
  312.                 ->arrayNode('models')
  313.                     ->addDefaultsIfNotSet()
  314.                     ->children()
  315.                         ->arrayNode('class_overrides')
  316.                             ->useAttributeAsKey('name')
  317.                             ->prototype('scalar');
  318.     }
  319.     /**
  320.      * @param ArrayNodeDefinition $rootNode
  321.      */
  322.     private function addHttpClientNode(ArrayNodeDefinition $rootNode)
  323.     {
  324.         $rootNode
  325.             ->children()
  326.                 ->arrayNode('httpclient')
  327.                 ->addDefaultsIfNotSet()
  328.                     ->children()
  329.                         ->scalarNode('adapter')
  330.                             ->info('Set to `Proxy` if proxy server should be used')
  331.                             ->defaultValue('Socket')
  332.                         ->end()
  333.                         ->scalarNode('proxy_host')
  334.                             ->defaultNull()
  335.                         ->end()
  336.                         ->scalarNode('proxy_port')
  337.                             ->defaultNull()
  338.                         ->end()
  339.                         ->scalarNode('proxy_user')
  340.                             ->defaultNull()
  341.                         ->end()
  342.                         ->scalarNode('proxy_pass')
  343.                             ->defaultNull()
  344.                         ->end()
  345.                     ->end()
  346.                 ->end()
  347.             ->end();
  348.     }
  349.     /**
  350.      * @param ArrayNodeDefinition $rootNode
  351.      */
  352.     private function addApplicationLogNode(ArrayNodeDefinition $rootNode)
  353.     {
  354.         $rootNode
  355.             ->children()
  356.                 ->arrayNode('applicationlog')
  357.                 ->addDefaultsIfNotSet()
  358.                     ->children()
  359.                         ->arrayNode('mail_notification')
  360.                             ->children()
  361.                                 ->booleanNode('send_log_summary')
  362.                                     ->info('Send log summary via email')
  363.                                     ->beforeNormalization()
  364.                                         ->ifString()
  365.                                         ->then(function ($v) {
  366.                                             return (bool)$v;
  367.                                         })
  368.                                     ->end()
  369.                                     ->defaultFalse()
  370.                                 ->end()
  371.                                 ->scalarNode('filter_priority')
  372.                                     ->info('Filter threshold for email summary, choose one of: 7 (debug), 6 (info), 5 (notice), 4 (warning), 3 (error), 2 (critical), 1 (alert) ,0 (emerg)')
  373.                                     ->defaultNull()
  374.                                 ->end()
  375.                                 ->scalarNode('mail_receiver')
  376.                                 ->info('Log summary receivers. Separate multiple email receivers by using ;')
  377.                                 ->end()
  378.                             ->end()
  379.                         ->end()
  380.                         ->scalarNode('archive_treshold')
  381.                             ->info('Archive threshold in days')
  382.                             ->defaultValue(30)
  383.                         ->end()
  384.                         ->scalarNode('archive_alternative_database')
  385.                             ->info('Archive database name (optional). Tables will get archived to a different database, recommended when huge amounts of logs will be generated')
  386.                             ->defaultValue('')
  387.                         ->end()
  388.                         ->scalarNode('delete_archive_threshold')
  389.                             ->info('Threshold for deleting application log archive tables (in months)')
  390.                             ->defaultValue('6')
  391.                         ->end()
  392.                     ->end()
  393.             ->end();
  394.     }
  395.     /**
  396.      * Add asset specific extension config
  397.      *
  398.      * @param ArrayNodeDefinition $rootNode
  399.      */
  400.     private function addAssetNode(ArrayNodeDefinition $rootNode)
  401.     {
  402.         $assetsNode $rootNode
  403.             ->children()
  404.                 ->arrayNode('assets')
  405.                 ->addDefaultsIfNotSet()
  406.                 ->children()
  407.                     ->arrayNode('frontend_prefixes')
  408.                         ->addDefaultsIfNotSet()
  409.                         ->children()
  410.                             ->scalarNode('source')
  411.                                 ->defaultValue('')
  412.                                 ->end()
  413.                             ->scalarNode('thumbnail')
  414.                                 ->defaultValue('')
  415.                                 ->end()
  416.                             ->scalarNode('thumbnail_deferred')
  417.                                 ->defaultValue('')
  418.                                 ->end()
  419.                         ->end()
  420.                     ->end()
  421.                     ->scalarNode('preview_image_thumbnail')
  422.                         ->defaultNull()
  423.                         ->end()
  424.                     ->scalarNode('default_upload_path')
  425.                         ->defaultValue('_default_upload_bucket')
  426.                         ->end()
  427.                     ->integerNode('tree_paging_limit')
  428.                         ->defaultValue(100)
  429.                         ->end()
  430.                     ->arrayNode('image')
  431.                         ->addDefaultsIfNotSet()
  432.                         ->children()
  433.                             ->integerNode('max_pixels')
  434.                                 ->defaultValue(40000000)
  435.                             ->end()
  436.                             ->arrayNode('low_quality_image_preview')
  437.                                 ->addDefaultsIfNotSet()
  438.                                 ->canBeDisabled()
  439.                             ->end()
  440.                             ->arrayNode('focal_point_detection')
  441.                                 ->addDefaultsIfNotSet()
  442.                                 ->canBeDisabled()
  443.                             ->end()
  444.                             ->arrayNode('thumbnails')
  445.                                 ->addDefaultsIfNotSet()
  446.                                 ->children()
  447.                                     ->arrayNode('definitions')
  448.                                         ->normalizeKeys(false)
  449.                                         ->prototype('array')
  450.                                             ->children()
  451.                                                 ->scalarNode('id')->end()
  452.                                                 ->scalarNode('name')->end()
  453.                                                 ->scalarNode('description')->end()
  454.                                                 ->scalarNode('group')->end()
  455.                                                 ->scalarNode('format')->end()
  456.                                                 ->scalarNode('quality')->end()
  457.                                                 ->scalarNode('highResolution')->end()
  458.                                                 ->booleanNode('preserveColor')->end()
  459.                                                 ->booleanNode('preserveMetaData')->end()
  460.                                                 ->booleanNode('rasterizeSVG')->end()
  461.                                                 ->booleanNode('downloadable')->end()
  462.                                                 ->integerNode('modificationDate')->end()
  463.                                                 ->integerNode('creationDate')->end()
  464.                                                 ->booleanNode('preserveAnimation')->end()
  465.                                                 ->arrayNode('items')
  466.                                                     ->prototype('array')
  467.                                                         ->children()
  468.                                                             ->scalarNode('method')->end()
  469.                                                             ->arrayNode('arguments')
  470.                                                                 ->prototype('variable')->end()
  471.                                                             ->end()
  472.                                                         ->end()
  473.                                                     ->end()
  474.                                                 ->end()
  475.                                                 ->arrayNode('medias')
  476.                                                     ->normalizeKeys(false)
  477.                                                     ->prototype('array')
  478.                                                         ->arrayProtoType()
  479.                                                             ->children()
  480.                                                                 ->scalarNode('method')->end()
  481.                                                                 ->arrayNode('arguments')
  482.                                                                     ->prototype('variable')->end()
  483.                                                                 ->end()
  484.                                                             ->end()
  485.                                                         ->end()
  486.                                                     ->end()
  487.                                                 ->end()
  488.                                             ->end()
  489.                                         ->end()
  490.                                     ->end()
  491.                                     ->booleanNode('clip_auto_support')
  492.                                         ->beforeNormalization()
  493.                                             ->ifString()
  494.                                             ->then(function ($v) {
  495.                                                 return (bool)$v;
  496.                                             })
  497.                                         ->end()
  498.                                         ->defaultTrue()
  499.                                     ->end()
  500.                                     ->arrayNode('auto_formats')
  501.                                         ->prototype('array')
  502.                                             ->canBeDisabled()
  503.                                             ->children()
  504.                                                 ->scalarNode('quality')->end()
  505.                                             ->end()
  506.                                         ->end()
  507.                                         ->defaultValue([
  508.                                             'avif' => [
  509.                                                 'enabled' => true,
  510.                                                 'quality' => 15,
  511.                                             ],
  512.                                             'webp' => [
  513.                                                 'enabled' => true,
  514.                                                 'quality' => null,
  515.                                             ],
  516.                                         ])
  517.                                     ->end()
  518.                                     ->booleanNode('auto_clear_temp_files')
  519.                                         ->beforeNormalization()
  520.                                             ->ifString()
  521.                                             ->then(function ($v) {
  522.                                                 return (bool)$v;
  523.                                             })
  524.                                         ->end()
  525.                                         ->defaultTrue()
  526.                                     ->end()
  527.                                 ->end()
  528.                             ->end()
  529.                         ->end()
  530.                     ->end()
  531.                     ->arrayNode('video')
  532.                         ->addDefaultsIfNotSet()
  533.                         ->children()
  534.                             ->arrayNode('thumbnails')
  535.                                 ->addDefaultsIfNotSet()
  536.                                 ->children()
  537.                                     ->arrayNode('definitions')
  538.                                         ->normalizeKeys(false)
  539.                                         ->prototype('array')
  540.                                             ->children()
  541.                                                 ->scalarNode('id')->end()
  542.                                                 ->scalarNode('name')->end()
  543.                                                 ->scalarNode('description')->end()
  544.                                                 ->scalarNode('group')->end()
  545.                                                 ->scalarNode('videoBitrate')->end()
  546.                                                 ->scalarNode('audioBitrate')->end()
  547.                                                 ->scalarNode('quality')->end()
  548.                                                 ->integerNode('modificationDate')->end()
  549.                                                 ->integerNode('creationDate')->end()
  550.                                                 ->arrayNode('items')
  551.                                                     ->prototype('array')
  552.                                                         ->children()
  553.                                                             ->scalarNode('method')->end()
  554.                                                             ->arrayNode('arguments')
  555.                                                                 ->prototype('variable')->end()
  556.                                                             ->end()
  557.                                                         ->end()
  558.                                                     ->end()
  559.                                                 ->end()
  560.                                                 ->arrayNode('medias')
  561.                                                     ->normalizeKeys(false)
  562.                                                     ->prototype('array')
  563.                                                         ->arrayProtoType()
  564.                                                             ->children()
  565.                                                                 ->scalarNode('method')->end()
  566.                                                                 ->arrayNode('arguments')
  567.                                                                     ->prototype('variable')->end()
  568.                                                                 ->end()
  569.                                                             ->end()
  570.                                                         ->end()
  571.                                                     ->end()
  572.                                                 ->end()
  573.                                             ->end()
  574.                                         ->end()
  575.                                     ->end()
  576.                                     ->booleanNode('auto_clear_temp_files')
  577.                                     ->defaultTrue()
  578.                                     ->end()
  579.                                 ->end()
  580.                             ->end()
  581.                         ->end()
  582.                     ->end()
  583.                     ->arrayNode('versions')
  584.                         ->addDefaultsIfNotSet()
  585.                         ->children()
  586.                             ->scalarNode('days')
  587.                                 ->defaultNull()
  588.                             ->end()
  589.                             ->scalarNode('steps')
  590.                                 ->defaultNull()
  591.                             ->end()
  592.                             ->booleanNode('use_hardlinks')
  593.                                 ->beforeNormalization()
  594.                                     ->ifString()
  595.                                     ->then(function ($v) {
  596.                                         return (bool)$v;
  597.                                     })
  598.                                 ->end()
  599.                                 ->defaultTrue()
  600.                             ->end()
  601.                             ->booleanNode('disable_stack_trace')
  602.                                 ->beforeNormalization()
  603.                                     ->ifString()
  604.                                     ->then(function ($v) {
  605.                                         return (bool)$v;
  606.                                     })
  607.                                 ->end()
  608.                                 ->defaultFalse()
  609.                             ->end()
  610.                         ->end()
  611.                     ->end()
  612.                     ->scalarNode('icc_rgb_profile')
  613.                         ->info('Absolute path to default ICC RGB profile (if no embedded profile is given)')
  614.                         ->defaultNull()
  615.                     ->end()
  616.                     ->scalarNode('icc_cmyk_profile')
  617.                         ->info('Absolute path to default ICC CMYK profile (if no embedded profile is given)')
  618.                         ->defaultNull()
  619.                     ->end()
  620.                     ->booleanNode('hide_edit_image')
  621.                         ->defaultFalse()
  622.                     ->end()
  623.                     ->booleanNode('disable_tree_preview')
  624.                         ->defaultTrue()
  625.                     ->end()
  626.                 ->end();
  627.         $assetsNode
  628.             ->children()
  629.                 ->arrayNode('metadata')
  630.                 ->addDefaultsIfNotSet()
  631.                     ->children()
  632.                         ->arrayNode('predefined')
  633.                             ->children()
  634.                                 ->arrayNode('definitions')
  635.                                 ->normalizeKeys(false)
  636.                                     ->prototype('array')
  637.                                         ->children()
  638.                                             ->scalarNode('name')->end()
  639.                                             ->scalarNode('description')->end()
  640.                                             ->scalarNode('group')->end()
  641.                                             ->scalarNode('language')->end()
  642.                                             ->scalarNode('type')->end()
  643.                                             ->scalarNode('data')->end()
  644.                                             ->scalarNode('targetSubtype')->end()
  645.                                             ->scalarNode('config')->end()
  646.                                             ->booleanNode('inheritable')
  647.                                                 ->beforeNormalization()
  648.                                                 ->ifString()
  649.                                                 ->then(function ($v) {
  650.                                                     return (bool)$v;
  651.                                                 })
  652.                                                 ->end()
  653.                                             ->end()
  654.                                             ->integerNode('creationDate')->end()
  655.                                             ->integerNode('modificationDate')->end()
  656.                                         ->end()
  657.                                     ->end()
  658.                                 ->end()
  659.                             ->end()
  660.                         ->end()
  661.                         ->arrayNode('class_definitions')
  662.                             ->children()
  663.                                 ->arrayNode('data')
  664.                                     ->children()
  665.                                         ->arrayNode('map')
  666.                                             ->useAttributeAsKey('name')
  667.                                             ->prototype('scalar')->end()
  668.                                         ->end()
  669.                                         ->arrayNode('prefixes')
  670.                                             ->prototype('scalar')->end()
  671.                                         ->end()
  672.                                     ->end()
  673.                                 ->end()
  674.                             ->end()
  675.                         ->end();
  676.     }
  677.     /**
  678.      * Add object specific extension config
  679.      *
  680.      * @param ArrayNodeDefinition $rootNode
  681.      */
  682.     private function addObjectsNode(ArrayNodeDefinition $rootNode)
  683.     {
  684.         $objectsNode $rootNode
  685.             ->children()
  686.                 ->arrayNode('objects')
  687.                     ->ignoreExtraKeys()
  688.                     ->addDefaultsIfNotSet()
  689.                     ->children()
  690.                         ->booleanNode('ignore_localized_query_fallback')
  691.                             ->beforeNormalization()
  692.                             ->ifString()
  693.                                 ->then(function ($v) {
  694.                                     return (bool)$v;
  695.                                 })
  696.                                 ->end()
  697.                             ->defaultFalse()
  698.                         ->end()
  699.                         ->integerNode('tree_paging_limit')
  700.                             ->defaultValue(30)
  701.                         ->end()
  702.                         ->integerNode('auto_save_interval')
  703.                             ->defaultValue(60)
  704.                         ->end()
  705.                         ->arrayNode('versions')
  706.                             ->children()
  707.                                 ->scalarNode('days')->defaultNull()->end()
  708.                                 ->scalarNode('steps')->defaultNull()->end()
  709.                                 ->booleanNode('disable_stack_trace')
  710.                                     ->beforeNormalization()
  711.                                     ->ifString()
  712.                                         ->then(function ($v) {
  713.                                             return (bool)$v;
  714.                                         })
  715.                                     ->end()
  716.                                     ->defaultFalse()
  717.                                 ->end()
  718.                             ->end()
  719.                         ->end()
  720.                     ->end();
  721.         $classDefinitionsNode $objectsNode
  722.             ->children()
  723.                 ->arrayNode('class_definitions')
  724.                     ->addDefaultsIfNotSet();
  725.         $this->addImplementationLoaderNode($classDefinitionsNode'data');
  726.         $this->addImplementationLoaderNode($classDefinitionsNode'layout');
  727.     }
  728.     /**
  729.      * Add encryption specific extension config
  730.      *
  731.      * @param ArrayNodeDefinition $rootNode
  732.      */
  733.     private function addEncryptionNode(ArrayNodeDefinition $rootNode)
  734.     {
  735.         $encryptionNode $rootNode
  736.             ->children()
  737.             ->arrayNode('encryption')->addDefaultsIfNotSet();
  738.         $encryptionNode
  739.             ->children()
  740.             ->scalarNode('secret')->defaultNull();
  741.     }
  742.     /**
  743.      * Add document specific extension config
  744.      *
  745.      * @param ArrayNodeDefinition $rootNode
  746.      */
  747.     private function addDocumentsNode(ArrayNodeDefinition $rootNode)
  748.     {
  749.         $documentsNode $rootNode
  750.             ->children()
  751.                 ->arrayNode('documents')
  752.                     ->ignoreExtraKeys()
  753.                     ->addDefaultsIfNotSet();
  754.         $documentsNode
  755.             ->children()
  756.                  ->arrayNode('doc_types')
  757.                     ->addDefaultsIfNotSet()
  758.                     ->children()
  759.                         ->arrayNode('definitions')
  760.                         ->normalizeKeys(false)
  761.                             ->prototype('array')
  762.                                 ->children()
  763.                                     ->scalarNode('name')->end()
  764.                                     ->scalarNode('group')->end()
  765.                                     ->scalarNode('module')->end()
  766.                                     ->scalarNode('controller')->end()
  767.                                     ->scalarNode('template')->end()
  768.                                     ->scalarNode('type')->end()
  769.                                     ->integerNode('priority')->end()
  770.                                     ->integerNode('creationDate')->end()
  771.                                     ->integerNode('modificationDate')->end()
  772.                                     ->scalarNode('staticGeneratorEnabled')->end()
  773.                                 ->end()
  774.                             ->end()
  775.                         ->end()
  776.                     ->end()
  777.                 ->end()
  778.                 ->arrayNode('versions')
  779.                     ->children()
  780.                         ->scalarNode('days')
  781.                             ->defaultNull()
  782.                         ->end()
  783.                         ->scalarNode('steps')
  784.                             ->defaultNull()
  785.                         ->end()
  786.                         ->booleanNode('disable_stack_trace')
  787.                             ->beforeNormalization()
  788.                             ->ifString()
  789.                                 ->then(function ($v) {
  790.                                     return (bool)$v;
  791.                                 })
  792.                             ->end()
  793.                             ->defaultFalse()
  794.                         ->end()
  795.                     ->end()
  796.                 ->end()
  797.                 ->scalarNode('default_controller')
  798.                     ->defaultValue('App\\Controller\\DefaultController::defaultAction')
  799.                 ->end()
  800.                 ->arrayNode('error_pages')
  801.                     ->children()
  802.                         ->scalarNode('default')
  803.                             ->defaultNull()
  804.                         ->end()
  805.                         ->arrayNode('localized')
  806.                             ->performNoDeepMerging()
  807.                             ->beforeNormalization()
  808.                                 ->ifArray()
  809.                                     ->then(function ($v) {
  810.                                         return $v;
  811.                                     })
  812.                             ->end()
  813.                             ->prototype('scalar')
  814.                             ->end()
  815.                         ->end()
  816.                     ->end()
  817.                 ->end()
  818.                 ->scalarNode('allow_trailing_slash')
  819.                     ->defaultValue('no')
  820.                 ->end()
  821.                 ->booleanNode('generate_preview')
  822.                     ->beforeNormalization()
  823.                         ->ifString()
  824.                         ->then(function ($v) {
  825.                             return (bool)$v;
  826.                         })
  827.                     ->end()
  828.                     ->defaultFalse()
  829.                 ->end()
  830.                 ->integerNode('tree_paging_limit')
  831.                     ->defaultValue(50)
  832.                 ->end()
  833.                 ->arrayNode('editables')
  834.                     ->addDefaultsIfNotSet()
  835.                     ->children()
  836.                         ->arrayNode('map')
  837.                             ->useAttributeAsKey('name')
  838.                             ->prototype('scalar')->end()
  839.                         ->end()
  840.                         ->arrayNode('prefixes')
  841.                             ->prototype('scalar')->end()
  842.                         ->end()
  843.                     ->end()
  844.                 ->end()
  845.                 ->arrayNode('types')
  846.                     ->info('list of supported document types')
  847.                     ->scalarPrototype()->end()
  848.                 ->end()
  849.                 ->arrayNode('valid_tables')
  850.                     ->info('list of supported documents_* tables')
  851.                     ->scalarPrototype()->end()
  852.                 ->end()
  853.                 ->arrayNode('areas')
  854.                     ->addDefaultsIfNotSet()
  855.                     ->children()
  856.                         ->booleanNode('autoload')
  857.                             ->beforeNormalization()
  858.                                 ->ifString()
  859.                                 ->then(function ($v) {
  860.                                     return (bool)$v;
  861.                                 })
  862.                             ->end()
  863.                             ->defaultTrue()
  864.                         ->end()
  865.                     ->end()
  866.                 ->end()
  867.                 ->arrayNode('newsletter')
  868.                     ->addDefaultsIfNotSet()
  869.                     ->children()
  870.                         ->scalarNode('defaultUrlPrefix')
  871.                             ->defaultNull()
  872.                         ->end()
  873.                     ->end()
  874.                 ->end()
  875.                 ->arrayNode('web_to_print')
  876.                     ->addDefaultsIfNotSet()
  877.                         ->children()
  878.                             ->scalarNode('pdf_creation_php_memory_limit')
  879.                                 ->defaultValue('2048M')
  880.                             ->end()
  881.                             ->scalarNode('default_controller_print_page')
  882.                                 ->defaultValue('App\\Controller\\Web2printController::defaultAction')
  883.                             ->end()
  884.                             ->scalarNode('default_controller_print_container')
  885.                                 ->defaultValue('App\\Controller\\Web2printController::containerAction')
  886.                             ->end()
  887.                             ->booleanNode('enableInDefaultView')->end()
  888.                             ->scalarNode('generalTool')->end()
  889.                             ->scalarNode('generalDocumentSaveMode')->end()
  890.                             ->scalarNode('pdfreactorVersion')->end()
  891.                             ->scalarNode('pdfreactorProtocol')->end()
  892.                             ->scalarNode('pdfreactorServer')->end()
  893.                             ->scalarNode('pdfreactorServerPort')->end()
  894.                             ->scalarNode('pdfreactorBaseUrl')->end()
  895.                             ->scalarNode('pdfreactorApiKey')->end()
  896.                             ->scalarNode('pdfreactorLicence')->end()
  897.                             ->booleanNode('pdfreactorEnableLenientHttpsMode')->end()
  898.                             ->booleanNode('pdfreactorEnableDebugMode')->end()
  899.                             ->scalarNode('wkhtmltopdfBin')->end()
  900.                             ->variableNode('wkhtml2pdfOptions')->end()
  901.                             ->scalarNode('wkhtml2pdfHostname')->end()
  902.                             ->scalarNode('headlessChromeSettings')->end()
  903.                         ->end()
  904.                 ->end()
  905.                 ->integerNode('auto_save_interval')
  906.                     ->defaultValue(60)
  907.                 ->end()
  908.                 ->arrayNode('static_page_router')
  909.                     ->addDefaultsIfNotSet()
  910.                     ->children()
  911.                         ->booleanNode('enabled')
  912.                             ->defaultFalse()
  913.                             ->info('Enable Static Page router for document when using remote storage for generated pages')
  914.                         ->end()
  915.                         ->scalarNode('route_pattern')
  916.                             ->defaultNull()
  917.                             ->info('Optionally define route patterns to lookup static pages. Regular Expressions like: /^\/en\/Magazine/')
  918.                         ->end()
  919.                 ->end()
  920.             ->end();
  921.     }
  922.     /**
  923.      * Add implementation node config (map, prefixes)
  924.      *
  925.      * @param ArrayNodeDefinition $node
  926.      * @param string $name
  927.      */
  928.     private function addImplementationLoaderNode(ArrayNodeDefinition $node$name)
  929.     {
  930.         $node
  931.             ->children()
  932.                 ->arrayNode($name)
  933.                     ->addDefaultsIfNotSet()
  934.                     ->children()
  935.                         ->arrayNode('map')
  936.                             ->useAttributeAsKey('name')
  937.                             ->prototype('scalar')->end()
  938.                         ->end()
  939.                         ->arrayNode('prefixes')
  940.                             ->prototype('scalar')->end()
  941.                         ->end()
  942.                     ->end()
  943.                 ->end()
  944.             ->end();
  945.     }
  946.     private function addRoutingNode(ArrayNodeDefinition $rootNode)
  947.     {
  948.         $rootNode
  949.             ->children()
  950.                 ->arrayNode('routing')
  951.                     ->addDefaultsIfNotSet()
  952.                     ->children()
  953.                         ->booleanNode('allow_processing_unpublished_fallback_document')
  954.                             ->beforeNormalization()
  955.                                 ->ifString()
  956.                                 ->then(function ($v) {
  957.                                     return (bool)$v;
  958.                                 })
  959.                             ->end()
  960.                             ->defaultFalse()
  961.                             ->setDeprecated(
  962.                                 'pimcore/pimcore',
  963.                                 '10.1',
  964.                                 'The "%node%" option is deprecated since Pimcore 10.1, it will be removed in Pimcore 11.'
  965.                             )
  966.                         ->end()
  967.                         ->arrayNode('direct_route_document_types')
  968.                             ->scalarPrototype()->end()
  969.                         ->end()
  970.                         ->arrayNode('static')
  971.                             ->addDefaultsIfNotSet()
  972.                             ->children()
  973.                                 ->arrayNode('locale_params')
  974.                                     ->info('Route params from this list will be mapped to _locale if _locale is not set explicitely')
  975.                                     ->prototype('scalar')
  976.                                     ->defaultValue([])
  977.                                 ->end()
  978.                             ->end()
  979.                         ->end()
  980.                     ->end()
  981.                 ->end();
  982.     }
  983.     /**
  984.      * Add context config
  985.      *
  986.      * @param ArrayNodeDefinition $rootNode
  987.      */
  988.     private function addContextNode(ArrayNodeDefinition $rootNode)
  989.     {
  990.         $contextNode $rootNode->children()
  991.             ->arrayNode('context')
  992.             ->useAttributeAsKey('name');
  993.         /** @var ArrayNodeDefinition|NodeDefinition $prototype */
  994.         $prototype $contextNode->prototype('array');
  995.         // define routes child on each context entry
  996.         $this->addRoutesChild($prototype'routes');
  997.     }
  998.     /**
  999.      * Add admin config
  1000.      *
  1001.      * @param ArrayNodeDefinition $rootNode
  1002.      */
  1003.     private function addAdminNode(ArrayNodeDefinition $rootNode)
  1004.     {
  1005.         $adminNode $rootNode->children()
  1006.             ->arrayNode('admin')
  1007.             ->ignoreExtraKeys()
  1008.             ->addDefaultsIfNotSet();
  1009.         // add session attribute bag config
  1010.         $this->addAdminSessionAttributeBags($adminNode);
  1011.         // unauthenticated routes won't be double checked for authentication in AdminControllerListener
  1012.         $this->addRoutesChild($adminNode'unauthenticated_routes');
  1013.         $adminNode
  1014.             ->children()
  1015.                 ->arrayNode('translations')
  1016.                     ->addDefaultsIfNotSet()
  1017.                     ->children()
  1018.                         ->scalarNode('path')->defaultNull()->end()
  1019.                     ->end()
  1020.                 ->end()
  1021.             ->end();
  1022.     }
  1023.     /**
  1024.      * @param ArrayNodeDefinition $adminNode
  1025.      */
  1026.     private function addAdminSessionAttributeBags(ArrayNodeDefinition $adminNode)
  1027.     {
  1028.         // Normalizes session bag config. Allows the following formats (all formats will be
  1029.         // normalized to the third format.
  1030.         //
  1031.         // attribute_bags:
  1032.         //      - foo
  1033.         //      - bar
  1034.         //
  1035.         // attribute_bags:
  1036.         //      foo: _foo
  1037.         //      bar: _bar
  1038.         //
  1039.         // attribute_bags:
  1040.         //      foo:
  1041.         //          storage_key: _foo
  1042.         //      bar:
  1043.         //          storage_key: _bar
  1044.         $normalizers = [
  1045.             'assoc' => function (array $array) {
  1046.                 $result = [];
  1047.                 foreach ($array as $name => $value) {
  1048.                     if (null === $value) {
  1049.                         $value = [
  1050.                             'storage_key' => '_' $name,
  1051.                         ];
  1052.                     }
  1053.                     if (is_string($value)) {
  1054.                         $value = [
  1055.                             'storage_key' => $value,
  1056.                         ];
  1057.                     }
  1058.                     $result[$name] = $value;
  1059.                 }
  1060.                 return $result;
  1061.             },
  1062.             'sequential' => function (array $array) {
  1063.                 $result = [];
  1064.                 foreach ($array as $name) {
  1065.                     $result[$name] = [
  1066.                         'storage_key' => '_' $name,
  1067.                     ];
  1068.                 }
  1069.                 return $result;
  1070.             },
  1071.         ];
  1072.         $adminNode
  1073.             ->children()
  1074.                 ->arrayNode('session')
  1075.                     ->addDefaultsIfNotSet()
  1076.                     ->children()
  1077.                         ->arrayNode('attribute_bags')
  1078.                             ->useAttributeAsKey('name')
  1079.                             ->beforeNormalization()
  1080.                                 ->ifArray()->then(function ($v) use ($normalizers) {
  1081.                                     if (isAssocArray($v)) {
  1082.                                         return $normalizers['assoc']($v);
  1083.                                     } else {
  1084.                                         return $normalizers['sequential']($v);
  1085.                                     }
  1086.                                 })
  1087.                             ->end()
  1088.                             ->example([
  1089.                                 ['foo''bar'],
  1090.                                 [
  1091.                                     'foo' => '_foo',
  1092.                                     'bar' => '_bar',
  1093.                                 ],
  1094.                                 [
  1095.                                     'foo' => [
  1096.                                         'storage_key' => '_foo',
  1097.                                     ],
  1098.                                     'bar' => [
  1099.                                         'storage_key' => '_bar',
  1100.                                     ],
  1101.                                 ],
  1102.                             ])
  1103.                             ->prototype('array')
  1104.                                 ->children()
  1105.                                     ->scalarNode('storage_key')
  1106.                                         ->defaultNull()
  1107.                                     ->end()
  1108.                                 ->end()
  1109.                             ->end()
  1110.                         ->end()
  1111.                     ->end()
  1112.                 ->end()
  1113.             ->end();
  1114.     }
  1115.     private function addSecurityNode(ArrayNodeDefinition $rootNode)
  1116.     {
  1117.         $rootNode
  1118.             ->children()
  1119.                 ->arrayNode('security')
  1120.                     ->addDefaultsIfNotSet()
  1121.                     ->children()
  1122.                         ->enumNode('factory_type')
  1123.                             ->values(['encoder''password_hasher'])
  1124.                             ->defaultValue('encoder')
  1125.                         ->end()
  1126.                         ->arrayNode('encoder_factories')
  1127.                             ->info('Encoder factories to use as className => factory service ID mapping')
  1128.                             ->example([
  1129.                                 'App\Model\DataObject\User1' => [
  1130.                                     'id' => 'website_demo.security.encoder_factory2',
  1131.                                 ],
  1132.                                 'App\Model\DataObject\User2' => 'website_demo.security.encoder_factory2',
  1133.                             ])
  1134.                             ->useAttributeAsKey('class')
  1135.                             ->prototype('array')
  1136.                             ->beforeNormalization()->ifString()->then(function ($v) {
  1137.                                 return ['id' => $v];
  1138.                             })->end()
  1139.                             ->children()
  1140.                                 ->scalarNode('id')->end()
  1141.                             ->end()
  1142.                             ->end()
  1143.                         ->end()
  1144.                         ->arrayNode('password_hasher_factories')
  1145.                             ->info('Password hasher factories to use as className => factory service ID mapping')
  1146.                             ->example([
  1147.                                 'App\Model\DataObject\User1' => [
  1148.                                     'id' => 'website_demo.security.encoder_factory2',
  1149.                                 ],
  1150.                                 'App\Model\DataObject\User2' => 'website_demo.security.encoder_factory2',
  1151.                             ])
  1152.                             ->useAttributeAsKey('class')
  1153.                             ->prototype('array')
  1154.                             ->beforeNormalization()->ifString()->then(function ($v) {
  1155.                                 return ['id' => $v];
  1156.                             })->end()
  1157.                             ->children()
  1158.                             ->scalarNode('id')->end()
  1159.                             ->end()
  1160.                         ->end()
  1161.                     ->end()
  1162.                 ->end()
  1163.             ->end();
  1164.     }
  1165.     /**
  1166.      * Configure exclude paths for web profiler toolbar
  1167.      *
  1168.      * @param ArrayNodeDefinition $rootNode
  1169.      */
  1170.     private function addWebProfilerNode(ArrayNodeDefinition $rootNode)
  1171.     {
  1172.         $webProfilerNode $rootNode->children()
  1173.             ->arrayNode('web_profiler')
  1174.                 ->example([
  1175.                     'toolbar' => [
  1176.                         'excluded_routes' => [
  1177.                             ['path' => '^/test/path'],
  1178.                         ],
  1179.                     ],
  1180.                 ])
  1181.                 ->addDefaultsIfNotSet();
  1182.         $toolbarNode $webProfilerNode->children()
  1183.             ->arrayNode('toolbar')
  1184.                 ->addDefaultsIfNotSet();
  1185.         $this->addRoutesChild($toolbarNode'excluded_routes');
  1186.     }
  1187.     /**
  1188.      * Add a route prototype child
  1189.      *
  1190.      * @param ArrayNodeDefinition $parent
  1191.      * @param string $name
  1192.      */
  1193.     private function addRoutesChild(ArrayNodeDefinition $parent$name)
  1194.     {
  1195.         $node $parent->children()->arrayNode($name);
  1196.         /** @var ArrayNodeDefinition $prototype */
  1197.         $prototype $node->prototype('array');
  1198.         $prototype
  1199.             ->beforeNormalization()
  1200.                 ->ifNull()->then(function () {
  1201.                     return [];
  1202.                 })
  1203.             ->end()
  1204.             ->children()
  1205.                 ->scalarNode('path')->defaultFalse()->end()
  1206.                 ->scalarNode('route')->defaultFalse()->end()
  1207.                 ->scalarNode('host')->defaultFalse()->end()
  1208.                 ->arrayNode('methods')
  1209.                     ->prototype('scalar')->end()
  1210.                 ->end()
  1211.             ->end();
  1212.     }
  1213.     /**
  1214.      * Add cache config
  1215.      *
  1216.      * @param ArrayNodeDefinition $rootNode
  1217.      */
  1218.     private function addCacheNode(ArrayNodeDefinition $rootNode)
  1219.     {
  1220.         $rootNode->children()
  1221.             ->arrayNode('full_page_cache')
  1222.                 ->ignoreExtraKeys()
  1223.                 ->canBeDisabled()
  1224.                 ->addDefaultsIfNotSet()
  1225.                 ->children()
  1226.                     ->scalarNode('lifetime')
  1227.                         ->info('Optional output-cache lifetime (in seconds) after the cache expires, if not defined the cache will be cleaned on every action inside the CMS, otherwise not (for high traffic sites)')
  1228.                         ->defaultNull()
  1229.                     ->end()
  1230.                     ->scalarNode('exclude_patterns')
  1231.                         ->info('Regular Expressions like: /^\/dir\/toexclude/')
  1232.                     ->end()
  1233.                     ->scalarNode('exclude_cookie')
  1234.                         ->info('Comma separated list of cookie names, that will automatically disable the full-page cache')
  1235.                     ->end()
  1236.                 ->end()
  1237.             ->end();
  1238.     }
  1239.     /**
  1240.      * Adds configuration for email source adapters
  1241.      *
  1242.      * @param ArrayNodeDefinition $rootNode
  1243.      */
  1244.     private function addEmailNode(ArrayNodeDefinition $rootNode)
  1245.     {
  1246.         $rootNode
  1247.             ->children()
  1248.                 ->arrayNode('email')
  1249.                 ->addDefaultsIfNotSet()
  1250.                     ->children()
  1251.                         ->arrayNode('sender')
  1252.                             ->addDefaultsIfNotSet()
  1253.                             ->children()
  1254.                                 ->scalarNode('name')
  1255.                                     ->defaultValue('')
  1256.                                 ->end()
  1257.                                 ->scalarNode('email')
  1258.                                     ->defaultValue('')
  1259.                                 ->end()
  1260.                             ->end()
  1261.                         ->end()
  1262.                         ->arrayNode('return')
  1263.                             ->addDefaultsIfNotSet()
  1264.                             ->children()
  1265.                                 ->scalarNode('name')
  1266.                                     ->defaultValue('')
  1267.                                 ->end()
  1268.                                 ->scalarNode('email')
  1269.                                     ->defaultValue('')
  1270.                                 ->end()
  1271.                             ->end()
  1272.                         ->end()
  1273.                         ->arrayNode('debug')
  1274.                             ->addDefaultsIfNotSet()
  1275.                             ->children()
  1276.                                 ->scalarNode('email_addresses')
  1277.                                     ->defaultValue('')
  1278.                                 ->end()
  1279.                             ->end()
  1280.                         ->end()
  1281.                         ->scalarNode('usespecific')
  1282.                             ->defaultFalse()
  1283.                         ->end()
  1284.                     ->end()
  1285.                 ->end()
  1286.             ->end();
  1287.     }
  1288.     /**
  1289.      * Adds configuration tree for newsletter source adapters
  1290.      *
  1291.      * @param ArrayNodeDefinition $rootNode
  1292.      */
  1293.     private function addNewsletterNode(ArrayNodeDefinition $rootNode)
  1294.     {
  1295.         $rootNode
  1296.             ->children()
  1297.                 ->arrayNode('newsletter')
  1298.                     ->addDefaultsIfNotSet()
  1299.                     ->children()
  1300.                         ->arrayNode('sender')
  1301.                             ->children()
  1302.                                 ->scalarNode('name')->end()
  1303.                                 ->scalarNode('email')->end()
  1304.                             ->end()
  1305.                         ->end()
  1306.                         ->arrayNode('return')
  1307.                             ->children()
  1308.                                 ->scalarNode('name')->end()
  1309.                                 ->scalarNode('email')->end()
  1310.                             ->end()
  1311.                         ->end()
  1312.                         ->scalarNode('method')
  1313.                             ->defaultNull()
  1314.                         ->end()
  1315.                         ->arrayNode('debug')
  1316.                             ->children()
  1317.                                 ->scalarNode('email_addresses')
  1318.                                     ->defaultValue('')
  1319.                                 ->end()
  1320.                             ->end()
  1321.                         ->end()
  1322.                         ->booleanNode('use_specific')
  1323.                             ->beforeNormalization()
  1324.                                 ->ifString()
  1325.                                 ->then(function ($v) {
  1326.                                     return (bool)$v;
  1327.                                 })
  1328.                             ->end()
  1329.                         ->end()
  1330.                         ->arrayNode('source_adapters')
  1331.                             ->useAttributeAsKey('name')
  1332.                                 ->prototype('scalar')
  1333.                             ->end()
  1334.                         ->end()
  1335.                     ->end()
  1336.                 ->end()
  1337.             ->end();
  1338.     }
  1339.     /**
  1340.      * Adds configuration tree for custom report adapters
  1341.      *
  1342.      * @param ArrayNodeDefinition $rootNode
  1343.      */
  1344.     private function addCustomReportsNode(ArrayNodeDefinition $rootNode)
  1345.     {
  1346.         $rootNode
  1347.             ->children()
  1348.                 ->arrayNode('custom_report')
  1349.                     ->addDefaultsIfNotSet()
  1350.                     ->children()
  1351.                         ->arrayNode('definitions')
  1352.                                 ->normalizeKeys(false)
  1353.                                 ->prototype('array')
  1354.                                     ->children()
  1355.                                         ->scalarNode('id')->end()
  1356.                                         ->scalarNode('name')->end()
  1357.                                         ->scalarNode('niceName')->end()
  1358.                                         ->scalarNode('sql')->end()
  1359.                                         ->scalarNode('group')->end()
  1360.                                         ->scalarNode('groupIconClass')->end()
  1361.                                         ->scalarNode('iconClass')->end()
  1362.                                         ->booleanNode('menuShortcut')->end()
  1363.                                         ->scalarNode('reportClass')->end()
  1364.                                         ->scalarNode('chartType')->end()
  1365.                                         ->scalarNode('pieColumn')->end()
  1366.                                         ->scalarNode('pieLabelColumn')->end()
  1367.                                         ->variableNode('xAxis')->end()
  1368.                                         ->variableNode('yAxis')->end()
  1369.                                         ->integerNode('modificationDate')->end()
  1370.                                         ->integerNode('creationDate')->end()
  1371.                                         ->booleanNode('shareGlobally')->end()
  1372.                                         ->variableNode('sharedUserNames')->end()
  1373.                                         ->variableNode('sharedRoleNames')->end()
  1374.                                         ->arrayNode('dataSourceConfig')
  1375.                                             ->prototype('variable')
  1376.                                             ->end()
  1377.                                         ->end()
  1378.                                         ->arrayNode('columnConfiguration')
  1379.                                             ->prototype('variable')
  1380.                                             ->end()
  1381.                                         ->end()
  1382.                                     ->end()
  1383.                                 ->end()
  1384.                         ->end()
  1385.                         ->arrayNode('adapters')
  1386.                             ->useAttributeAsKey('name')
  1387.                                 ->prototype('scalar')
  1388.                             ->end()
  1389.                         ->end()
  1390.                     ->end()
  1391.                 ->end()
  1392.             ->end();
  1393.     }
  1394.     private function addTargetingNode(ArrayNodeDefinition $rootNode)
  1395.     {
  1396.         $rootNode
  1397.             ->children()
  1398.                 ->arrayNode('targeting')
  1399.                     ->canBeDisabled()
  1400.                     ->addDefaultsIfNotSet()
  1401.                     ->children()
  1402.                         ->scalarNode('storage_id')
  1403.                             ->info('Service ID of the targeting storage which should be used. This ID will be aliased to ' TargetingStorageInterface::class)
  1404.                             ->defaultValue(CookieStorage::class)
  1405.                             ->cannotBeEmpty()
  1406.                         ->end()
  1407.                         ->arrayNode('session')
  1408.                             ->info('Enables HTTP session support by configuring session bags and the full page cache')
  1409.                             ->canBeEnabled()
  1410.                         ->end()
  1411.                         ->arrayNode('data_providers')
  1412.                             ->useAttributeAsKey('key')
  1413.                                 ->prototype('scalar')
  1414.                             ->end()
  1415.                         ->end()
  1416.                         ->arrayNode('conditions')
  1417.                             ->useAttributeAsKey('key')
  1418.                                 ->prototype('scalar')
  1419.                             ->end()
  1420.                         ->end()
  1421.                         ->arrayNode('action_handlers')
  1422.                             ->useAttributeAsKey('name')
  1423.                                 ->prototype('scalar')
  1424.                             ->end()
  1425.                         ->end()
  1426.                     ->end()
  1427.                 ->end()
  1428.             ->end();
  1429.     }
  1430.     private function addSitemapsNode(ArrayNodeDefinition $rootNode)
  1431.     {
  1432.         $rootNode
  1433.             ->children()
  1434.                 ->arrayNode('sitemaps')
  1435.                     ->addDefaultsIfNotSet()
  1436.                     ->children()
  1437.                         ->arrayNode('generators')
  1438.                             ->useAttributeAsKey('name')
  1439.                             ->prototype('array')
  1440.                                 ->beforeNormalization()
  1441.                                     ->ifString()
  1442.                                     ->then(function ($v) {
  1443.                                         return [
  1444.                                             'enabled' => true,
  1445.                                             'generator_id' => $v,
  1446.                                             'priority' => 0,
  1447.                                         ];
  1448.                                     })
  1449.                                 ->end()
  1450.                                 ->addDefaultsIfNotSet()
  1451.                                 ->canBeDisabled()
  1452.                                 ->children()
  1453.                                     ->scalarNode('generator_id')
  1454.                                         ->cannotBeEmpty()
  1455.                                     ->end()
  1456.                                     ->integerNode('priority')
  1457.                                         ->defaultValue(0)
  1458.                                     ->end()
  1459.                                 ->end()
  1460.                             ->end()
  1461.                         ->end()
  1462.                     ->end()
  1463.                 ->end()
  1464.             ->end()
  1465.         ->end();
  1466.     }
  1467.     private function addWorkflowNode(ArrayNodeDefinition $rootNode)
  1468.     {
  1469.         $rootNode
  1470.             ->children()
  1471.                  ->arrayNode('workflows')
  1472.                         ->useAttributeAsKey('name')
  1473.                         ->prototype('array')
  1474.                             ->children()
  1475.                                 ->arrayNode('placeholders')
  1476.                                     ->info('Placeholder values in this workflow configuration (locale: "%%locale%%") will be replaced by the given placeholder value (eg. "de_AT")')
  1477.                                     ->example([
  1478.                                         'placeholders' => [
  1479.                                             '%%locale%%' => 'de_AT',
  1480.                                         ],
  1481.                                     ])
  1482.                                     ->defaultValue([])
  1483.                                     ->beforeNormalization()
  1484.                                         ->castToArray()
  1485.                                         ->always()
  1486.                                         ->then(function ($placeholders) {
  1487.                                             $this->placeholders $placeholders;
  1488.                                             return $placeholders;
  1489.                                         })
  1490.                                     ->end()
  1491.                                     ->prototype('scalar')->end()
  1492.                                 ->end()
  1493.                                 ->arrayNode('custom_extensions')->ignoreExtraKeys(false)->info('Use this key to attach additional config information to a workflow, for example via bundles, etc.')->end()
  1494.                                 ->booleanNode('enabled')
  1495.                                     ->defaultTrue()
  1496.                                     ->info('Can be used to enable or disable the workflow.')
  1497.                                 ->end()
  1498.                                 ->integerNode('priority')
  1499.                                     ->defaultValue(0)
  1500.                                     ->info('When multiple custom view or permission settings from different places in different workflows are valid, the workflow with the highest priority will be used.')
  1501.                                 ->end()
  1502.                                 ->scalarNode('label')
  1503.                                     ->info('Will be used in the backend interface as nice name for the workflow. If not set the technical workflow name will be used as label too.')
  1504.                                 ->end()
  1505.                                 ->arrayNode('audit_trail')
  1506.                                     ->canBeEnabled()
  1507.                                     ->info('Enable default audit trail feature provided by Symfony. Take a look at the Symfony docs for more details.')
  1508.                                 ->end()
  1509.                                 ->enumNode('type')
  1510.                                     ->values(['workflow''state_machine'])
  1511.                                     ->info('A workflow with type "workflow" can handle multiple places at one time whereas a state_machine provides a finite state_machine (only one place at one time). Take a look at the Symfony docs for more details.')
  1512.                                 ->end()
  1513.                                 ->arrayNode('marking_store')
  1514.                                     ->fixXmlConfig('argument')
  1515.                                     ->children()
  1516.                                         ->enumNode('type')
  1517.                                             ->values(['multiple_state''single_state''state_table''data_object_multiple_state''data_object_splitted_state'])
  1518.                                         ->end()
  1519.                                         ->arrayNode('arguments')
  1520.                                             ->beforeNormalization()
  1521.                                                 ->always()
  1522.                                                 ->then(function ($arguments) {
  1523.                                                     if (is_string($arguments)) {
  1524.                                                         $arguments = [$arguments];
  1525.                                                     }
  1526.                                                     if (!empty($this->placeholders)) {
  1527.                                                         $arguments $this->placeholderProcessor->mergePlaceholders($arguments$this->placeholders);
  1528.                                                     }
  1529.                                                     return $arguments;
  1530.                                                 })
  1531.                                             ->end()
  1532.                                             ->requiresAtLeastOneElement()
  1533.                                             ->prototype('variable')
  1534.                                             ->end()
  1535.                                         ->end()
  1536.                                         ->scalarNode('service')
  1537.                                             ->cannotBeEmpty()
  1538.                                         ->end()
  1539.                                     ->end()
  1540.                                     ->info('Handles the way how the state/place is stored. If not defined "state_table" will be used as default. Take a look at @TODO for a description of the different types.')
  1541.                                     ->validate()
  1542.                                         ->ifTrue(function ($v) {
  1543.                                             return isset($v['type']) && isset($v['service']);
  1544.                                         })
  1545.                                         ->thenInvalid('"type" and "service" cannot be used together.')
  1546.                                     ->end()
  1547.                                     ->validate()
  1548.                                         ->ifTrue(function ($v) {
  1549.                                             return !empty($v['arguments']) && isset($v['service']);
  1550.                                         })
  1551.                                         ->thenInvalid('"arguments" and "service" cannot be used together.')
  1552.                                     ->end()
  1553.                                 ->end()
  1554.                                 ->arrayNode('supports')
  1555.                                     ->beforeNormalization()
  1556.                                         ->ifString()
  1557.                                         ->then(function ($v) {
  1558.                                             return [$v];
  1559.                                         })
  1560.                                     ->end()
  1561.                                     ->prototype('scalar')
  1562.                                         ->cannotBeEmpty()
  1563.                                     ->end()
  1564.                                     ->info('List of supported entity classes. Take a look at the Symfony docs for more details.')
  1565.                                     ->example(['\Pimcore\Model\DataObject\Product'])
  1566.                                 ->end()
  1567.                                 ->arrayNode('support_strategy')
  1568.                                     ->fixXmlConfig('argument')
  1569.                                     ->children()
  1570.                                         ->enumNode('type')
  1571.                                             ->values(['expression'])
  1572.                                             ->info('Type "expression": a symfony expression to define a criteria.')
  1573.                                         ->end()
  1574.                                         ->arrayNode('arguments')
  1575.                                             ->beforeNormalization()
  1576.                                                 ->ifString()
  1577.                                                 ->then(function ($v) {
  1578.                                                     return [$v];
  1579.                                                 })
  1580.                                             ->end()
  1581.                                             ->requiresAtLeastOneElement()
  1582.                                             ->prototype('variable')
  1583.                                             ->end()
  1584.                                         ->end()
  1585.                                         ->scalarNode('service')
  1586.                                             ->cannotBeEmpty()
  1587.                                             ->info('Define a custom service to handle the logic. Take a look at the Symfony docs for more details.')
  1588.                                         ->end()
  1589.                                     ->end()
  1590.                                     ->validate()
  1591.                                         ->ifTrue(function ($v) {
  1592.                                             return isset($v['type']) && isset($v['service']);
  1593.                                         })
  1594.                                         ->thenInvalid('"type" and "service" cannot be used together.')
  1595.                                     ->end()
  1596.                                     ->validate()
  1597.                                         ->ifTrue(function ($v) {
  1598.                                             return !empty($v['arguments']) && isset($v['service']);
  1599.                                         })
  1600.                                         ->thenInvalid('"arguments" and "service" cannot be used together.')
  1601.                                     ->end()
  1602.                                     ->info('Can be used to implement a special logic which subjects are supported by the workflow. For example only products matching certain criteria.')
  1603.                                     ->example([
  1604.                                         'type' => 'expression',
  1605.                                         'arguments' => [
  1606.                                             '\Pimcore\Model\DataObject\Product',
  1607.                                             'subject.getProductType() == "article" and is_fully_authenticated() and "ROLE_PIMCORE_ADMIN" in roles',
  1608.                                         ],
  1609.                                     ])
  1610.                                 ->end()
  1611.                                 ->arrayNode('initial_markings')
  1612.                                     ->info('Can be used to set the initial places (markings) for a workflow. Note that this option is Symfony 4.3+ only')
  1613.                                     ->beforeNormalization()
  1614.                                         ->ifString()
  1615.                                             ->then(function ($v) {
  1616.                                                 return [$v];
  1617.                                             })
  1618.                                         ->end()
  1619.                                         ->requiresAtLeastOneElement()
  1620.                                         ->prototype('scalar')
  1621.                                         ->cannotBeEmpty()
  1622.                                     ->end()
  1623.                                 ->end()
  1624.                                 ->arrayNode('places')
  1625.                                     ->prototype('array')
  1626.                                         ->children()
  1627.                                             ->scalarNode('label')->info('Nice name which will be used in the Pimcore backend.')->end()
  1628.                                             ->scalarNode('title')->info('Title/tooltip for this place when it is displayed in the header of the Pimcore element detail view in the backend.')->defaultValue('')->end()
  1629.                                             ->scalarNode('color')->info('Color of the place which will be used in the Pimcore backend.')->defaultValue('#bfdadc')->end()
  1630.                                             ->booleanNode('colorInverted')->info('If set to true the color will be used as border and font color otherwise as background color.')->defaultFalse()->end()
  1631.                                             ->booleanNode('visibleInHeader')->info('If set to false, the place will be hidden in the header of the Pimcore element detail view in the backend.')->defaultTrue()->end()
  1632.                                             ->arrayNode('permissions')
  1633.                                                 ->prototype('array')
  1634.                                                     ->children()
  1635.                                                         ->scalarNode('condition')->info('A symfony expression can be configured here. The first set of permissions which are matching the condition will be used.')->end()
  1636.                                                         ->booleanNode('save')->info('save permission as it can be configured in Pimcore workplaces')->end()
  1637.                                                         ->booleanNode('publish')->info('publish permission as it can be configured in Pimcore workplaces')->end()
  1638.                                                         ->booleanNode('unpublish')->info('unpublish permission as it can be configured in Pimcore workplaces')->end()
  1639.                                                         ->booleanNode('delete')->info('delete permission as it can be configured in Pimcore workplaces')->end()
  1640.                                                         ->booleanNode('rename')->info('rename permission as it can be configured in Pimcore workplaces')->end()
  1641.                                                         ->booleanNode('view')->info('view permission as it can be configured in Pimcore workplaces')->end()
  1642.                                                         ->booleanNode('settings')->info('settings permission as it can be configured in Pimcore workplaces')->end()
  1643.                                                         ->booleanNode('versions')->info('versions permission as it can be configured in Pimcore workplaces')->end()
  1644.                                                         ->booleanNode('properties')->info('properties permission as it can be configured in Pimcore workplaces')->end()
  1645.                                                         ->booleanNode('modify')->info('a short hand for save, publish, unpublish, delete + rename')->end()
  1646.                                                         ->scalarNode('objectLayout')->info('if set, the user will see the configured custom data object layout')->end()
  1647.                                                     ->end()
  1648.                                                 ->end()
  1649.                                             ->end()
  1650.                                         ->end()
  1651.                                     ->end()
  1652.                                     ->beforeNormalization()
  1653.                                         ->always()
  1654.                                         ->then(function ($places) {
  1655.                                             if (!empty($this->placeholders)) {
  1656.                                                 foreach ($places as $name => $place) {
  1657.                                                     $places[$name] = $this->placeholderProcessor->mergePlaceholders($place$this->placeholders);
  1658.                                                 }
  1659.                                             }
  1660.                                             return $places;
  1661.                                         })
  1662.                                     ->end()
  1663.                                     ->example([
  1664.                                         'places' => [
  1665.                                             'closed' => [
  1666.                                                 'label' => 'close product',
  1667.                                                 'permissions' => [
  1668.                                                     [
  1669.                                                         'condition' => "is_fully_authenticated() and 'ROLE_PIMCORE_ADMIN' in roles",
  1670.                                                         'modify' => false,
  1671.                                                     ],
  1672.                                                     [
  1673.                                                         'modify' => false,
  1674.                                                         'objectLayout' => 2,
  1675.                                                     ],
  1676.                                                 ],
  1677.                                             ],
  1678.                                         ],
  1679.                                     ])
  1680.                                 ->end()
  1681.                                 ->arrayNode('transitions')
  1682.                                     ->beforeNormalization()
  1683.                                         ->always()
  1684.                                         ->then(function ($transitions) {
  1685.                                             // It's an indexed array, we let the validation occurs
  1686.                                             if (isset($transitions[0])) {
  1687.                                                 return $transitions;
  1688.                                             }
  1689.                                             foreach ($transitions as $name => $transition) {
  1690.                                                 if (array_key_exists('name', (array) $transition)) {
  1691.                                                     continue;
  1692.                                                 }
  1693.                                                 $transition['name'] = $name;
  1694.                                                 $transitions[$name] = $transition;
  1695.                                             }
  1696.                                             return $transitions;
  1697.                                         })
  1698.                                     ->end()
  1699.                                     ->isRequired()
  1700.                                     ->requiresAtLeastOneElement()
  1701.                                     ->prototype('array')
  1702.                                         ->children()
  1703.                                             ->scalarNode('name')
  1704.                                                 ->isRequired()
  1705.                                                 ->cannotBeEmpty()
  1706.                                             ->end()
  1707.                                             ->scalarNode('guard')
  1708.                                                 ->cannotBeEmpty()
  1709.                                                 ->info('An expression to block the transition')
  1710.                                                 ->example('is_fully_authenticated() and has_role(\'ROLE_JOURNALIST\') and subject.getTitle() == \'My first article\'')
  1711.                                             ->end()
  1712.                                             ->arrayNode('from')
  1713.                                                 ->beforeNormalization()
  1714.                                                     ->ifString()
  1715.                                                     ->then(function ($v) {
  1716.                                                         return [$v];
  1717.                                                     })
  1718.                                                 ->end()
  1719.                                                 ->requiresAtLeastOneElement()
  1720.                                                 ->prototype('scalar')
  1721.                                                     ->cannotBeEmpty()
  1722.                                                 ->end()
  1723.                                             ->end()
  1724.                                             ->arrayNode('to')
  1725.                                                 ->beforeNormalization()
  1726.                                                     ->ifString()
  1727.                                                     ->then(function ($v) {
  1728.                                                         return [$v];
  1729.                                                     })
  1730.                                                 ->end()
  1731.                                                 ->requiresAtLeastOneElement()
  1732.                                                 ->prototype('scalar')
  1733.                                                     ->cannotBeEmpty()
  1734.                                                 ->end()
  1735.                                             ->end()
  1736.                                             ->arrayNode('options')
  1737.                                                 ->children()
  1738.                                                     ->scalarNode('label')->info('Nice name for the Pimcore backend.')->end()
  1739.                                                     ->arrayNode('notes')
  1740.                                                         ->children()
  1741.                                                             ->booleanNode('commentEnabled')->defaultFalse()->info('If enabled a detail window will open when the user executes the transition. In this detail view the user be asked to enter a "comment". This comment then will be used as comment for the notes/events feature.')->end()
  1742.                                                             ->booleanNode('commentRequired')->defaultFalse()->info('Set this to true if the comment should be a required field.')->end()
  1743.                                                             ->scalarNode('commentSetterFn')->info('Can be used for data objects. The comment will be saved to the data object additionally to the notes/events through this setter function.')->end()
  1744.                                                             ->scalarNode('commentGetterFn')->info('Can be used for data objects to prefill the comment field with data from the data object.')->end()
  1745.                                                             ->scalarNode('type')->defaultValue('Status update')->info('Set\'s the type string in the saved note.')->end()
  1746.                                                             ->scalarNode('title')->info('An optional alternative "title" for the note, if blank the actions transition result is used.')->end()
  1747.                                                             ->arrayNode('additionalFields')
  1748.                                                                 ->prototype('array')
  1749.                                                                     ->children()
  1750.                                                                         ->scalarNode('name')->isRequired()->info('The technical name used in the input form.')->end()
  1751.                                                                         ->enumNode('fieldType')
  1752.                                                                             ->isRequired()
  1753.                                                                             ->values(['input''numeric''textarea''select''datetime''date''user''checkbox'])
  1754.                                                                             ->info('The data component name/field type.')
  1755.                                                                         ->end()
  1756.                                                                         ->scalarNode('title')->info('The label used by the field')->end()
  1757.                                                                         ->booleanNode('required')->defaultFalse()->info('Whether or not the field is required.')->end()
  1758.                                                                         ->scalarNode('setterFn')->info('Optional setter function (available in the element, for example in the updated object), if not specified, data will be added to notes. The Workflow manager will call the function with the whole field data.')->end()
  1759.                                                                         ->arrayNode('fieldTypeSettings')
  1760.                                                                              ->prototype('variable')->end()
  1761.                                                                              ->info('Will be passed to the underlying Pimcore data object field type. Can be used to configure the options of a select box for example.')
  1762.                                                                         ->end()
  1763.                                                                     ->end()
  1764.                                                                 ->end()
  1765.                                                                 ->info('Add additional field to the transition detail window.')
  1766.                                                             ->end()
  1767.                                                             ->arrayNode('customHtml')
  1768.                                                                 ->children()
  1769.                                                                     ->enumNode('position')
  1770.                                                                         ->values(['top''center''bottom'])
  1771.                                                                         ->defaultValue('top')
  1772.                                                                         ->info('Set position of custom HTML inside modal (top, center, bottom).')
  1773.                                                                     ->end()
  1774.                                                                     ->scalarNode('service')
  1775.                                                                         ->cannotBeEmpty()
  1776.                                                                         ->info('Define a custom service for rendering custom HTML within the note modal.')
  1777.                                                                     ->end()
  1778.                                                                 ->end()
  1779.                                                             ->end()
  1780.                                                         ->end()
  1781.                                                     ->end()
  1782.                                                     ->scalarNode('iconClass')->info('Css class to define the icon which will be used in the actions button in the backend.')->end()
  1783.                                                     ->scalarNode('objectLayout')->defaultValue(false)->info('Forces an object layout after the transition was performed. This objectLayout setting overrules all objectLayout settings within the places configs.')->end()
  1784.                                                     ->arrayNode('notificationSettings')
  1785.                                                         ->prototype('array')
  1786.                                                             ->children()
  1787.                                                                 ->scalarNode('condition')->info('A symfony expression can be configured here. All sets of notification which are matching the condition will be used.')->end()
  1788.                                                                 ->arrayNode('notifyUsers')
  1789.                                                                     ->prototype('scalar')
  1790.                                                                         ->cannotBeEmpty()
  1791.                                                                     ->end()
  1792.                                                                     ->info('Send an email notification to a list of users (user names) when the transition get\'s applied')
  1793.                                                                 ->end()
  1794.                                                                 ->arrayNode('notifyRoles')
  1795.                                                                     ->prototype('scalar')
  1796.                                                                         ->cannotBeEmpty()
  1797.                                                                     ->end()
  1798.                                                                     ->info('Send an email notification to a list of user roles (role names) when the transition get\'s applied')
  1799.                                                                 ->end()
  1800.                                                                 ->arrayNode('channelType')
  1801.                                                                     ->requiresAtLeastOneElement()
  1802.                                                                     ->enumPrototype()
  1803.                                                                         ->values([NotificationSubscriber::NOTIFICATION_CHANNEL_MAILNotificationSubscriber::NOTIFICATION_CHANNEL_PIMCORE_NOTIFICATION])
  1804.                                                                         ->cannotBeEmpty()
  1805.                                                                         ->defaultValue(NotificationSubscriber::NOTIFICATION_CHANNEL_MAIL)
  1806.                                                                     ->end()
  1807.                                                                     ->info('Define which channel notification should be sent to, possible values "' NotificationSubscriber::NOTIFICATION_CHANNEL_MAIL '" and "' NotificationSubscriber::NOTIFICATION_CHANNEL_PIMCORE_NOTIFICATION '", default value is "' NotificationSubscriber::NOTIFICATION_CHANNEL_MAIL '".')
  1808.                                                                     ->addDefaultChildrenIfNoneSet()
  1809.                                                                 ->end()
  1810.                                                                 ->enumNode('mailType')
  1811.                                                                     ->values([NotificationSubscriber::MAIL_TYPE_TEMPLATENotificationSubscriber::MAIL_TYPE_DOCUMENT])
  1812.                                                                     ->defaultValue(NotificationSubscriber::MAIL_TYPE_TEMPLATE)
  1813.                                                                     ->info('Type of mail source.')
  1814.                                                                 ->end()
  1815.                                                                 ->scalarNode('mailPath')
  1816.                                                                     ->defaultValue(NotificationSubscriber::DEFAULT_MAIL_TEMPLATE_PATH)
  1817.                                                                     ->info('Path to mail source - either Symfony path to template or fullpath to Pimcore document. Optional use ' NotificationEmailService::MAIL_PATH_LANGUAGE_PLACEHOLDER ' as placeholder for language.')
  1818.                                                                 ->end()
  1819.                                                             ->end()
  1820.                                                         ->end()
  1821.                                                     ->end()
  1822.                                                     ->enumNode('changePublishedState')
  1823.                                                         ->values([ChangePublishedStateSubscriber::NO_CHANGEChangePublishedStateSubscriber::FORCE_UNPUBLISHEDChangePublishedStateSubscriber::FORCE_PUBLISHEDChangePublishedStateSubscriber::SAVE_VERSION])
  1824.                                                         ->defaultValue(ChangePublishedStateSubscriber::NO_CHANGE)
  1825.                                                         ->info('Change published state of element while transition (only available for documents and data objects).')
  1826.                                                     ->end()
  1827.                                                 ->end()
  1828.                                             ->end()
  1829.                                         ->end()
  1830.                                     ->end()
  1831.                                     ->example([
  1832.                                         'close_product' => [
  1833.                                             'from' => 'open',
  1834.                                             'to' => 'closed',
  1835.                                             'options' => [
  1836.                                                 'label' => 'close product',
  1837.                                                 'notes' => [
  1838.                                                     'commentEnabled' => true,
  1839.                                                     'commentRequired' => true,
  1840.                                                     'additionalFields' => [
  1841.                                                         [
  1842.                                                             'name' => 'accept',
  1843.                                                             'title' => 'accept terms',
  1844.                                                             'required' => true,
  1845.                                                             'fieldType' => 'checkbox',
  1846.                                                         ],
  1847.                                                         [
  1848.                                                             'name' => 'select',
  1849.                                                             'title' => 'please select a type',
  1850.                                                             'setterFn' => 'setSpecialWorkflowType',
  1851.                                                             'fieldType' => 'select',
  1852.                                                             'fieldTypeSettings' => [
  1853.                                                                 'options' => [
  1854.                                                                     ['key' => 'Option A''value' => 'a'],
  1855.                                                                     ['key' => 'Option B''value' => 'b'],
  1856.                                                                     ['key' => 'Option C''value' => 'c'],
  1857.                                                                 ],
  1858.                                                             ],
  1859.                                                         ],
  1860.                                                     ],
  1861.                                                 ],
  1862.                                             ],
  1863.                                         ],
  1864.                                     ])
  1865.                                 ->end()
  1866.                                 ->arrayNode('globalActions')
  1867.                                     ->prototype('array')
  1868.                                         ->children()
  1869.                                             ->scalarNode('label')->info('Nice name for the Pimcore backend.')->end()
  1870.                                             ->scalarNode('iconClass')->info('Css class to define the icon which will be used in the actions button in the backend.')->end()
  1871.                                             ->scalarNode('objectLayout')->defaultValue(false)->info('Forces an object layout after the global action was performed. This objectLayout setting overrules all objectLayout settings within the places configs.')->end()
  1872.                                             ->scalarNode('guard')
  1873.                                                 ->cannotBeEmpty()
  1874.                                                 ->info('An expression to block the action')
  1875.                                                 ->example('is_fully_authenticated() and has_role(\'ROLE_JOURNALIST\') and subject.getTitle() == \'My first article\'')
  1876.                                             ->end()
  1877.                                             ->arrayNode('to')
  1878.                                                 ->beforeNormalization()
  1879.                                                     ->ifString()
  1880.                                                     ->then(function ($v) {
  1881.                                                         return [$v];
  1882.                                                     })
  1883.                                                 ->end()
  1884.                                                 ->requiresAtLeastOneElement()
  1885.                                                 ->prototype('scalar')
  1886.                                                     ->cannotBeEmpty()
  1887.                                                 ->end()
  1888.                                                 ->info('Optionally set the current place of the workflow. Can be used for example to reset the workflow to the initial place.')
  1889.                                             ->end()
  1890.                                             ->arrayNode('notes')
  1891.                                                 ->children()
  1892.                                                     ->booleanNode('commentEnabled')->defaultFalse()->end()
  1893.                                                     ->booleanNode('commentRequired')->defaultFalse()->end()
  1894.                                                     ->scalarNode('commentSetterFn')->end()
  1895.                                                     ->scalarNode('commentGetterFn')->end()
  1896.                                                     ->scalarNode('type')->defaultValue('Status update')->end()
  1897.                                                     ->scalarNode('title')->end()
  1898.                                                     ->arrayNode('additionalFields')
  1899.                                                         ->prototype('array')
  1900.                                                             ->children()
  1901.                                                                 ->scalarNode('name')->isRequired()->end()
  1902.                                                                 ->enumNode('fieldType')
  1903.                                                                     ->isRequired()
  1904.                                                                     ->values(['input''textarea''select''datetime''date''user''checkbox'])
  1905.                                                                 ->end()
  1906.                                                                 ->scalarNode('title')->end()
  1907.                                                                 ->booleanNode('required')->defaultFalse()->end()
  1908.                                                                 ->scalarNode('setterFn')->end()
  1909.                                                                 ->arrayNode('fieldTypeSettings')
  1910.                                                                      ->prototype('variable')->end()
  1911.                                                                 ->end()
  1912.                                                             ->end()
  1913.                                                         ->end()
  1914.                                                     ->end()
  1915.                                                     ->arrayNode('customHtml')
  1916.                                                         ->children()
  1917.                                                             ->enumNode('position')
  1918.                                                                 ->values(['top''center''bottom'])
  1919.                                                                 ->defaultValue('top')
  1920.                                                                 ->info('Set position of custom HTML inside modal (top, center, bottom).')
  1921.                                                             ->end()
  1922.                                                             ->scalarNode('service')
  1923.                                                                 ->cannotBeEmpty()
  1924.                                                                 ->info('Define a custom service for rendering custom HTML within the note modal.')
  1925.                                                             ->end()
  1926.                                                         ->end()
  1927.                                                     ->end()
  1928.                                                 ->end()
  1929.                                                 ->info('See notes section of transitions. It works exactly the same way.')
  1930.                                             ->end()
  1931.                                         ->end()
  1932.                                     ->end()
  1933.                                     ->info('Actions which will be added to actions button independently of the current workflow place.')
  1934.                                 ->end()
  1935.                             ->end()
  1936.                             ->validate()
  1937.                                 ->ifTrue(function ($v) {
  1938.                                     return $v['supports'] && isset($v['support_strategy']);
  1939.                                 })
  1940.                                 ->thenInvalid('"supports" and "support_strategy" cannot be used together.')
  1941.                             ->end()
  1942.                             ->validate()
  1943.                                 ->ifTrue(function ($v) {
  1944.                                     return !$v['supports'] && !isset($v['support_strategy']);
  1945.                                 })
  1946.                                 ->thenInvalid('"supports" or "support_strategy" should be configured.')
  1947.                             ->end()
  1948.                             ->validate()
  1949.                                 ->ifTrue(function ($v) {
  1950.                                     if (($v['type'] ?? 'workflow') === 'state_machine') {
  1951.                                         foreach ($v['transitions'] ?? [] as $transition) {
  1952.                                             if (count($transition['to']) > 1) {
  1953.                                                 return true;
  1954.                                             }
  1955.                                         }
  1956.                                         foreach ($v['globalActions'] ?? [] as $transition) {
  1957.                                             if (count($transition['to']) > 1) {
  1958.                                                 return true;
  1959.                                             }
  1960.                                         }
  1961.                                     }
  1962.                                     return false;
  1963.                                 })
  1964.                                 ->thenInvalid('Type `state_machine` does not support multiple `to` definitions for transitions and global actions. Change definition or type to `workflow`.')
  1965.                             ->end()
  1966.                         ->end()
  1967.                     ->end()
  1968.                 ->end()
  1969.                 ->addDefaultsIfNotSet()
  1970.             ->end();
  1971.     }
  1972.     /**
  1973.      * Add predefined properties specific extension config
  1974.      *
  1975.      * @param ArrayNodeDefinition $rootNode
  1976.      */
  1977.     private function addPredefinedPropertiesNode(ArrayNodeDefinition $rootNode)
  1978.     {
  1979.         $predefinedPropertiesNode $rootNode
  1980.             ->children()
  1981.             ->arrayNode('properties')
  1982.             ->ignoreExtraKeys()
  1983.             ->addDefaultsIfNotSet();
  1984.         $predefinedPropertiesNode
  1985.         ->children()
  1986.             ->arrayNode('predefined')
  1987.                 ->addDefaultsIfNotSet()
  1988.                 ->children()
  1989.                     ->arrayNode('definitions')
  1990.                     ->normalizeKeys(false)
  1991.                         ->prototype('array')
  1992.                             ->children()
  1993.                                 ->scalarNode('name')->end()
  1994.                                 ->scalarNode('description')->end()
  1995.                                 ->scalarNode('key')->end()
  1996.                                 ->scalarNode('type')->end()
  1997.                                 ->scalarNode('data')->end()
  1998.                                 ->scalarNode('config')->end()
  1999.                                 ->scalarNode('ctype')->end()
  2000.                                 ->booleanNode('inheritable')
  2001.                                     ->beforeNormalization()
  2002.                                         ->ifString()
  2003.                                         ->then(function ($v) {
  2004.                                             return (bool)$v;
  2005.                                         })
  2006.                                         ->end()
  2007.                                 ->end()
  2008.                                 ->integerNode('creationDate')->end()
  2009.                                 ->integerNode('modificationDate')->end()
  2010.                             ->end()
  2011.                         ->end()
  2012.                     ->end()
  2013.                 ->end()
  2014.             ->end()
  2015.         ->end();
  2016.     }
  2017.     /**
  2018.      * Add static routes specific extension config
  2019.      *
  2020.      * @param ArrayNodeDefinition $rootNode
  2021.      */
  2022.     private function addStaticroutesNode(ArrayNodeDefinition $rootNode)
  2023.     {
  2024.         $rootNode
  2025.         ->children()
  2026.             ->arrayNode('staticroutes')
  2027.                 ->ignoreExtraKeys()
  2028.                 ->addDefaultsIfNotSet()
  2029.                 ->children()
  2030.                     ->arrayNode('definitions')
  2031.                     ->normalizeKeys(false)
  2032.                         ->prototype('array')
  2033.                             ->children()
  2034.                                 ->scalarNode('name')->end()
  2035.                                 ->scalarNode('pattern')->end()
  2036.                                 ->scalarNode('reverse')->end()
  2037.                                 ->scalarNode('controller')->end()
  2038.                                 ->scalarNode('variables')->end()
  2039.                                 ->scalarNode('defaults')->end()
  2040.                                 ->arrayNode('siteId')
  2041.                                     ->integerPrototype()->end()
  2042.                                 ->end()
  2043.                                 ->arrayNode('methods')
  2044.                                     ->scalarPrototype()->end()
  2045.                                 ->end()
  2046.                                 ->integerNode('priority')->end()
  2047.                                 ->integerNode('creationDate')->end()
  2048.                                 ->integerNode('modificationDate')->end()
  2049.                             ->end()
  2050.                         ->end()
  2051.                     ->end()
  2052.                 ->end()
  2053.             ->end()
  2054.         ->end();
  2055.     }
  2056.     /**
  2057.      * Add perspectives specific extension config
  2058.      *
  2059.      * @param ArrayNodeDefinition $rootNode
  2060.      */
  2061.     private function addPerspectivesNode(ArrayNodeDefinition $rootNode)
  2062.     {
  2063.         $rootNode
  2064.             ->children()
  2065.                 ->arrayNode('perspectives')
  2066.                     ->ignoreExtraKeys()
  2067.                     ->addDefaultsIfNotSet()
  2068.                     ->children()
  2069.                         ->arrayNode('definitions')
  2070.                         ->normalizeKeys(false)
  2071.                             ->prototype('array')
  2072.                                 ->children()
  2073.                                     ->scalarNode('iconCls')->end()
  2074.                                     ->scalarNode('icon')->end()
  2075.                                     ->variableNode('toolbar')->end()
  2076.                                     ->arrayNode('dashboards')
  2077.                                         ->children()
  2078.                                             ->variableNode('predefined')->end()
  2079.                                         ->end()
  2080.                                     ->end()
  2081.                                     ->arrayNode('elementTree')
  2082.                                         ->prototype('array')
  2083.                                             ->children()
  2084.                                                 ->scalarNode('type')->end()
  2085.                                                 ->scalarNode('position')->end()
  2086.                                                 ->scalarNode('name')->end()
  2087.                                                 ->booleanNode('expanded')->end()
  2088.                                                 ->scalarNode('hidden')->end()
  2089.                                                 ->integerNode('sort')->end()
  2090.                                                 ->scalarNode('id')->end()
  2091.                                                 ->variableNode('treeContextMenu')->end()
  2092.                                             ->end()
  2093.                                         ->end()
  2094.                                     ->end()
  2095.                                 ->end()
  2096.                             ->end()
  2097.                         ->end()
  2098.                     ->end()
  2099.                 ->end()
  2100.             ->end()
  2101.         ->end();
  2102.     }
  2103.     /**
  2104.      * Add custom views specific extension config
  2105.      *
  2106.      * @param ArrayNodeDefinition $rootNode
  2107.      */
  2108.     private function addCustomViewsNode(ArrayNodeDefinition $rootNode)
  2109.     {
  2110.         $rootNode
  2111.             ->children()
  2112.                 ->arrayNode('custom_views')
  2113.                     ->ignoreExtraKeys()
  2114.                     ->addDefaultsIfNotSet()
  2115.                     ->children()
  2116.                         ->arrayNode('definitions')
  2117.                         ->normalizeKeys(false)
  2118.                             ->prototype('array')
  2119.                             ->children()
  2120.                                 ->scalarNode('id')->end()
  2121.                                 ->scalarNode('treetype')->end()
  2122.                                 ->scalarNode('name')->end()
  2123.                                 ->scalarNode('condition')->end()
  2124.                                 ->scalarNode('icon')->end()
  2125.                                 ->scalarNode('rootfolder')->end()
  2126.                                 ->scalarNode('showroot')->end()
  2127.                                 ->variableNode('classes')->end()
  2128.                                 ->scalarNode('position')->end()
  2129.                                 ->scalarNode('sort')->end()
  2130.                                 ->booleanNode('expanded')->end()
  2131.                                 ->scalarNode('having')->end()
  2132.                                 ->scalarNode('where')->end()
  2133.                                 ->variableNode('treeContextMenu')->end()
  2134.                                 ->arrayNode('joins')
  2135.                                     ->protoType('array')
  2136.                                         ->children()
  2137.                                             ->scalarNode('type')->end()
  2138.                                             ->scalarNode('condition')->end()
  2139.                                             ->variableNode('name')->end()
  2140.                                             ->variableNode('columns')->end()
  2141.                                         ->end()
  2142.                                     ->end()
  2143.                                 ->end()
  2144.                             ->end()
  2145.                         ->end()
  2146.                     ->end()
  2147.                 ->end()
  2148.             ->end();
  2149.     }
  2150. }