diff --git a/esmvalcore/experimental/recipe_output.py b/esmvalcore/experimental/recipe_output.py index 69f63765a2..9765d15c77 100644 --- a/esmvalcore/experimental/recipe_output.py +++ b/esmvalcore/experimental/recipe_output.py @@ -2,7 +2,7 @@ import base64 import logging import os.path -from collections.abc import Mapping +from collections.abc import Mapping, Sequence from pathlib import Path from typing import Optional, Tuple, Type @@ -123,6 +123,13 @@ class RecipeOutput(Mapping): The session used to run the recipe. """ + FILTER_ATTRS: list = [ + "realms", + "plot_type", # Used by several diagnostics + "plot_types", + "long_names", + ] + def __init__(self, task_output: dict, session=None, info=None): self._raw_task_output = task_output self._task_output = {} @@ -141,6 +148,7 @@ def __init__(self, task_output: dict, session=None, info=None): diagnostics[name].append(task) # Create diagnostic output + filters: dict = {} for name, tasks in diagnostics.items(): diagnostic_info = info.data['diagnostics'][name] self.diagnostics[name] = DiagnosticOutput( @@ -150,6 +158,36 @@ def __init__(self, task_output: dict, session=None, info=None): description=diagnostic_info.get('description'), ) + # Add data to filters + for task in tasks: + for file in task.files: + RecipeOutput._add_to_filters(filters, file.attributes) + + # Sort at the end because sets are unordered + self.filters = RecipeOutput._sort_filters(filters) + + @classmethod + def _add_to_filters(cls, filters, attributes): + """Add valid values to the HTML output filters.""" + for attr in RecipeOutput.FILTER_ATTRS: + if attr not in attributes: + continue + values = attributes[attr] + # `set()` to avoid duplicates + attr_list = filters.get(attr, set()) + if (isinstance(values, str) or not isinstance(values, Sequence)): + attr_list.add(values) + else: + attr_list.update(values) + filters[attr] = attr_list + + @classmethod + def _sort_filters(cls, filters): + """Sort the HTML output filters.""" + for _filter, _attrs in filters.items(): + filters[_filter] = sorted(_attrs) + return filters + def __repr__(self): """Return canonical string representation.""" string = '\n'.join(repr(item) for item in self._task_output.values()) @@ -218,6 +256,7 @@ def render(self, template=None): diagnostics=self.diagnostics.values(), session=self.session, info=self.info, + filters=self.filters, relpath=os.path.relpath, ) diff --git a/esmvalcore/experimental/templates/RecipeOutput.j2 b/esmvalcore/experimental/templates/RecipeOutput.j2 index c502c39a33..cde84362f9 100644 --- a/esmvalcore/experimental/templates/RecipeOutput.j2 +++ b/esmvalcore/experimental/templates/RecipeOutput.j2 @@ -1,12 +1,72 @@ + + +
+ +{{ diagnostic.description }}
+{{ diagnostic.description }}
- {% for task in diagnostic.task_output %} + {% set diagnostic_loop = loop %} + {% for task in diagnostic.task_output %} - {% include 'TaskOutput.j2' %} + {% include 'TaskOutput.j2' %} - {% endfor %} + {% endfor %} +