diff --git a/core/modules/help_topics/help_topics.module b/core/modules/help_topics/help_topics.module index 5b89b36de1fe93c85c6560d2b59e78657348101b..4e63e952c359bc9b86b9f559e3973cae87fd6e43 100644 --- a/core/modules/help_topics/help_topics.module +++ b/core/modules/help_topics/help_topics.module @@ -24,7 +24,7 @@ function help_topics_help($route_name, RouteMatchInterface $route_match) { $output .= '<dt>' . t('Viewing help topics') . '</dt>'; $output .= '<dd>' . t('The top-level help topics are listed on the main <a href=":help_page">Help page</a>. Links to other topics, including non-top-level help topics, can be found under the "Related" heading when viewing a topic page.', [':help_page' => $help_home]) . '</dd>'; $output .= '<dt>' . t('Providing help topics') . '</dt>'; - $output .= '<dd>' . t("Modules and themes can provide help topics as Twig-file-based plugins in a project sub-directory called <em>help_topics</em>; plugin meta-data is provided in meta tags within each Twig file. Plugin-based help topics provided by modules and themes will automatically be updated when a module or theme is updated. Use the plugins in <em>core/modules/help_topics/help_topics</em> as a guide when writing and formatting a help topic plugin for your theme or module.") . '</dd>'; + $output .= '<dd>' . t("Modules and themes can provide help topics as Twig-file-based plugins in a project sub-directory called <em>help_topics</em>; plugin meta-data is provided in YAML front matter within each Twig file. Plugin-based help topics provided by modules and themes will automatically be updated when a module or theme is updated. Use the plugins in <em>core/modules/help_topics/help_topics</em> as a guide when writing and formatting a help topic plugin for your theme or module.") . '</dd>'; $output .= '<dt>' . t('Translating help topics') . '</dt>'; $output .= '<dd>' . t('The title and body text of help topics provided by contributed modules and themes are translatable using the <a href=":locale_help">Interface Translation module</a>. Topics provided by custom modules and themes are also translatable if they have been viewed at least once in a non-English language, which triggers putting their translatable text into the translation database.', [':locale_help' => $locale_help]) . '</dd>'; $output .= '</dl>'; diff --git a/core/modules/help_topics/help_topics/core.config_basic.html.twig b/core/modules/help_topics/help_topics/core.config_basic.html.twig index 09fcb5b9d0a20f29ef9c1a7a6fbed08ae7a77b3c..ca42399303906b682c2d30d3b3dcc801a09b3b2e 100644 --- a/core/modules/help_topics/help_topics/core.config_basic.html.twig +++ b/core/modules/help_topics/help_topics/core.config_basic.html.twig @@ -1,6 +1,9 @@ -<meta name="help_topic:label" content="Changing basic site settings"/> -<meta name="help_topic:top_level"/> -<meta name="help_topic:related" content="user.security_account_settings"/> +--- +label: 'Changing basic site settings' +top_level: true +related: + - user.security_account_settings +--- {% set regional_url = render_var(url('system.regional_settings')) %} {% set information_url = render_var(url('system.site_information_settings')) %} {% set datetime_url = render_var(url('entity.date_format.collection')) %} diff --git a/core/modules/help_topics/help_topics/core.config_error.html.twig b/core/modules/help_topics/help_topics/core.config_error.html.twig index f91fc2a64e4c5d0f8f17321f828a6d22346df003..a8b840ba84f9bc19ac5f33105a58e8e763dcc6df 100644 --- a/core/modules/help_topics/help_topics/core.config_error.html.twig +++ b/core/modules/help_topics/help_topics/core.config_error.html.twig @@ -1,5 +1,9 @@ -<meta name="help_topic:label" content="Configuring error responses, including 403/404 pages"/> -<meta name="help_topic:related" content="core.config_basic,core.maintenance"/> +--- +label: 'Configuring error responses, including 403/404 pages' +related: + - core.config_basic + - core.maintenance +--- {% set log_settings_url = render_var(url('system.logging_settings')) %} {% set site_settings_url = render_var(url('system.site_information_settings')) %} <h2>{% trans %}Configuring 403/404 pages{% endtrans %}</h2> diff --git a/core/modules/help_topics/help_topics/core.maintenance.html.twig b/core/modules/help_topics/help_topics/core.maintenance.html.twig index 8d86f14fdde3cede822c8df3f6f58250daf5ade1..f713192e4e23b0a05d820c82852448b54617f135 100644 --- a/core/modules/help_topics/help_topics/core.maintenance.html.twig +++ b/core/modules/help_topics/help_topics/core.maintenance.html.twig @@ -1,3 +1,5 @@ -<meta name="help_topic:label" content="Maintaining and troubleshooting your site"/> -<meta name="help_topic:top_level"/> +--- +label: 'Maintaining and troubleshooting your site' +top_level: true +--- <p>{% trans %}The related topics listed here will help you keep your site running and troubleshoot problems.{% endtrans %}</p> diff --git a/core/modules/help_topics/help_topics/core.menu_overview.html.twig b/core/modules/help_topics/help_topics/core.menu_overview.html.twig index 4e27967f94e98ac5db866c531aa735ec62ac17fb..bb4a2218e2a63f178367a3b1ebfba4a38c2dd493 100644 --- a/core/modules/help_topics/help_topics/core.menu_overview.html.twig +++ b/core/modules/help_topics/help_topics/core.menu_overview.html.twig @@ -1,3 +1,5 @@ -<meta name="help_topic:label" content="Defining navigation and URLs"/> -<meta name="help_topic:top_level"/> +--- +label: 'Defining navigation and URLs' +top_level: true +--- <p>{% trans %}The related topics listed here describe how to set up various aspects of site navigation and URLs.{% endtrans %}</p> diff --git a/core/modules/help_topics/help_topics/core.security.html.twig b/core/modules/help_topics/help_topics/core.security.html.twig index 63b12c49373b77947790c53af445571768ba7ba2..97c0ea1079ca9530cbd979203453ad869ba58819 100644 --- a/core/modules/help_topics/help_topics/core.security.html.twig +++ b/core/modules/help_topics/help_topics/core.security.html.twig @@ -1,4 +1,7 @@ -<meta name="help_topic:label" content="Making your site secure"/> -<meta name="help_topic:top_level"/> -<meta name="help_topic:related" content="menu_ui.menu_overview"/> +--- +label: 'Making your site secure' +top_level: true +related: + - menu_ui.menu_overview +--- <p>{% trans %}The topics listed here will help you make and keep your site secure.{% endtrans %}</p> diff --git a/core/modules/help_topics/help_topics/core.ui_accessibility.html.twig b/core/modules/help_topics/help_topics/core.ui_accessibility.html.twig index f93aa4d580ce6e8c94c5fc34d2b693702e47d3f1..d1f30157ee01eb8ebb49703d28aebda961c5f113 100644 --- a/core/modules/help_topics/help_topics/core.ui_accessibility.html.twig +++ b/core/modules/help_topics/help_topics/core.ui_accessibility.html.twig @@ -1,5 +1,8 @@ -<meta name="help_topic:label" content="Accessibility features"/> -<meta name="help_topic:related" content="core.ui_components"/> +--- +label: 'Accessibility features' +related: + - core.ui_components +--- <p>{% trans %}The following features of the administrative user interface may help administrative users with disabilities access your site:{% endtrans %}</p> <dl> <dt>{% trans %}Disabling drag-and-drop functionality{% endtrans %}</dt> diff --git a/core/modules/help_topics/help_topics/core.ui_components.html.twig b/core/modules/help_topics/help_topics/core.ui_components.html.twig index f2b28f45ac9b25dc6dcf7eb9384f8f3841107805..0a14a1b61b4260a0bd9dc6419cb28ff057800b9e 100644 --- a/core/modules/help_topics/help_topics/core.ui_components.html.twig +++ b/core/modules/help_topics/help_topics/core.ui_components.html.twig @@ -1,3 +1,5 @@ -<meta name="help_topic:label" content="Using the administrative interface"/> -<meta name="help_topic:top_level"/> +--- +label: 'Using the administrative interface' +top_level: true +--- <p>{% trans %}The related topics listed here describe various aspects of the administrative interface, and tell how to use them.{% endtrans %}</p> diff --git a/core/modules/help_topics/help_topics/core.ui_contextual.html.twig b/core/modules/help_topics/help_topics/core.ui_contextual.html.twig index 202a285404bbceabeb24c13944db3f5d3206ad43..d269556156b307e708133e2da90ddb7084ab7276 100644 --- a/core/modules/help_topics/help_topics/core.ui_contextual.html.twig +++ b/core/modules/help_topics/help_topics/core.ui_contextual.html.twig @@ -1,5 +1,8 @@ -<meta name="help_topic:label" content="Contextual links"/> -<meta name="help_topic:related" content="core.ui_components"/> +--- +label: 'Contextual links' +related: + - core.ui_components +--- <h2>{% trans %}What are contextual links?{% endtrans %}</h2> <p>{% trans %}<em>Contextual links</em> give users with the <em>Use contextual links</em> permission quick access to administrative tasks related to areas of non-administrative pages. For example, if a page on your site displays a block, the block would have a contextual link that would allow users with permission to configure the block. If the block contains a menu or a view, it would also have a contextual link for editing the menu links or the view. Clicking a contextual link takes you to the related administrative page directly, without needing to navigate through the administrative menu system.{% endtrans %}</p> <h2>{% trans %}Displaying and using contextual links{% endtrans %}</h2> diff --git a/core/modules/help_topics/help_topics/core.ui_tours.html.twig b/core/modules/help_topics/help_topics/core.ui_tours.html.twig index 7edf84283b5a30da083e7654e333caedbba02e5d..4bbaec2209813d5f5d1594941c977e5eccd2c794 100644 --- a/core/modules/help_topics/help_topics/core.ui_tours.html.twig +++ b/core/modules/help_topics/help_topics/core.ui_tours.html.twig @@ -1,5 +1,8 @@ -<meta name="help_topic:label" content="Tours"/> -<meta name="help_topic:related" content="core.ui_components"/> +--- +label: 'Tours' +related: + - core.ui_components +--- <h2>{% trans %}What are tours?{% endtrans %}</h2> <p>{% trans %}The core Tour module provides users with <em>tours</em>, which are guided tours of the administrative interface. Each tour starts on a particular administrative page, and consists of one or more <em>tips</em> that highlight elements of the page, guide you through a workflow, or explain key concepts. Users need <em>Access tour</em> permission to view tours, and JavaScript must be enabled in their browsers.{% endtrans %}</p> <h2>{% trans %}Viewing tours{% endtrans %}</h2> diff --git a/core/modules/help_topics/help_topics/help_topics.help_topic_writing.html.twig b/core/modules/help_topics/help_topics/help_topics.help_topic_writing.html.twig index cf61fc46307d8bc57e494c85b454d6a995c091d2..b5d1bfd992cc26983cd7aee3e9dca31ad912a12f 100644 --- a/core/modules/help_topics/help_topics/help_topics.help_topic_writing.html.twig +++ b/core/modules/help_topics/help_topics/help_topics.help_topic_writing.html.twig @@ -1,5 +1,7 @@ -<meta name="help_topic:label" content="Writing good help"/> -<meta name="help_topic:top_level"/> +--- +label: 'Writing good help' +top_level: true +--- <p>{% trans %}Here are some suggestions for how to make your help topics as useful as possible for readers:{% endtrans %}</p> <ul> <li>{% trans %}Choose short titles. If the topic describes a task, start with a verb in -ing form, like "Writing good help".{% endtrans %}</li> diff --git a/core/modules/help_topics/help_topics/shortcut.ui_shortcuts.html.twig b/core/modules/help_topics/help_topics/shortcut.ui_shortcuts.html.twig index 0228a37acf56157c303e5c19498b5d936de6dff8..832b89facd0f56201b04abadd1e56470af600ff4 100644 --- a/core/modules/help_topics/help_topics/shortcut.ui_shortcuts.html.twig +++ b/core/modules/help_topics/help_topics/shortcut.ui_shortcuts.html.twig @@ -1,5 +1,8 @@ -<meta name="help_topic:label" content="Shortcuts"/> -<meta name="help_topic:related" content="core.ui_components"/> +--- +label: 'Shortcuts' +related: + - core.ui_components +--- <h2>{% trans %}What are shortcuts?{% endtrans %}</h2> <p>{% trans %}<em>Shortcuts</em> are quick links to administrative pages; they are managed by the core Shortcut module. A site can have one or more <em>shortcut sets</em>, which can be shared by one or more users; each set contains one or more shortcuts. Users need <em>Use shortcuts</em> permission to view shortcuts; <em>Edit current shortcut set</em> permission to add, delete, or edit the shortcuts in the set assigned to them; and <em>Select any shortcut set</em> permission to select a different shortcut set when editing their user profile. There is also an <em>Administer shortcuts</em> permission, which allows an administrator to do any of these actions, and also permits assigning shortcut sets to other users.{% endtrans %}</p> <h2>{% trans %}Creating and deleting shortcuts{% endtrans %}</h2> diff --git a/core/modules/help_topics/help_topics/user.security_account_settings.html.twig b/core/modules/help_topics/help_topics/user.security_account_settings.html.twig index ef2d25a7577fb85c1450f0859789b78286fbdacd..fbf5052936d48deff1979405f9a65b5d348e2b8e 100644 --- a/core/modules/help_topics/help_topics/user.security_account_settings.html.twig +++ b/core/modules/help_topics/help_topics/user.security_account_settings.html.twig @@ -1,5 +1,8 @@ -<meta name="help_topic:label" content="Defining how user accounts are created"/> -<meta name="help_topic:related" content="core.security"/> +--- +label: 'Defining how user accounts are created' +related: + - core.security +--- {% set account_settings_url = render_var(url('entity.user.admin_form')) %} <p>{% trans %}On the <a href="{{ account_settings_url }}"><em>Account settings</em></a> page, which you can reach from the <em>Manage</em> administrative menu, by navigating to <em>Configuration</em> > <em>People</em> > <em>Account settings</em> (requires the <em>Administer account settings</em> permission), you can configure several settings related to how user accounts are created:{% endtrans %}</p> <ul> diff --git a/core/modules/help_topics/src/FrontMatter.php b/core/modules/help_topics/src/FrontMatter.php new file mode 100644 index 0000000000000000000000000000000000000000..4d95f268eb66762a4a4210be36fb09222dd23545 --- /dev/null +++ b/core/modules/help_topics/src/FrontMatter.php @@ -0,0 +1,173 @@ +<?php + +namespace Drupal\help_topics; + +/** + * Extracts Front Matter from the beginning of a source. + * + * @internal + * This front matter extractor only supports help topic discovery and is not + * part of the public API. + */ +final class FrontMatter { + + /** + * The separator used to indicate front matter data. + * + * @var string + */ + const FRONT_MATTER_SEPARATOR = '---'; + + /** + * The regular expression used to extract the YAML front matter content. + * + * @var string + */ + const FRONT_MATTER_REGEXP = "{^(?:" . self::FRONT_MATTER_SEPARATOR . ")[\r\n|\n]*(.*?)[\r\n|\n]+(?:" . self::FRONT_MATTER_SEPARATOR . ")[\r\n|\n]*(.*)$}s"; + + /** + * The parsed source. + * + * @var array + */ + protected $parsed; + + /** + * A serializer class. + * + * @var string + */ + protected $serializer; + + /** + * The source. + * + * @var string + */ + protected $source; + + /** + * FrontMatter constructor. + * + * @param string $source + * A string source. + * @param string $serializer + * A class that implements + * \Drupal\Component\Serialization\SerializationInterface. + */ + public function __construct($source, $serializer = '\Drupal\Component\Serialization\Yaml') { + assert(is_string($source), '$source must be a string'); + assert(is_string($serializer), '$serializer must be a string'); + if (!is_subclass_of($serializer, '\Drupal\Component\Serialization\SerializationInterface')) { + throw new \InvalidArgumentException('The $serializer parameter must reference a class that implements \Drupal\Component\Serialization\SerializationInterface.'); + } + $this->serializer = $serializer; + $this->source = $source; + } + + /** + * Creates a new FrontMatter instance. + * + * @param string $source + * A string source. + * @param string $serializer + * A class that implements + * \Drupal\Component\Serialization\SerializationInterface. + * + * @return static + */ + public static function load($source, $serializer = '\Drupal\Component\Serialization\Yaml') { + return new static($source, $serializer); + } + + /** + * Parses the source. + * + * @return array + * An associative array containing: + * - code: The real source code. + * - data: The front matter data extracted and decoded. + * - line: The line number where the real source code starts. + * + * @throws \Drupal\Component\Serialization\Exception\InvalidDataTypeException + * Exception thrown when the Front Matter cannot be parsed. + */ + private function parse() { + if (!$this->parsed) { + $this->parsed = [ + 'code' => $this->source, + 'data' => [], + 'line' => 1, + ]; + + // Check for front matter data. + $len = strlen(static::FRONT_MATTER_SEPARATOR); + $matches = []; + if (substr($this->parsed['code'], 0, $len + 1) === static::FRONT_MATTER_SEPARATOR . "\n" || substr($this->parsed['code'], 0, $len + 2) === static::FRONT_MATTER_SEPARATOR . "\r\n") { + preg_match(static::FRONT_MATTER_REGEXP, $this->parsed['code'], $matches); + $matches = array_map('trim', $matches); + } + + // Immediately return if the code doesn't contain front matter data. + if (empty($matches)) { + return $this->parsed; + } + + // Set the extracted source code. + $this->parsed['code'] = $matches[2]; + + // Set the extracted front matter data. Do not catch any exceptions here + // as doing so would only obfuscate any errors found in the front matter + // data. Typecast to an array to ensure top level scalars are in an array. + if ($matches[1]) { + $this->parsed['data'] = (array) $this->serializer::decode($matches[1]); + } + + // Determine the real source line by counting newlines from the data and + // then adding 2 to account for the front matter separator (---) wrappers + // and then adding 1 more for the actual line number after the data. + $this->parsed['line'] = count(preg_split('/\r\n|\n/', $matches[1])) + 3; + } + return $this->parsed; + } + + /** + * Retrieves the extracted source code. + * + * @return string + * The extracted source code. + * + * @throws \Drupal\Component\Serialization\Exception\InvalidDataTypeException + * Exception thrown when the Front Matter cannot be parsed. + */ + public function getCode() { + return $this->parse()['code']; + } + + /** + * Retrieves the extracted front matter data. + * + * @return array + * The extracted front matter data. + * + * @throws \Drupal\Component\Serialization\Exception\InvalidDataTypeException + * Exception thrown when the Front Matter cannot be parsed. + */ + public function getData() { + return $this->parse()['data']; + } + + /** + * Retrieves the line where the source code starts, after any data. + * + * @return int + * The source code line. + * + * @throws \Drupal\Component\Serialization\Exception\InvalidDataTypeException + * Exception thrown when the Front Matter cannot be parsed. + */ + public function getLine() { + return $this->parse()['line']; + } + +} diff --git a/core/modules/help_topics/src/HelpTopicDiscovery.php b/core/modules/help_topics/src/HelpTopicDiscovery.php index 63564f3a8580dd7526e6dc6feb3fe2d1b23b9986..159cb00b91cfde7edee0d67d7ca45b30018b8ed9 100644 --- a/core/modules/help_topics/src/HelpTopicDiscovery.php +++ b/core/modules/help_topics/src/HelpTopicDiscovery.php @@ -7,6 +7,8 @@ use Drupal\Component\FileSystem\RegexDirectoryIterator; use Drupal\Component\Plugin\Discovery\DiscoveryInterface; use Drupal\Component\Plugin\Discovery\DiscoveryTrait; +use Drupal\Component\Serialization\Exception\InvalidDataTypeException; +use Drupal\Core\Serialization\Yaml; use Drupal\Core\StringTranslation\TranslatableMarkup; /** @@ -113,29 +115,37 @@ public function findAll() { static::FILE_KEY => $file, ]; - // Get the rest of the plugin definition from meta tags contained in the - // help topic Twig file. - foreach (get_meta_tags($file) as $key => $value) { - $key = substr($key, 11); + // Get the rest of the plugin definition from front matter contained in + // the help topic Twig file. + try { + $front_matter = FrontMatter::load(file_get_contents($file), Yaml::class)->getData(); + } + catch (InvalidDataTypeException $e) { + throw new DiscoveryException(sprintf('Malformed YAML in help topic "%s": %s.', $file, $e->getMessage())); + } + foreach ($front_matter as $key => $value) { switch ($key) { case 'related': - $data[$key] = array_map('trim', explode(',', $value)); + if (!is_array($value)) { + throw new DiscoveryException("$file contains invalid value for 'related' key, the value must be an array of strings"); + } + $data[$key] = $value; break; case 'top_level': - $data[$key] = TRUE; - if ($value !== '') { - throw new DiscoveryException("$file contains invalid meta tag with name='help_topic:top_level', the 'content' property should not exist"); + if (!is_bool($value)) { + throw new DiscoveryException("$file contains invalid value for 'top_level' key, the value must be a Boolean"); } + $data[$key] = $value; break; case 'label': $data[$key] = new TranslatableMarkup($value); break; default: - throw new DiscoveryException("$file contains invalid meta tag with name='$key'"); + throw new DiscoveryException("$file contains invalid key='$key'"); } } if (!isset($data['label'])) { - throw new DiscoveryException("$file does not contain the required meta tag with name='help_topic:label'"); + throw new DiscoveryException("$file does not contain the required key with name='label'"); } $all[$provider][$data['id']] = $data; diff --git a/core/modules/help_topics/src/HelpTopicPluginManager.php b/core/modules/help_topics/src/HelpTopicPluginManager.php index bfe63d798bd95c2bd49433f61185d851f31ae5fe..636bdda0a42895370792f47bb137b911d59472bc 100644 --- a/core/modules/help_topics/src/HelpTopicPluginManager.php +++ b/core/modules/help_topics/src/HelpTopicPluginManager.php @@ -17,18 +17,20 @@ * help_topics. The provider is validated to be the extension that provides the * help topic. * - * The Twig file must contain a meta tag named 'help_topic:label'. It can also - * contain meta tags named 'help_topic:top_level' and 'help_topic:related'. For - * example: + * The Twig file must contain YAML front matter with a key named 'label'. It can + * also contain keys named 'top_level' and 'related'. For example: * @code - * <!–– The label/title of the topic. --> - * <meta name="help_topic:label" content="Configuring error responses, including 403/404 pages"/> + * --- + * label: 'Configuring error responses, including 403/404 pages' * - * <!–– Related help topics in a comma separated help topic ID list. --> - * <meta name="help_topic:related" content="core.config_basic,core.maintenance"/> + * # Related help topics in an array. + * related: + * - core.config_basic + * - core.maintenance * - * <!–– If present then the help topic will appear on admin/help. --> - * <meta name="help_topic:top_level"/> + * # If the value is true then the help topic will appear on admin/help. + * top_level: true + * --- * @endcode * * In addition, modules wishing to add plugins can define them in a diff --git a/core/modules/help_topics/src/HelpTopicTwig.php b/core/modules/help_topics/src/HelpTopicTwig.php index 34f3804adb9e7de375ab0cb78154ab0ed0e0e01f..2d383ebee8667b16ba505c1fe92b1a48951aaf73 100644 --- a/core/modules/help_topics/src/HelpTopicTwig.php +++ b/core/modules/help_topics/src/HelpTopicTwig.php @@ -62,8 +62,6 @@ public static function create(ContainerInterface $container, array $configuratio */ public function getBody() { return [ - // Note that #markup elements are automatically XSS admin filtered which - // removes the meta tags from the rendered HTML. '#markup' => $this->twig->load('@help_topics/' . $this->getPluginId() . '.html.twig')->render(), ]; } diff --git a/core/modules/help_topics/src/HelpTopicTwigLoader.php b/core/modules/help_topics/src/HelpTopicTwigLoader.php index 1e8ebfd217a1a743ef07189635588002ad28e0f4..186572550bb988d526948483a5c9822c670157e1 100644 --- a/core/modules/help_topics/src/HelpTopicTwigLoader.php +++ b/core/modules/help_topics/src/HelpTopicTwigLoader.php @@ -2,8 +2,12 @@ namespace Drupal\help_topics; +use Drupal\Component\Serialization\Exception\InvalidDataTypeException; use Drupal\Core\Extension\ModuleHandlerInterface; use Drupal\Core\Extension\ThemeHandlerInterface; +use Drupal\Core\Serialization\Yaml; +use Twig\Error\LoaderError; +use Twig\Source; /** * Loads help topic Twig files from the filesystem. @@ -59,4 +63,33 @@ protected function addExtension($path) { } } + /** + * {@inheritdoc} + */ + public function getSourceContext($name) { + $path = $this->findTemplate($name); + + $contents = file_get_contents($path); + try { + // Note: always use \Drupal\Core\Serialization\Yaml here instead of the + // "serializer.yaml" service. This allows the core serializer to utilize + // core related functionality which isn't available as the standalone + // component based serializer. + $front_matter = FrontMatter::load($contents, Yaml::class); + + // Reconstruct the content if there is front matter data detected. Prepend + // the source with {% line \d+ %} to inform Twig that the source code + // actually starts on a different line past the front matter data. This is + // particularly useful when used in error reporting. + if ($front_matter->getData() && ($line = $front_matter->getLine())) { + $contents = "{% line $line %}" . $front_matter->getCode(); + } + } + catch (InvalidDataTypeException $e) { + throw new LoaderError(sprintf('Malformed YAML in help topic "%s": %s.', $path, $e->getMessage())); + } + + return new Source($contents, $name, $path); + } + } diff --git a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.additional.html.twig b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.additional.html.twig index 3696dad0ce8ba249ca9e56a3cda953d24929b6a8..cb70357edfcf00a78f11afe13db82ad6a531f71c 100644 --- a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.additional.html.twig +++ b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.additional.html.twig @@ -1,3 +1,6 @@ -<meta name="help_topic:label" content="Additional topic"/> -<meta name="help_topic:related" content="help_topics_test.test"/> +--- +label: 'Additional topic' +related: + - help_topics_test.test +--- <p>{% trans %}This topic should get listed automatically on the Help test topic.{% endtrans %}</p> diff --git a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.linked.html.twig b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.linked.html.twig index 9a1d37b219e3bd4790d80fe24df14c42f1efc7c8..a101a1c15c975374aeab4c0f7c213f481ae0aa9f 100644 --- a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.linked.html.twig +++ b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.linked.html.twig @@ -1,2 +1,4 @@ -<meta name="help_topic:label" content="Linked topic"/> +--- +label: 'Linked topic' +--- <p>{% trans %}This topic is not supposed to be top-level.{% endtrans %}</p> diff --git a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.test.html.twig b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.test.html.twig index eaf23b14d28bc8ba87753a61e50a8c1e41968601..0a1453602f035fbc552fa189d6282aee0ee63d8d 100644 --- a/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.test.html.twig +++ b/core/modules/help_topics/tests/modules/help_topics_test/help_topics/help_topics_test.test.html.twig @@ -1,6 +1,10 @@ -<meta name="help_topic:label" content="ABC Help Test module"/> -<meta name="help_topic:top_level"/> -<meta name="help_topic:related" content="help_topics_test.linked,does_not_exist.and_no_error"/> +--- +label: "ABC Help Test module" +top_level: true +related: + - help_topics_test.linked + - does_not_exist.and_no_error +--- {% set help_topic_url = render_var(url('help_topics.help_topic', {id: 'help_topics_test.additional'})) %} <p>{% trans %}This is a test. It should <a href="{{ help_topic_url }}">link to the additional topic</a>. Also there should be a related topic link below to the Help module topic page and the linked topic.{% endtrans %}</p> <p>{% trans %}Test translation.{% endtrans %}</p> diff --git a/core/modules/help_topics/tests/src/Functional/HelpTopicTest.php b/core/modules/help_topics/tests/src/Functional/HelpTopicTest.php index 769850a16ed9b845a3b7951784f970005318d7d7..b5877f7d48b3efb3b891275d6c04e8c6fa8ff8da 100644 --- a/core/modules/help_topics/tests/src/Functional/HelpTopicTest.php +++ b/core/modules/help_topics/tests/src/Functional/HelpTopicTest.php @@ -196,12 +196,7 @@ protected function verifyHelpLinks() { // Verify theme provided help topics work and can be related. $this->drupalGet('admin/help/topic/help_topics_test_theme.test'); $session->pageTextContains('This is a theme provided topic.'); - // Use the article element to provide a positive assertion to improve the - // assertion that the help html does not contain meta tags. $this->assertContains('This is a theme provided topic.', $session->elementExists('css', 'article')->getText()); - // Ensure that meta tags containing plugin information do not appear on - // topic pages - $session->elementNotExists('css', 'article meta'); $this->clickLink('Additional topic'); $session->linkExists('XYZ Help Test theme'); diff --git a/core/modules/help_topics/tests/src/Unit/HelpTopicDiscoveryTest.php b/core/modules/help_topics/tests/src/Unit/HelpTopicDiscoveryTest.php index bad221673d744ade0bb6cdb0591aa51a5215e41b..99cfb13834dc4f97da74112e6df3da2013efff1f 100644 --- a/core/modules/help_topics/tests/src/Unit/HelpTopicDiscoveryTest.php +++ b/core/modules/help_topics/tests/src/Unit/HelpTopicDiscoveryTest.php @@ -41,7 +41,7 @@ public function testDiscoveryExceptionProviderMismatch() { /** * @covers ::findAll */ - public function testDiscoveryExceptionMissingLabelMetaTag() { + public function testDiscoveryExceptionMissingLabel() { vfsStream::setup('root'); vfsStream::create([ @@ -57,21 +57,20 @@ public function testDiscoveryExceptionMissingLabelMetaTag() { $discovery = new HelpTopicDiscovery(['test' => vfsStream::url('root/modules/test/help_topics')]); $this->expectException(DiscoveryException::class); - $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig does not contain the required meta tag with name='help_topic:label'"); + $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig does not contain the required key with name='label'"); $discovery->getDefinitions(); } /** * @covers ::findAll */ - public function testDiscoveryExceptionInvalidMetaTag() { + public function testDiscoveryExceptionInvalidYamlKey() { vfsStream::setup('root'); - // Note a blank line is required after the last meta tag otherwise the last - // meta tag is not parsed. $topic_content = <<<EOF -<meta name="help_topic:label" content="A label"/> -<meta name="help_topic:foo" content="bar"/> - +--- +label: 'A label' +foo: bar +--- EOF; vfsStream::create([ @@ -86,7 +85,7 @@ public function testDiscoveryExceptionInvalidMetaTag() { $discovery = new HelpTopicDiscovery(['test' => vfsStream::url('root/modules/test/help_topics')]); $this->expectException(DiscoveryException::class); - $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig contains invalid meta tag with name='foo'"); + $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig contains invalid key='foo'"); $discovery->getDefinitions(); } @@ -95,12 +94,39 @@ public function testDiscoveryExceptionInvalidMetaTag() { */ public function testDiscoveryExceptionInvalidTopLevel() { vfsStream::setup('root'); - // Note a blank line is required after the last meta tag otherwise the last - // meta tag is not parsed. $topic_content = <<<EOF -<meta name="help_topic:label" content="A label"/> -<meta name="help_topic:top_level" content="bar"/> +--- +label: 'A label' +top_level: bar +--- +EOF; + + vfsStream::create([ + 'modules' => [ + 'test' => [ + 'help_topics' => [ + 'test.topic.html.twig' => $topic_content, + ], + ], + ], + ]); + $discovery = new HelpTopicDiscovery(['test' => vfsStream::url('root/modules/test/help_topics')]); + + $this->expectException(DiscoveryException::class); + $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig contains invalid value for 'top_level' key, the value must be a Boolean"); + $discovery->getDefinitions(); + } + /** + * @covers ::findAll + */ + public function testDiscoveryExceptionInvalidRelated() { + vfsStream::setup('root'); + $topic_content = <<<EOF +--- +label: 'A label' +related: "one, two" +--- EOF; vfsStream::create([ @@ -115,7 +141,7 @@ public function testDiscoveryExceptionInvalidTopLevel() { $discovery = new HelpTopicDiscovery(['test' => vfsStream::url('root/modules/test/help_topics')]); $this->expectException(DiscoveryException::class); - $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig contains invalid meta tag with name='help_topic:top_level', the 'content' property should not exist"); + $this->expectExceptionMessage("vfs://root/modules/test/help_topics/test.topic.html.twig contains invalid value for 'related' key, the value must be an array of strings"); $discovery->getDefinitions(); } @@ -125,7 +151,9 @@ public function testDiscoveryExceptionInvalidTopLevel() { public function testHelpTopicsExtensionProviderSpecialCase() { vfsStream::setup('root'); $topic_content = <<<EOF -<meta name="help_topic:label" content="Test"/> +--- +label: Test +--- <h2>Test</h2> EOF; @@ -142,6 +170,33 @@ public function testHelpTopicsExtensionProviderSpecialCase() { $this->assertArrayHasKey('core.topic', $discovery->getDefinitions()); } + /** + * @covers ::findAll + */ + public function testHelpTopicsBrokenYaml() { + vfsStream::setup('root'); + $topic_content = <<<EOF +--- +foo : [bar} +--- +<h2>Test</h2> +EOF; + + vfsStream::create([ + 'modules' => [ + 'help_topics' => [ + 'help_topics' => [ + 'core.topic.html.twig' => $topic_content, + ], + ], + ], + ]); + $discovery = new HelpTopicDiscovery(['help_topics' => vfsStream::url('root/modules/help_topics/help_topics')]); + $this->expectException(DiscoveryException::class); + $this->expectExceptionMessage("Malformed YAML in help topic \"vfs://root/modules/help_topics/help_topics/core.topic.html.twig\":"); + $discovery->getDefinitions(); + } + /** * @covers ::findAll */ @@ -152,9 +207,14 @@ public function testHelpTopicsDefinition() { vfsStream::setup('root'); $topic_content = <<<EOF -<meta name="help_topic:label" content="Test"/> -<meta name="help_topic:top_level"/> -<meta name="help_topic:related" content="one, two ,three"/> +--- +label: 'Test' +top_level: true +related: + - one + - two + - three +--- <h2>Test</h2> EOF; diff --git a/core/modules/help_topics/tests/src/Unit/HelpTopicTwigLoaderTest.php b/core/modules/help_topics/tests/src/Unit/HelpTopicTwigLoaderTest.php index cf4b8c61c8ae198be4f5f2c66ad11b0b6c6700d6..2f5258d054b620fc5084adc48e1b7d46e1036671 100644 --- a/core/modules/help_topics/tests/src/Unit/HelpTopicTwigLoaderTest.php +++ b/core/modules/help_topics/tests/src/Unit/HelpTopicTwigLoaderTest.php @@ -5,6 +5,7 @@ use Drupal\help_topics\HelpTopicTwigLoader; use Drupal\Tests\UnitTestCase; use org\bovigo\vfs\vfsStream; +use Twig\Error\LoaderError; /** * Unit test for the HelpTopicTwigLoader class. @@ -51,6 +52,24 @@ public function testConstructor() { $this->assertTrue(in_array($this->directories['theme']['test'] . '/help_topics', $paths)); } + /** + * @covers ::getSourceContext + */ + public function testGetSourceContext() { + $source = $this->helpLoader->getSourceContext('@' . HelpTopicTwigLoader::MAIN_NAMESPACE . '/test.topic.html.twig'); + $this->assertEquals('{% line 4 %}<h2>Test</h2>', $source->getCode()); + } + + /** + * @covers ::getSourceContext + */ + public function testGetSourceContextException() { + $this->expectException(LoaderError::class); + $this->expectExceptionMessage("Malformed YAML in help topic \"vfs://root/modules/test/help_topics/test.invalid_yaml.html.twig\":"); + + $source = $this->helpLoader->getSourceContext('@' . HelpTopicTwigLoader::MAIN_NAMESPACE . '/test.invalid_yaml.html.twig'); + } + /** * Creates a mock module or theme handler class for the test. * @@ -86,9 +105,22 @@ protected function getHandlerMock($type) { * Sets up the virtual file system. */ protected function setUpVfs() { + $content = <<<EOF +--- +label: Test +--- +<h2>Test</h2> +EOF; + $invalid_content = <<<EOF +--- +foo : [bar} +--- +<h2>Test</h2> +EOF; $help_topics_dir = [ 'help_topics' => [ - 'test.topic.html.twig' => '', + 'test.topic.html.twig' => $content, + 'test.invalid_yaml.html.twig' => $invalid_content, ], ]; diff --git a/core/modules/help_topics/tests/src/Unit/HelpTopicTwigTest.php b/core/modules/help_topics/tests/src/Unit/HelpTopicTwigTest.php index 70e085236fe68b5406135fe957cdbd3b240d8e92..201b73b70fd35e76c87f6787088df8ea15ebc31d 100644 --- a/core/modules/help_topics/tests/src/Unit/HelpTopicTwigTest.php +++ b/core/modules/help_topics/tests/src/Unit/HelpTopicTwigTest.php @@ -65,7 +65,7 @@ public function testText() { * @covers ::isTopLevel * @covers ::getRelated */ - public function testMeta() { + public function testDefinition() { $this->assertEquals($this->helpTopic->getProvider(), self::PLUGIN_INFORMATION['provider']); $this->assertEquals($this->helpTopic->isTopLevel(), diff --git a/core/modules/help_topics/tests/themes/help_topics_test_theme/help_topics/help_topics_test_theme.test.html.twig b/core/modules/help_topics/tests/themes/help_topics_test_theme/help_topics/help_topics_test_theme.test.html.twig index c2893d74bf340cbeaf542129a8368603c1d66218..79c85eed4d3e7dd91b89b65e94369cad4a58358c 100644 --- a/core/modules/help_topics/tests/themes/help_topics_test_theme/help_topics/help_topics_test_theme.test.html.twig +++ b/core/modules/help_topics/tests/themes/help_topics_test_theme/help_topics/help_topics_test_theme.test.html.twig @@ -1,4 +1,7 @@ -<meta name="help_topic:label" content="XYZ Help Test theme"/> -<meta name="help_topic:top_level"/> -<meta name="help_topic:related" content="help_topics_test.additional"/> +--- +label: 'XYZ Help Test theme' +top_level: true +related: + - help_topics_test.additional +--- <p>{% trans %}This is a theme provided topic.{% endtrans %}</p>