From 3b1d39b03089e228b6ea5ce17e59cdc7ea063387 Mon Sep 17 00:00:00 2001 From: Juan-Pablo Scaletti Date: Wed, 28 Aug 2024 23:14:37 -0500 Subject: [PATCH] Make compiled_css and compiled_js context vars (#90) * Make compiled_css and compiled_js context vars --- benchmark/benchmark.py | 3 +- poetry.lock | 8 +- src/jinjax/catalog.py | 224 +++++++++++++++++++++++++++++------------ tests/test_render.py | 134 +++++++++++++++++++++++- 4 files changed, 299 insertions(+), 70 deletions(-) diff --git a/benchmark/benchmark.py b/benchmark/benchmark.py index 09a7447..b939952 100644 --- a/benchmark/benchmark.py +++ b/benchmark/benchmark.py @@ -3,6 +3,7 @@ from fastapi.templating import Jinja2Templates from jinja2 import Environment, FileSystemLoader + from jinjax import Catalog @@ -72,7 +73,7 @@ def print_separator(): if __name__ == "__main__": - print(f"Benchmarking...\n") + print("Benchmarking...\n") time_jinja = timeit.timeit(render_jinja, number=number) time_fastapi = timeit.timeit(render_fastapi, number=number) diff --git a/poetry.lock b/poetry.lock index 6bd91d8..5bcb8e2 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.3 and should not be changed by hand. [[package]] name = "cachetools" @@ -355,13 +355,13 @@ testing = ["covdefaults (>=2.3)", "pytest (>=8.2.2)", "pytest-cov (>=5)", "pytes [[package]] name = "pyright" -version = "1.1.369" +version = "1.1.377" description = "Command line wrapper for pyright" optional = false python-versions = ">=3.7" files = [ - {file = "pyright-1.1.369-py3-none-any.whl", hash = "sha256:06d5167a8d7be62523ced0265c5d2f1e022e110caf57a25d92f50fb2d07bcda0"}, - {file = "pyright-1.1.369.tar.gz", hash = "sha256:ad290710072d021e213b98cc7a2f90ae3a48609ef5b978f749346d1a47eb9af8"}, + {file = "pyright-1.1.377-py3-none-any.whl", hash = "sha256:af0dd2b6b636c383a6569a083f8c5a8748ae4dcde5df7914b3f3f267e14dd162"}, + {file = "pyright-1.1.377.tar.gz", hash = "sha256:aabc30fedce0ded34baa0c49b24f10e68f4bfc8f68ae7f3d175c4b0f256b4fcf"}, ] [package.dependencies] diff --git a/src/jinjax/catalog.py b/src/jinjax/catalog.py index c70f81d..d0d0635 100644 --- a/src/jinjax/catalog.py +++ b/src/jinjax/catalog.py @@ -1,6 +1,7 @@ import os import typing as t from collections import UserString +from contextvars import ContextVar from hashlib import sha256 from pathlib import Path @@ -22,6 +23,10 @@ ARGS_ATTRS = "attrs" ARGS_CONTENT = "content" +# Create ContextVars containers at module level +collected_css: dict[int, ContextVar[list[str]]] = {} +collected_js: dict[int, ContextVar[list[str]]] = {} + class CallerWrapper(UserString): def __init__(self, caller: t.Callable | None, content: str = "") -> None: @@ -42,7 +47,6 @@ def data(self) -> str: # type: ignore return self() - class Catalog: """ The object that manages the components and their global settings. @@ -50,12 +54,12 @@ class Catalog: Arguments: globals: - Dictionary of Jinja globals to add to the Catalog's Jinja environment - (or the one passed in `jinja_env`). + Dictionary of Jinja globals to add to the Catalog's Jinja + environment (or the one passed in `jinja_env`). filters: - Dictionary of Jinja filters to add to the Catalog's Jinja environment - (or the one passed in `jinja_env`). + Dictionary of Jinja filters to add to the Catalog's Jinja + environment (or the one passed in `jinja_env`). tests: Dictionary of Jinja tests to add to the Catalog's Jinja environment @@ -63,34 +67,36 @@ class Catalog: extensions: List of Jinja extensions to add to the Catalog's Jinja environment - (or the one passed in `jinja_env`). The `jinja2.ext.do` extension is - always added at the end of these. + (or the one passed in `jinja_env`). The `jinja2.ext.do` extension + is always added at the end of these. jinja_env: - Custom Jinja environment to use. This argument is useful to reuse an - existing Jinja Environment from your web framework. + Custom Jinja environment to use. This argument is useful to reuse + an existing Jinja Environment from your web framework. root_url: - Add this prefix to every asset URL of the static middleware. By default, - it is `/static/components/`, so, for example, the URL of the CSS file of - a `Card` component is `/static/components/Card.css`. + Add this prefix to every asset URL of the static middleware. + By default, it is `/static/components/`, so, for example, + the URL of the CSS file of a `Card` component is + `/static/components/Card.css`. - You can also change this argument so the assets are requested from a - Content Delivery Network (CDN) in production, for example, + You can also change this argument so the assets are requested from + a Content Delivery Network (CDN) in production, for example, `root_url="https://example.my-cdn.com/"`. file_ext: The extensions the components files have. By default, ".jinja". - This argument can also be a list to allow more than one type of file to - be a component. + This argument can also be a list to allow more than one type of + file to be a component. use_cache: Cache the metadata of the component in memory. auto_reload: - Used with `use_cache`. If `True`, the last-modified date of the component - file is checked every time to see if the cache is up-to-date. + Used with `use_cache`. If `True`, the last-modified date of the + component file is checked every time to see if the cache + is up-to-date. Set to `False` in production. @@ -98,12 +104,12 @@ class Catalog: If `True`, inserts a hash of the updated time into the URL of the asset files (after the name but before the extension). - This strategy encourages long-term caching while ensuring that new copies - are only requested when the content changes, as any modification alters the - fingerprint and thus the filename. + This strategy encourages long-term caching while ensuring that + new copies are only requested when the content changes, as any + modification alters the fingerprint and thus the filename. - **WARNING**: Only works if the server knows how to filter the fingerprint - to get the real name of the file. + **WARNING**: Only works if the server knows how to filter the + fingerprint to get the real name of the file. Attributes: @@ -124,12 +130,11 @@ class Catalog: "file_ext", "jinja_env", "fingerprint", - "collected_css", - "collected_js", "auto_reload", "use_cache", "_tmpl_globals", "_cache", + "_key", ) def __init__( @@ -141,14 +146,14 @@ def __init__( extensions: "list | None" = None, jinja_env: "jinja2.Environment | None" = None, root_url: str = DEFAULT_URL_ROOT, - file_ext: "str | tuple[str, ...]" = DEFAULT_EXTENSION, + file_ext: "str | list[str] | tuple[str, ...]" = DEFAULT_EXTENSION, use_cache: bool = True, auto_reload: bool = True, fingerprint: bool = False, ) -> None: self.prefixes: dict[str, jinja2.FileSystemLoader] = {} - self.collected_css: list[str] = [] - self.collected_js: list[str] = [] + if isinstance(file_ext, list): + file_ext = tuple(file_ext) self.file_ext = file_ext self.use_cache = use_cache self.auto_reload = auto_reload @@ -186,11 +191,49 @@ def __init__( self._tmpl_globals: "t.MutableMapping[str, t.Any] | None" = None self._cache: dict[str, dict] = {} + self._key = id(self) + + def __del__(self) -> None: + name = f"collected_css_{self._key}" + if name in collected_css: + del collected_css[name] + name = f"collected_js_{self._key}" + if name in collected_js: + del collected_js[name] + + @property + def collected_css(self) -> list[str]: + if self._key not in collected_css: + name = f"collected_css_{self._key}" + collected_css[self._key] = ContextVar(name, default=[]) + return collected_css[self._key].get() + + @collected_css.setter + def collected_css(self, value: list[str]) -> None: + if self._key not in collected_css: + name = f"collected_css_{self._key}" + collected_css[self._key] = ContextVar(name, default=[]) + collected_css[self._key].set(list(value)) + + @property + def collected_js(self) -> list[str]: + if self._key not in collected_js: + name = f"collected_js_{self._key}" + collected_js[self._key] = ContextVar(name, default=[]) + return collected_js[self._key].get() + + @collected_js.setter + def collected_js(self, value: list[str]) -> None: + if self._key not in collected_js: + name = f"collected_js_{self._key}" + collected_js[self._key] = ContextVar(name, default=[]) + collected_js[self._key].set(list(value)) @property def paths(self) -> list[Path]: """ - A helper property that returns a list of all the components folder paths. + A helper property that returns a list of all the components + folder paths. """ _paths = [] for loader in self.prefixes.values(): @@ -204,15 +247,17 @@ def add_folder( prefix: str = DEFAULT_PREFIX, ) -> None: """ - Add a folder path from where to search for components, optionally under a prefix. + Add a folder path from where to search for components, optionally + under a prefix. The prefix acts like a namespace. For example, the name of a `components/Card.jinja` component is, by default, "Card", but under the prefix "common", it becomes "common.Card". - The rule for subfolders remains the same: a `components/wrappers/Card.jinja` - name is, by default, "wrappers.Card", but under the prefix "common", - it becomes "common.wrappers.Card". + The rule for subfolders remains the same: a + `components/wrappers/Card.jinja` name is, by default, + "wrappers.Card", but under the prefix "common", it + becomes "common.wrappers.Card". If there is more than one component with the same name in multiple added folders under the same prefix, the one in the folder added @@ -228,26 +273,35 @@ def add_folder( have. The default is empty. """ - prefix = prefix.strip().strip(f"{DELIMITER}{SLASH}").replace(SLASH, DELIMITER) + prefix = ( + prefix.strip() + .strip(f"{DELIMITER}{SLASH}") + .replace(SLASH, DELIMITER) + ) root_path = str(root_path) if prefix in self.prefixes: loader = self.prefixes[prefix] if root_path in loader.searchpath: return - logger.debug(f"Adding folder `{root_path}` with the prefix `{prefix}`") + logger.debug( + f"Adding folder `{root_path}` with the prefix `{prefix}`" + ) loader.searchpath.append(root_path) else: - logger.debug(f"Adding folder `{root_path}` with the prefix `{prefix}`") + logger.debug( + f"Adding folder `{root_path}` with the prefix `{prefix}`" + ) self.prefixes[prefix] = jinja2.FileSystemLoader(root_path) def add_module(self, module: t.Any, *, prefix: str | None = None) -> None: """ - Reads an absolute path from `module.components_path` and an optional prefix - from `module.prefix`, then calls `Catalog.add_folder(path, prefix)`. + Reads an absolute path from `module.components_path` and an optional + prefix from `module.prefix`, then calls + `Catalog.add_folder(path, prefix)`. - The prefix can also be passed as an argument instead of being read from - the module. + The prefix can also be passed as an argument instead of being read + from the module. This method exists to make it easy and consistent to have components installable as Python libraries. @@ -263,7 +317,9 @@ def add_module(self, module: t.Any, *, prefix: str | None = None) -> None: """ mprefix = ( - prefix if prefix is not None else getattr(module, "prefix", DEFAULT_PREFIX) + prefix + if prefix is not None + else getattr(module, "prefix", DEFAULT_PREFIX) ) self.add_folder(module.components_path, prefix=mprefix) @@ -314,16 +370,25 @@ def irender( if source: logger.debug("Rendering from source %s", __name) - component = self._get_from_source(name=name, prefix=prefix, source=source) + component = self._get_from_source( + name=name, prefix=prefix, source=source + ) elif self.use_cache: logger.debug("Rendering from cache or file %s", __name) - component = self._get_from_cache(prefix=prefix, name=name, file_ext=file_ext) + component = self._get_from_cache( + prefix=prefix, name=name, file_ext=file_ext + ) else: logger.debug("Rendering from file %s", __name) - component = self._get_from_file(prefix=prefix, name=name, file_ext=file_ext) + component = self._get_from_file( + prefix=prefix, name=name, file_ext=file_ext + ) root_path = component.path.parent if component.path else None + css = self.collected_css + js = self.collected_js + for url in component.css: if ( root_path @@ -332,8 +397,8 @@ def irender( ): url = self._fingerprint(root_path, url) - if url not in self.collected_css: - self.collected_css.append(url) + if url not in css: + css.append(url) for url in component.js: if ( @@ -343,8 +408,8 @@ def irender( ): url = self._fingerprint(root_path, url) - if url not in self.collected_js: - self.collected_js.append(url) + if url not in js: + js.append(url) attrs = attrs.as_dict if isinstance(attrs, HTMLAttrs) else attrs attrs.update(kw) @@ -368,7 +433,8 @@ def get_middleware( **kwargs, ) -> ComponentsMiddleware: """ - Wraps you application with [Withenoise](https://whitenoise.readthedocs.io/), + Wraps you application with + [Withenoise](https://whitenoise.readthedocs.io/), a static file serving middleware. Tecnically not neccesary if your components doesn't use static assets @@ -380,13 +446,15 @@ def get_middleware( A WSGI application allowed_ext: - A list of file extensions the static middleware is allowed to read - and return. By default, is just ".css", ".js", and ".mjs". + A list of file extensions the static middleware is allowed to + read and return. By default, is just ".css", ".js", and ".mjs". """ logger.debug("Creating middleware") middleware = ComponentsMiddleware( - application=application, allowed_ext=tuple(allowed_ext or []), **kwargs + application=application, + allowed_ext=tuple(allowed_ext or []), + **kwargs, ) for prefix, loader in self.prefixes.items(): url_prefix = get_url_prefix(prefix) @@ -396,7 +464,11 @@ def get_middleware( return middleware - def get_source(self, cname: str, file_ext: "tuple[str, ...] | str" = "") -> str: + def get_source( + self, + cname: str, + file_ext: "tuple[str, ...] | str" = "", + ) -> str: """ A helper method that returns the source file of a component. """ @@ -445,12 +517,26 @@ def _fingerprint(self, root: Path, filename: str) -> str: return f"{parent}{stem}-{fingerprint}{ext}" - def _get_from_source(self, *, name: str, prefix: str, source: str) -> Component: + def _get_from_source( + self, + *, + name: str, + prefix: str, + source: str, + ) -> Component: tmpl = self.jinja_env.from_string(source, globals=self._tmpl_globals) - component = Component(name=name, prefix=prefix, source=source, tmpl=tmpl) + component = Component( + name=name, prefix=prefix, source=source, tmpl=tmpl + ) return component - def _get_from_cache(self, *, prefix: str, name: str, file_ext: str) -> Component: + def _get_from_cache( + self, + *, + prefix: str, + name: str, + file_ext: str, + ) -> Component: key = f"{prefix}.{name}.{file_ext}" cache = self._from_cache(key) if cache: @@ -461,7 +547,9 @@ def _get_from_cache(self, *, prefix: str, name: str, file_ext: str) -> Component return component logger.debug("Loading %s", key) - component = self._get_from_file(prefix=prefix, name=name, file_ext=file_ext) + component = self._get_from_file( + prefix=prefix, name=name, file_ext=file_ext + ) self._to_cache(key, component) return component @@ -475,10 +563,16 @@ def _from_cache(self, key: str) -> dict[str, t.Any]: def _to_cache(self, key: str, component: Component) -> None: self._cache[key] = component.serialize() - def _get_from_file(self, *, prefix: str, name: str, file_ext: str) -> Component: - path, tmpl_name = self._get_component_path(prefix, name, file_ext=file_ext) + def _get_from_file( + self, *, prefix: str, name: str, file_ext: str + ) -> Component: + path, tmpl_name = self._get_component_path( + prefix, name, file_ext=file_ext + ) component = Component(name=name, prefix=prefix, path=path) - component.tmpl = self.jinja_env.get_template(tmpl_name, globals=self._tmpl_globals) + component.tmpl = self.jinja_env.get_template( + tmpl_name, globals=self._tmpl_globals + ) return component def _split_name(self, cname: str) -> tuple[str, str]: @@ -492,7 +586,10 @@ def _split_name(self, cname: str) -> tuple[str, str]: return DEFAULT_PREFIX, cname def _get_component_path( - self, prefix: str, name: str, file_ext: "tuple[str, ...] | str" = "" + self, + prefix: str, + name: str, + file_ext: "tuple[str, ...] | str" = "", ) -> tuple[Path, str]: name = name.replace(DELIMITER, SLASH) root_paths = self.prefixes[prefix].searchpath @@ -512,7 +609,10 @@ def _get_component_path( filepath = f"{relfolder}/{filename}" else: filepath = filename - if filepath.startswith(name_dot) and filepath.endswith(file_ext): + if ( + filepath.startswith(name_dot) and + filepath.endswith(file_ext) + ): return Path(curr_folder) / filename, filepath raise ComponentNotFound( diff --git a/tests/test_render.py b/tests/test_render.py index 1e52cda..0f37ea7 100644 --- a/tests/test_render.py +++ b/tests/test_render.py @@ -1,6 +1,7 @@ import time from pathlib import Path from textwrap import dedent +from threading import Thread import jinja2 import pytest @@ -923,7 +924,7 @@ def test_vue_like_syntax(catalog, folder): ) html = catalog.render("Caller") print(html) - expected = """4 2+2 {'lorem': 'ipsum'} False""".strip() + expected = """4 2+2 {'lorem': 'ipsum'} False""".strip() assert html == Markup(expected) @@ -937,7 +938,7 @@ def test_jinja_like_syntax(catalog, folder): ) html = catalog.render("Caller") print(html) - expected = """4 2+2 {'lorem': 'ipsum'} False""".strip() + expected = """4 2+2 {'lorem': 'ipsum'} False""".strip() assert html == Markup(expected) @@ -951,7 +952,7 @@ def test_mixed_syntax(catalog, folder): ) html = catalog.render("Caller") print(html) - expected = """4 {{2+2}} {'lorem': 'ipsum'} False""".strip() + expected = """4 {{2+2}} {'lorem': 'ipsum'} False""".strip() assert html == Markup(expected) @@ -990,3 +991,130 @@ def test_slots(catalog, folder, autoescape):

Default

""".strip() assert html == Markup(expected) + + +class ThreadWithReturnValue(Thread): + def __init__(self, group=None, target=None, name=None, args=None, kwargs=None): + args = args or () + kwargs = kwargs or {} + Thread.__init__( + self, + group=group, + target=target, + name=name, + args=args, + kwargs=kwargs, + ) + self._target = target + self._args = args + self._kwargs = kwargs + self._return = None + + def run(self): + if self._target is not None: + self._return = self._target(*self._args, **self._kwargs) + + def join(self, *args): + Thread.join(self, *args) + return self._return + + +def test_thread_safety_of_render_assets(catalog, folder): + NUM_THREADS = 5 + + child_tmpl = """ +{#css "c{i}.css" #} +{#js "c{i}.js" #} +

Child {i}

""".strip() + + parent_tmpl = """ +{{ catalog.render_assets() }} +{{ content }}""".strip() + + comp_tmpl = """ +{#css "a{i}.css", "b{i}.css" #} +{#js "a{i}.js", "b{i}.js" #} +""".strip() + + expected_tmpl = """ + + + + + + +

Child {i}

""".strip() + + def render(i): + return catalog.render(f"Page{i}") + + + for i in range(NUM_THREADS): + si = str(i) + child_name = f"Child{i}.jinja" + child_src = child_tmpl.replace("{i}", si) + + parent_name = f"Parent{i}.jinja" + parent_src = parent_tmpl.replace("{i}", si) + + comp_name = f"Page{i}.jinja" + comp_src = comp_tmpl.replace("{i}", si) + + (folder / child_name).write_text(child_src) + (folder / comp_name).write_text(comp_src) + (folder / parent_name).write_text(parent_src) + + threads = [] + + for i in range(NUM_THREADS): + thread = ThreadWithReturnValue(target=render, args=(i,)) + threads.append(thread) + thread.start() + + results = [thread.join() for thread in threads] + + for i, result in enumerate(results): + expected = expected_tmpl.replace("{i}", str(i)) + print(f"---- EXPECTED {i}----") + print(expected) + print(f"---- RESULT {i}----") + print(result) + assert result == Markup(expected) + + +def test_same_thread_assets_independence(catalog, folder): + catalog2 = jinjax.Catalog() + catalog2.add_folder(folder) + + print(catalog._key) + print(catalog2._key) + + (folder / "Parent.jinja").write_text(""" +{{ catalog.render_assets() }} +{{ content }}""".strip()) + + (folder / "Comp1.jinja").write_text(""" +{#css "a.css" #} +{#js "a.js" #} +""".strip()) + + (folder / "Comp2.jinja").write_text(""" +{#css "b.css" #} +{#js "b.js" #} +""".strip()) + + expected_1 = """ + +""".strip() + + expected_2 = """ + +""".strip() + + html1 = catalog.render("Comp1") + # `irender` instead of `render` so the assets are not cleared + html2 = catalog2.irender("Comp2") + print(html1) + print(html2) + assert html1 == Markup(expected_1) + assert html2 == Markup(expected_2)