From 48ec10dfdc96f49879bc5784967d3476eeafac00 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andreas=20Kr=C3=BCger?= Date: Thu, 5 Dec 2024 11:36:17 +0100 Subject: [PATCH] Allow user-defined template engine configuration via conf.py. Also some improvements to make the template handling code better and more readable; mostly type hints. --- CHANGES.txt | 3 ++ CONTRIBUTING.rst | 2 +- docs/theming.rst | 36 ++++++++++++-- nikola/__init__.py | 3 +- nikola/conf.py.in | 14 ++++++ nikola/log.py | 9 +--- nikola/nikola.py | 28 ++++++----- nikola/plugin_categories.py | 83 +++++++++++++++++--------------- nikola/plugin_manager.py | 7 ++- nikola/plugins/task/archive.py | 4 +- nikola/plugins/template/jinja.py | 20 +++++++- nikola/plugins/template/mako.py | 15 +++++- nikola/utils.py | 34 ++++++------- 13 files changed, 168 insertions(+), 90 deletions(-) diff --git a/CHANGES.txt b/CHANGES.txt index af4666f562..5615f7e1f3 100644 --- a/CHANGES.txt +++ b/CHANGES.txt @@ -6,6 +6,9 @@ Features * Trace template usage when an environment variable ``NIKOLA_TEMPLATES_TRACE`` is set to any non-empty value. +* Give user control over the raw underlying template engine + (either `mako.lookup.TemplateLookup` or `jinja2.Environment`) + via an optional `conf.py` method `TEMPLATE_ENGINE_FACTORY`. Bugfixes -------- diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index cc4266cbb0..9d01ba53ad 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -86,4 +86,4 @@ Here are some guidelines about how you can contribute to Nikola: .. [1] Very inspired by `fabric’s `_ — thanks! -.. [2] For example, logging or always making sure directories are created using ``utils.makedirs()``. +.. [2] For example, logging, or always making sure directories are created using ``utils.makedirs()``. diff --git a/docs/theming.rst b/docs/theming.rst index 59007e5588..742822b72e 100644 --- a/docs/theming.rst +++ b/docs/theming.rst @@ -130,9 +130,17 @@ The following keys are currently supported: The parent is so you don’t have to create a full theme each time: just create an empty theme, set the parent, and add the bits you want modified. - It is strongly recommended you define a parent. If you don't, many features - won’t work due to missing templates, messages, and assets until your home-grown - template is complete. + + While it is possible to create a theme without a parent, it is + **strongly discouraged** and not officially supported, in the sense: + We won't help with issues that are caused by a theme being parentless, + and we won't guarantee that it will always work with new Nikola versions. + The `base` and `base-jinja` themes provide assets, messages, and generic templates + that Nikola expects to be able to use in all sites. That said, if you are making + something very custom, Nikola will not prevent the creation of a theme + without `base`, but you will need to manually determine which templates and + messages are required in your theme. (Initially setting the ``NIKOLA_TEMPLATES_TRACE`` + environment variable might be of some help, see below.) The following settings are recommended: @@ -484,6 +492,28 @@ at https://www.transifex.com/projects/p/nikola/ If you want to create a theme that has new strings, and you want those strings to be translatable, then your theme will need a custom ``messages`` folder. +Configuration of the raw template engine +---------------------------------------- + +For usage not covered by the above, you can define a method +`TEMPLATE_ENGINE_FACTORY` in `conf.py` that constructs the raw +underlying templating engine. That `raw_engine` that your method +needs to return is either a `jinja2.Environment` or a +`mako.loopkup.TemplateLookup` object. Your factory method is +called with the same arguments as is the pertinent `__init__`. + +E.g., to configure `jinja2` to bark and error out on missing values, +instead of silently continuing with empty content, you might do this: + +.. code:: python + + # Somewhere in conf.py: + def TEMPLATE_ENGINE_FACTORY(**args) -> jinja2.Environment: + augmented_args = dict(args) + augmented_args['undefined'] = jinja2.StrictUndefined + return jinja2.Environment(**augmented_args) + + `LESS `__ and `Sass `__ -------------------------------------------------------------------- diff --git a/nikola/__init__.py b/nikola/__init__.py index 4675518d04..cb33dc26be 100644 --- a/nikola/__init__.py +++ b/nikola/__init__.py @@ -31,13 +31,14 @@ # The current Nikola version: __version__ = '8.3.1' -# A flag whether logging should emmit debug information: +# A flag whether logging should emit debug information: DEBUG = bool(os.getenv('NIKOLA_DEBUG')) # A flag whether special templates trace logging should be generated: TEMPLATES_TRACE = bool(os.getenv('NIKOLA_TEMPLATES_TRACE')) # When this flag is set, fewer exceptions are handled internally; # instead they are left unhandled for the run time system to deal with them, # which typically leads to the stack traces being exposed. +# This is a less noisy alternative to the NIKOLA_DEBUG flag. SHOW_TRACEBACKS = bool(os.getenv('NIKOLA_SHOW_TRACEBACKS')) if sys.version_info[0] == 2: diff --git a/nikola/conf.py.in b/nikola/conf.py.in index d0d5e79a4d..b7f64c8d3a 100644 --- a/nikola/conf.py.in +++ b/nikola/conf.py.in @@ -1319,6 +1319,20 @@ WARN_ABOUT_TAG_METADATA = False # those. # TEMPLATE_FILTERS = {} +# If you want to, you can augment or change Nikola's configuration +# of the underlying template engine used +# in any way you please, by defining this function: +# def TEMPLATE_ENGINE_FACTORY(**args): +# pass +# This should return either a jinja2.Environment or a mako.lookup.TemplateLookup +# object that have been configured with the args received plus any additional configuration wanted. +# +# E.g., to configure Jinja2 to bark on non-existing values instead of silently omitting: +# def TEMPLATE_ENGINE_FACTORY(**args) -> jinja2.Environment: +# augmented_args = dict(args) +# augmented_args['undefined'] = jinja2.StrictUndefined +# return jinja2.Environment(**augmented_args) + # Put in global_context things you want available on all your templates. # It can be anything, data, functions, modules, etc. GLOBAL_CONTEXT = {} diff --git a/nikola/log.py b/nikola/log.py index 91db4ae554..9848477b53 100644 --- a/nikola/log.py +++ b/nikola/log.py @@ -105,12 +105,7 @@ def configure_logging(logging_mode: LoggingMode = LoggingMode.NORMAL) -> None: return handler = logging.StreamHandler() - handler.setFormatter( - ColorfulFormatter( - fmt=_LOGGING_FMT, - datefmt=_LOGGING_DATEFMT, - ) - ) + handler.setFormatter(ColorfulFormatter(fmt=_LOGGING_FMT, datefmt=_LOGGING_DATEFMT)) handlers = [handler] if logging_mode == LoggingMode.STRICT: @@ -152,7 +147,7 @@ def init_template_trace_logging(filename: str) -> None: As there is lots of other stuff happening on the normal output stream, this info is also written to a log file. - """ + """ TEMPLATES_LOGGER.level = logging.DEBUG formatter = logging.Formatter( fmt=_LOGGING_FMT, diff --git a/nikola/nikola.py b/nikola/nikola.py index 5609b408df..9684470d3a 100644 --- a/nikola/nikola.py +++ b/nikola/nikola.py @@ -36,6 +36,7 @@ import pathlib import sys import typing +from typing import Any, Callable, Dict, Iterable, List, Optional, Set import mimetypes from collections import defaultdict from copy import copy @@ -373,7 +374,7 @@ class Nikola(object): plugin_manager: PluginManager _template_system: TemplateSystem - def __init__(self, **config): + def __init__(self, **config) -> None: """Initialize proper environment for running tasks.""" # Register our own path handlers self.path_handlers = { @@ -395,7 +396,7 @@ def __init__(self, **config): self.timeline = [] self.pages = [] self._scanned = False - self._template_system: typing.Optional[TemplateSystem] = None + self._template_system: Optional[TemplateSystem] = None self._THEMES = None self._MESSAGES = None self.filters = {} @@ -996,13 +997,13 @@ def __init__(self, **config): # WebP files have no official MIME type yet, but we need to recognize them (Issue #3671) mimetypes.add_type('image/webp', '.webp') - def _filter_duplicate_plugins(self, plugin_list: typing.Iterable[PluginCandidate]): + def _filter_duplicate_plugins(self, plugin_list: Iterable[PluginCandidate]): """Find repeated plugins and discard the less local copy.""" def plugin_position_in_places(plugin: PluginInfo): # plugin here is a tuple: # (path to the .plugin file, path to plugin module w/o .py, plugin metadata) + place: pathlib.Path for i, place in enumerate(self._plugin_places): - place: pathlib.Path try: # Path.is_relative_to backport plugin.source_dir.relative_to(place) @@ -1025,7 +1026,7 @@ def plugin_position_in_places(plugin: PluginInfo): result.append(plugins[-1]) return result - def init_plugins(self, commands_only=False, load_all=False): + def init_plugins(self, commands_only=False, load_all=False) -> None: """Load plugins as needed.""" extra_plugins_dirs = self.config['EXTRA_PLUGINS_DIRS'] self._loading_commands_only = commands_only @@ -1086,9 +1087,9 @@ def init_plugins(self, commands_only=False, load_all=False): # Search for compiler plugins which we disabled but shouldn't have self._activate_plugins_of_category("PostScanner") if not load_all: - file_extensions = set() + file_extensions: Set[str] = set() + post_scanner: PostScanner for post_scanner in [p.plugin_object for p in self.plugin_manager.get_plugins_of_category('PostScanner')]: - post_scanner: PostScanner exts = post_scanner.supported_extensions() if exts is not None: file_extensions.update(exts) @@ -1126,8 +1127,8 @@ def init_plugins(self, commands_only=False, load_all=False): self._activate_plugins_of_category("Taxonomy") self.taxonomy_plugins = {} + taxonomy: Taxonomy for taxonomy in [p.plugin_object for p in self.plugin_manager.get_plugins_of_category('Taxonomy')]: - taxonomy: Taxonomy if not taxonomy.is_enabled(): continue if taxonomy.classification_name in self.taxonomy_plugins: @@ -1322,7 +1323,7 @@ def _activate_plugin(self, plugin_info: PluginInfo) -> None: if candidate.exists() and candidate.is_dir(): self.template_system.inject_directory(str(candidate)) - def _activate_plugins_of_category(self, category) -> typing.List[PluginInfo]: + def _activate_plugins_of_category(self, category) -> List[PluginInfo]: """Activate all the plugins of a given category and return them.""" # this code duplicated in tests/base.py plugins = [] @@ -1397,6 +1398,11 @@ def _get_template_system(self): "plugin\n".format(template_sys_name)) sys.exit(1) self._template_system = typing.cast(TemplateSystem, pi.plugin_object) + + engine_factory: Optional[Callable[..., Any]] = self.config.get("TEMPLATE_ENGINE_FACTORY") + if engine_factory is not None: + self._template_system.user_engine_factory(engine_factory) + lookup_dirs = ['templates'] + [os.path.join(utils.get_theme_path(name), "templates") for name in self.THEMES] self._template_system.set_directories(lookup_dirs, @@ -1444,7 +1450,7 @@ def get_compiler(self, source_name): return compiler - def render_template(self, template_name, output_name, context, url_type=None, is_fragment=False): + def render_template(self, template_name: str, output_name: str, context, url_type=None, is_fragment=False): """Render a template with the global context. If ``output_name`` is None, will return a string and all URL @@ -1463,7 +1469,7 @@ def render_template(self, template_name, output_name, context, url_type=None, is utils.TEMPLATES_LOGGER.debug("For %s, template %s builds %s", context["post"].source_path, template_name, output_name) else: utils.TEMPLATES_LOGGER.debug("Template %s builds %s", template_name, output_name) - local_context: typing.Dict[str, typing.Any] = {} + local_context: Dict[str, Any] = {} local_context["template_name"] = template_name local_context.update(self.GLOBAL_CONTEXT) local_context.update(context) diff --git a/nikola/plugin_categories.py b/nikola/plugin_categories.py index 7b650a6c7d..0b4265a114 100644 --- a/nikola/plugin_categories.py +++ b/nikola/plugin_categories.py @@ -29,16 +29,15 @@ import io import logging import os -import typing + +from typing import Any, Callable, Dict, Iterable, List, Optional, Tuple import doit from doit.cmd_base import Command as DoitCommand from .utils import LOGGER, first_line, get_logger, req_missing -if typing.TYPE_CHECKING: - import nikola - import nikola.post +from nikola.post import Post __all__ = ( 'Command', @@ -61,7 +60,7 @@ class BasePlugin: """Base plugin class.""" logger = None - site: 'nikola.nikola.Nikola' + site: Optional['nikola.nikola.Nikola'] = None # NOQA # Fixme: Circular import def set_site(self, site): """Set site, which is a Nikola instance.""" @@ -89,11 +88,11 @@ def register_auto_watched_folder(self, folder: str) -> None: class PostScanner(BasePlugin): """The scan method of these plugins is called by Nikola.scan_posts.""" - def scan(self) -> 'typing.List[nikola.post.Post]': + def scan(self) -> List[Post]: """Create a list of posts from some source. Returns a list of Post objects.""" raise NotImplementedError() - def supported_extensions(self) -> 'typing.Optional[typing.List]': + def supported_extensions(self) -> Optional[List[str]]: """Return a list of supported file extensions, or None if such a list isn't known beforehand.""" return None @@ -172,11 +171,11 @@ class BaseTask(BasePlugin): # the others have to be specifie in the command line. is_default = True - def gen_tasks(self) -> 'typing.List[dict]': + def gen_tasks(self) -> List[dict]: """Generate tasks.""" raise NotImplementedError() - def group_task(self) -> dict: + def group_task(self) -> Dict[str, Optional[str]]: """Return dict for group task.""" return { 'basename': self.name, @@ -202,10 +201,14 @@ class TemplateSystem(BasePlugin): name = "dummy_templates" - def set_directories(self, directories: 'typing.List[str]', cache_folder: str): + def set_directories(self, directories: List[str], cache_folder: str) -> None: """Set the list of folders where templates are located and cache.""" raise NotImplementedError() + def user_engine_factory(self, factory: Callable[..., Any]) -> None: + """Accept a function that will be used to produce the underlying raw template engine.""" + raise NotImplementedError() + def template_deps(self, template_name: str, context=None): """Return filenames which are dependencies for a template.""" raise NotImplementedError() @@ -218,7 +221,7 @@ def get_string_deps(self, text: str, context=None): """Find dependencies for a template string.""" raise NotImplementedError() - def render_template(self, template_name: str, output_name: str, context: 'typing.Dict[str, str]'): + def render_template(self, template_name: str, output_name: str, context: Dict[str, str]) -> str: """Render template to a file using context. This must save the data to output_name *and* return it @@ -226,7 +229,7 @@ def render_template(self, template_name: str, output_name: str, context: 'typing """ raise NotImplementedError() - def render_template_to_string(self, template: str, context: 'typing.Dict[str, str]') -> str: + def render_template_to_string(self, template: str, context: Dict[str, str]) -> str: """Render template to a string using context.""" raise NotImplementedError() @@ -271,11 +274,11 @@ class PageCompiler(BasePlugin): } config_dependencies = [] - def get_dep_filename(self, post: 'nikola.post.Post', lang: str) -> str: + def get_dep_filename(self, post: Post, lang: str) -> str: """Return the .dep file's name for the given post and language.""" return post.translated_base_path(lang) + '.dep' - def _read_extra_deps(self, post: 'nikola.post.Post', lang: str) -> 'typing.List[str]': + def _read_extra_deps(self, post: Post, lang: str) -> List[str]: """Read contents of .dep file and return them as a list.""" dep_path = self.get_dep_filename(post, lang) if os.path.isfile(dep_path): @@ -284,9 +287,9 @@ def _read_extra_deps(self, post: 'nikola.post.Post', lang: str) -> 'typing.List[ return deps return [] - def register_extra_dependencies(self, post: 'nikola.post.Post'): + def register_extra_dependencies(self, post: Post): """Add dependency to post object to check .dep file.""" - def create_lambda(lang: str) -> 'typing.Callable': + def create_lambda(lang: str) -> Callable: # We create a lambda like this so we can pass `lang` to it, because if we didn’t # add that function, `lang` would always be the last language in TRANSLATIONS. # (See https://docs.python-guide.org/writing/gotchas/#late-binding-closures) @@ -295,7 +298,7 @@ def create_lambda(lang: str) -> 'typing.Callable': for lang in self.site.config['TRANSLATIONS']: post.add_dependency(create_lambda(lang), 'fragment', lang=lang) - def get_extra_targets(self, post: 'nikola.post.Post', lang: str, dest: str) -> 'typing.List[str]': + def get_extra_targets(self, post: Post, lang: str, dest: str) -> List[str]: """Return a list of extra targets for the render_posts task when compiling the post for the specified language.""" if self.use_dep_file: return [self.get_dep_filename(post, lang)] @@ -322,11 +325,11 @@ def extension(self) -> str: """Return the preferred extension for the output of this compiler.""" return ".html" - def read_metadata(self, post: 'nikola.post.Post', lang=None) -> 'typing.Dict[str, str]': + def read_metadata(self, post: Post, lang=None) -> Dict[str, str]: """Read the metadata from a post, and return a metadata dict.""" return {} - def split_metadata(self, data: str, post=None, lang=None) -> (str, str): + def split_metadata(self, data: str, post=None, lang=None) -> Tuple[str, str]: """Split data from metadata in the raw post content.""" if lang and post: extractor = post.used_extractor[lang] @@ -397,14 +400,14 @@ class MetadataExtractor(BasePlugin): # Whether or not the extractor supports writing metadata. supports_write = False - def _extract_metadata_from_text(self, source_text: str) -> 'typing.Dict[str, str]': + def _extract_metadata_from_text(self, source_text: str) -> Dict[str, str]: """Extract metadata from text.""" raise NotImplementedError() - def split_metadata_from_text(self, source_text: str) -> (str, str): + def split_metadata_from_text(self, source_text: str) -> Tuple[str, str]: """Split text into metadata and content (both strings).""" if self.split_metadata_re is None: - return source_text + return "", source_text else: split_result = self.split_metadata_re.split(source_text.lstrip(), maxsplit=1) if len(split_result) == 1: @@ -413,19 +416,19 @@ def split_metadata_from_text(self, source_text: str) -> (str, str): # Necessary? return split_result[0], split_result[-1] - def extract_text(self, source_text: str) -> 'typing.Dict[str, str]': + def extract_text(self, source_text: str) -> Dict[str, str]: """Split file, return metadata and the content.""" + # TODO: The name and interface of this method is a mess and needs to be cleaned up. split = self.split_metadata_from_text(source_text) - if not split: + if len(split[0]) == 0: return {} - meta = self._extract_metadata_from_text(split[0]) - return meta + return self._extract_metadata_from_text(split[0]) - def extract_filename(self, filename: str, lang: str) -> 'typing.Dict[str, str]': + def extract_filename(self, filename: str, lang: str) -> Dict[str, str]: """Extract metadata from filename.""" return {} - def write_metadata(self, metadata: 'typing.Dict[str, str]', comment_wrap=False) -> str: + def write_metadata(self, metadata: Dict[str, str], comment_wrap=False) -> str: """Write metadata in this extractor’s format. ``comment_wrap`` is either True, False, or a 2-tuple of comments to use for wrapping, if necessary. @@ -716,18 +719,18 @@ def is_enabled(self, lang=None) -> bool: """ return True - def get_implicit_classifications(self, lang: str) -> 'typing.List[str]': + def get_implicit_classifications(self, lang: str) -> List[str]: """Return a list of classification strings which should always appear in posts_per_classification.""" return [] - def classify(self, post: 'nikola.post.Post', lang: str) -> 'typing.Iterable[str]': + def classify(self, post: Post, lang: str) -> Iterable[str]: """Classify the given post for the given language. Must return a list or tuple of strings. """ raise NotImplementedError() - def sort_posts(self, posts: 'typing.List[nikola.post.Post]', classification: str, lang: str): + def sort_posts(self, posts: List[Post], classification: str, lang: str): """Sort the given list of posts. Allows the plugin to order the posts per classification as it wants. @@ -736,7 +739,7 @@ def sort_posts(self, posts: 'typing.List[nikola.post.Post]', classification: str """ pass - def sort_classifications(self, classifications: 'typing.List[str]', lang: str, level=None): + def sort_classifications(self, classifications: List[str], lang: str, level=None): """Sort the given list of classification strings. Allows the plugin to order the classifications as it wants. The @@ -810,7 +813,7 @@ def get_path(self, classification: str, lang: str, dest_type='page') -> str: """ raise NotImplementedError() - def extract_hierarchy(self, classification: str) -> 'typing.List[str]': + def extract_hierarchy(self, classification: str) -> List[str]: """Given a classification, return a list of parts in the hierarchy. For non-hierarchical taxonomies, it usually suffices to return @@ -818,7 +821,7 @@ def extract_hierarchy(self, classification: str) -> 'typing.List[str]': """ return [classification] - def recombine_classification_from_hierarchy(self, hierarchy: 'typing.List[str]') -> str: + def recombine_classification_from_hierarchy(self, hierarchy: List[str]) -> str: """Given a list of parts in the hierarchy, return the classification string. For non-hierarchical taxonomies, it usually suffices to return hierarchy[0]. @@ -835,7 +838,7 @@ def provide_overview_context_and_uptodate(self, lang: str) -> str: """ raise NotImplementedError() - def provide_context_and_uptodate(self, classification: str, lang: str, node=None) -> 'typing.Tuple[typing.Dict]': + def provide_context_and_uptodate(self, classification: str, lang: str, node=None) -> Tuple[Dict, Dict]: """Provide data for the context and the uptodate list for the list of the given classification. Must return a tuple of two dicts. The first is merged into the page's context, @@ -848,19 +851,19 @@ def provide_context_and_uptodate(self, classification: str, lang: str, node=None """ raise NotImplementedError() - def should_generate_classification_page(self, classification: str, post_list: 'typing.List[nikola.post.Post]', lang: str) -> bool: + def should_generate_classification_page(self, classification: str, post_list: List[Post], lang: str) -> bool: """Only generates list of posts for classification if this function returns True.""" return True - def should_generate_atom_for_classification_page(self, classification: str, post_list: 'typing.List[nikola.post.Post]', lang: str) -> bool: + def should_generate_atom_for_classification_page(self, classification: str, post_list: List[Post], lang: str) -> bool: """Only generates Atom feed for list of posts for classification if this function returns True.""" return self.should_generate_classification_page(classification, post_list, lang) - def should_generate_rss_for_classification_page(self, classification: str, post_list: 'typing.List[nikola.post.Post]', lang: str) -> bool: + def should_generate_rss_for_classification_page(self, classification: str, post_list: List[Post], lang: str) -> bool: """Only generates RSS feed for list of posts for classification if this function returns True.""" return self.should_generate_classification_page(classification, post_list, lang) - def postprocess_posts_per_classification(self, posts_per_classification_per_language: 'typing.List[nikola.post.Post]', flat_hierarchy_per_lang=None, hierarchy_lookup_per_lang=None) -> 'typing.List[nikola.post.Post]': + def postprocess_posts_per_classification(self, posts_per_classification_per_language: List[Post], flat_hierarchy_per_lang=None, hierarchy_lookup_per_lang=None) -> None: """Rearrange, modify or otherwise use the list of posts per classification and per language. For compatibility reasons, the list could be stored somewhere else as well. @@ -872,7 +875,7 @@ def postprocess_posts_per_classification(self, posts_per_classification_per_lang """ pass - def get_other_language_variants(self, classification: str, lang: str, classifications_per_language: 'typing.List[str]') -> 'typing.List[str]': + def get_other_language_variants(self, classification: str, lang: str, classifications_per_language: List[str]) -> List[str]: """Return a list of variants of the same classification in other languages. Given a `classification` in a language `lang`, return a list of pairs diff --git a/nikola/plugin_manager.py b/nikola/plugin_manager.py index 60162a4a10..ab8e5e97e5 100644 --- a/nikola/plugin_manager.py +++ b/nikola/plugin_manager.py @@ -35,13 +35,12 @@ from collections import deque from dataclasses import dataclass from pathlib import Path -from typing import Dict, List, Optional, Type, TYPE_CHECKING, Set +from typing import Dict, Iterable, List, Optional, Type, Set from .plugin_categories import BasePlugin, CATEGORIES from .utils import get_logger -if TYPE_CHECKING: - import logging +import logging LEGACY_PLUGIN_NAMES: Dict[str, str] = { "Compiler": "PageCompiler", @@ -158,7 +157,7 @@ def locate_plugins(self) -> List[PluginCandidate]: ) return self.candidates - def load_plugins(self, candidates: List[PluginCandidate]) -> List[PluginInfo]: + def load_plugins(self, candidates: Iterable[PluginCandidate]) -> List[PluginInfo]: """Load selected candidate plugins.""" plugins_root = Path(__file__).parent.parent new_plugins = [] diff --git a/nikola/plugins/task/archive.py b/nikola/plugins/task/archive.py index 44849fa007..f4eaa5e2cb 100644 --- a/nikola/plugins/task/archive.py +++ b/nikola/plugins/task/archive.py @@ -209,7 +209,7 @@ def provide_context_and_uptodate(self, classification, lang, node=None): kw.update(context) return context, kw - def postprocess_posts_per_classification(self, posts_per_classification_per_language, flat_hierarchy_per_lang=None, hierarchy_lookup_per_lang=None): + def postprocess_posts_per_classification(self, posts_per_classification_per_language, flat_hierarchy_per_lang=None, hierarchy_lookup_per_lang=None) -> None: """Rearrange, modify or otherwise use the list of posts per classification and per language.""" # Build a lookup table for archive navigation, if we’ll need one. if self.site.config['CREATE_ARCHIVE_NAVIGATION']: @@ -228,7 +228,7 @@ def postprocess_posts_per_classification(self, posts_per_classification_per_lang for k, v in self.archive_navigation[lang].items(): self.archive_navigation[lang][k] = natsort.natsorted(v, alg=natsort.ns.F | natsort.ns.IC) - return super().postprocess_posts_per_classification(posts_per_classification_per_language, flat_hierarchy_per_lang, hierarchy_lookup_per_lang) + super().postprocess_posts_per_classification(posts_per_classification_per_language, flat_hierarchy_per_lang, hierarchy_lookup_per_lang) def should_generate_classification_page(self, classification, post_list, lang): """Only generates list of posts for classification if this function returns True.""" diff --git a/nikola/plugins/template/jinja.py b/nikola/plugins/template/jinja.py index e0ddf4aae7..3c331c0370 100644 --- a/nikola/plugins/template/jinja.py +++ b/nikola/plugins/template/jinja.py @@ -29,6 +29,7 @@ import io import json import os +from typing import Callable, Optional from nikola.plugin_categories import TemplateSystem from nikola.utils import makedirs, req_missing, slugify, sort_posts, _smartjoin_filter @@ -45,7 +46,15 @@ class JinjaTemplates(TemplateSystem): """Support for Jinja2 templates.""" name = "jinja" - lookup = None + if jinja2 is None: + lookup = None + else: + lookup: Optional[jinja2.Environment] = None + + def _basic_environment_factory(self, **args): + return jinja2.Environment(**args) + _environment_factory = _basic_environment_factory + dependency_cache = {} per_file_cache = {} @@ -54,6 +63,13 @@ def __init__(self): if jinja2 is None: return + def user_engine_factory(self, factory: Callable[..., jinja2.Environment]) -> None: + """Accept a factory that will be used to produce the underlying jinja2.Environment. + + Not normally needed, but it is there if you have special requirements. + """ + self._environment_factory = factory + def set_directories(self, directories, cache_folder): """Create a new template lookup with set directories.""" if jinja2 is None: @@ -61,7 +77,7 @@ def set_directories(self, directories, cache_folder): cache_folder = os.path.join(cache_folder, 'jinja') makedirs(cache_folder) cache = jinja2.FileSystemBytecodeCache(cache_folder) - self.lookup = jinja2.Environment(bytecode_cache=cache) + self.lookup = self._environment_factory(bytecode_cache=cache) self.lookup.trim_blocks = True self.lookup.lstrip_blocks = True self.lookup.filters['tojson'] = json.dumps diff --git a/nikola/plugins/template/mako.py b/nikola/plugins/template/mako.py index 9517b05f12..2e77af0d41 100644 --- a/nikola/plugins/template/mako.py +++ b/nikola/plugins/template/mako.py @@ -30,6 +30,7 @@ import os import re import shutil +from typing import Callable from mako import exceptions, util, lexer, parsetree from mako.lookup import TemplateLookup @@ -53,6 +54,11 @@ class MakoTemplates(TemplateSystem): directories = [] cache_dir = None + def _basic_environment_factory(self, **args) -> TemplateLookup: + return TemplateLookup(**args) + + _environment_factory = _basic_environment_factory + def get_string_deps(self, text, context=None, *, filename=None): """Find dependencies for a template string.""" lex = lexer.Lexer(text=text, filename=filename, input_encoding='utf-8') @@ -100,7 +106,7 @@ def inject_directory(self, directory): def create_lookup(self): """Create a template lookup.""" - self.lookup = TemplateLookup( + self.lookup = self._basic_environment_factory( directories=self.directories, module_directory=self.cache_dir, input_encoding='utf-8', @@ -111,6 +117,13 @@ def set_site(self, site): self.site = site self.filters.update(self.site.config['TEMPLATE_FILTERS']) + def user_engine_factory(self, factory: Callable[..., TemplateLookup]) -> None: + """Accept a factory that will be used to produce the underlying TemplateLookup. + + Not normally needed, but it is there if you have special requirements. + """ + self._environment_factory = factory + def render_template(self, template_name, output_name, context): """Render the template into output_name using context.""" context['striphtml'] = striphtml diff --git a/nikola/utils.py b/nikola/utils.py index 18489fe109..cc1405330b 100644 --- a/nikola/utils.py +++ b/nikola/utils.py @@ -40,9 +40,7 @@ import subprocess import sys import threading -import typing from collections import defaultdict, OrderedDict -from collections.abc import Callable, Iterable from html import unescape as html_unescape from importlib import reload as _reload from unicodedata import normalize as unicodenormalize @@ -61,7 +59,7 @@ from doit import tools from doit.cmdparse import CmdParse from nikola.packages.pygments_better_html import BetterHtmlFormatter -from typing import List +from typing import Any, Callable, Dict, Iterable, List, Match, Optional, Union from unidecode import unidecode # Renames @@ -589,7 +587,7 @@ def pkg_resources_path(package, resource): return str(resources.files(package).joinpath(resource)) -def get_theme_path_real(theme, themes_dirs): +def get_theme_path_real(theme, themes_dirs) -> str: """Return the path where the given theme's files are located. Looks in ./themes and in the place where themes go when installed. @@ -609,7 +607,7 @@ def get_theme_path(theme): return theme -def parse_theme_meta(theme_dir): +def parse_theme_meta(theme_dir) -> Optional[configparser.ConfigParser]: """Parse a .theme meta file.""" cp = configparser.ConfigParser() # The `or` case is in case theme_dir ends with a trailing slash @@ -619,7 +617,7 @@ def parse_theme_meta(theme_dir): return cp if cp.has_section('Theme') else None -def get_template_engine(themes): +def get_template_engine(themes) -> str: """Get template engine used by a given theme.""" for theme_name in themes: meta = parse_theme_meta(theme_name) @@ -637,7 +635,7 @@ def get_template_engine(themes): return 'mako' -def get_parent_theme_name(theme_name, themes_dirs=None): +def get_parent_theme_name(theme_name, themes_dirs=None) -> Optional[str]: """Get name of parent theme.""" meta = parse_theme_meta(theme_name) if meta: @@ -657,7 +655,7 @@ def get_parent_theme_name(theme_name, themes_dirs=None): return None -def get_theme_chain(theme, themes_dirs): +def get_theme_chain(theme, themes_dirs) -> List[str]: """Create the full theme inheritance chain including paths.""" themes = [get_theme_path_real(theme, themes_dirs)] @@ -1197,7 +1195,7 @@ class LocaleBorg(object): in_string_formatter = None @classmethod - def initialize(cls, locales: 'typing.Dict[str, str]', initial_lang: str): + def initialize(cls, locales: Dict[str, str], initial_lang: str): """Initialize LocaleBorg. locales: dict with custom locale name overrides. @@ -1250,8 +1248,8 @@ def set_locale(self, lang: str) -> str: return '' def formatted_date(self, date_format: 'str', - date: 'typing.Union[datetime.date, datetime.datetime]', - lang: 'typing.Optional[str]' = None) -> str: + date: Union[datetime.date, datetime.datetime], + lang: Optional[str] = None) -> str: """Return the formatted date/datetime as a string.""" if lang is None: lang = self.current_lang @@ -1270,7 +1268,7 @@ def formatted_date(self, date_format: 'str', else: return format_datetime(date, date_format, locale=locale) - def format_date_in_string(self, message: str, date: datetime.date, lang: 'typing.Optional[str]' = None) -> str: + def format_date_in_string(self, message: str, date: datetime.date, lang: Optional[str] = None) -> str: """Format date inside a string (message). Accepted modes: month, month_year, month_day_year. @@ -1286,7 +1284,7 @@ def format_date_in_string(self, message: str, date: datetime.date, lang: 'typing lang = self.current_lang locale = self.locales.get(lang, lang) - def date_formatter(match: typing.Match) -> str: + def date_formatter(match: Match) -> str: """Format a date as requested.""" mode, custom_format = match.groups() if LocaleBorg.in_string_formatter is not None: @@ -1940,8 +1938,8 @@ def sort_posts(posts, *keys): return posts -def smartjoin(join_char: str, string_or_iterable) -> str: - """Join string_or_iterable with join_char if it is iterable; otherwise converts it to string. +def smartjoin(join_char: str, string_or_iterable: Union[None, str, bytes, Iterable[Any]]) -> str: + """Join string_or_iterable with join_char if it is iterable; otherwise convert it to string. >>> smartjoin('; ', 'foo, bar') 'foo, bar' @@ -1949,10 +1947,10 @@ def smartjoin(join_char: str, string_or_iterable) -> str: 'foo; bar' >>> smartjoin(' to ', ['count', 42]) 'count to 42' + + The treatment of bytes (calling str(string_or_iterable)) is somewhat dubious. Is this needed? """ - if isinstance(string_or_iterable, (str, bytes)): - return string_or_iterable - elif isinstance(string_or_iterable, Iterable): + if isinstance(string_or_iterable, Iterable): return join_char.join([str(e) for e in string_or_iterable]) else: return str(string_or_iterable)