diff --git a/great_expectations/render/renderer/notebook_renderer.py b/great_expectations/render/renderer/notebook_renderer.py index fa636df53f5c..0ce614caafe0 100644 --- a/great_expectations/render/renderer/notebook_renderer.py +++ b/great_expectations/render/renderer/notebook_renderer.py @@ -4,6 +4,7 @@ import nbformat +from great_expectations.compatibility.typing_extensions import override from great_expectations.render.renderer.renderer import Renderer from great_expectations.util import ( convert_json_string_to_be_python_compliant, @@ -48,7 +49,7 @@ def add_code_cell( code = lint_code(code).rstrip("\n") cell = nbformat.v4.new_code_cell(code) - self._notebook["cells"].append(cell) + self._notebook["cells"].append(cell) # type: ignore[index] # _notebook could be None def add_markdown_cell(self, markdown: str) -> None: """ @@ -60,7 +61,7 @@ def add_markdown_cell(self, markdown: str) -> None: Nothing, adds a cell to the class instance notebook """ cell = nbformat.v4.new_markdown_cell(markdown) - self._notebook["cells"].append(cell) + self._notebook["cells"].append(cell) # type: ignore[index] # _notebook could be None @classmethod def write_notebook_to_disk( @@ -75,6 +76,7 @@ def write_notebook_to_disk( with open(notebook_file_path, "w") as f: nbformat.write(notebook, f) + @override def render(self, **kwargs: dict) -> nbformat.NotebookNode: """ Render a notebook from parameters. diff --git a/great_expectations/render/view/view.py b/great_expectations/render/view/view.py index 0c40034e3c16..58fd022082ec 100644 --- a/great_expectations/render/view/view.py +++ b/great_expectations/render/view/view.py @@ -1,8 +1,11 @@ +from __future__ import annotations + import datetime import json import re from collections import OrderedDict from string import Template as pTemplate +from typing import TYPE_CHECKING, Any, ClassVar, Iterable, Mapping from uuid import uuid4 import mistune @@ -14,8 +17,10 @@ select_autoescape, ) +from great_expectations.compatibility.typing_extensions import override + try: - from jinja2 import contextfilter + from jinja2 import contextfilter # type: ignore[attr-defined] # for jinja 2.0 except ImportError: from jinja2 import pass_context as contextfilter @@ -26,10 +31,9 @@ RenderedDocumentContent, ) - -class NoOpTemplate: - def render(self, document): - return document +if TYPE_CHECKING: + from jinja2 import BaseLoader + from jinja2 import Template as jTemplate class PrettyPrintTemplate: @@ -55,7 +59,7 @@ class DefaultJinjaView: """ - _template = NoOpTemplate + _template: ClassVar[str] def __init__( self, custom_styles_directory=None, custom_views_directory=None @@ -66,7 +70,7 @@ def __init__( templates_loader = PackageLoader("great_expectations", "render/view/templates") styles_loader = PackageLoader("great_expectations", "render/view/static/styles") - loaders = [templates_loader, styles_loader] + loaders: list[BaseLoader] = [templates_loader, styles_loader] if self.custom_styles_directory: loaders.append(FileSystemLoader(self.custom_styles_directory)) if self.custom_views_directory: @@ -109,17 +113,16 @@ def render(self, document, template=None, **kwargs): document = document.to_json_dict() return t.render(document, **kwargs) - def _get_template(self, template): - if template is None: - return NoOpTemplate - - template = self.env.get_template(template) + def _get_template(self, template_str: str) -> jTemplate: + template = self.env.get_template(template_str) template.globals["now"] = lambda: datetime.datetime.now(datetime.timezone.utc) return template - @contextfilter - def add_data_context_id_to_url(self, jinja_context, url, add_datetime=True): + @contextfilter # type: ignore[misc] # untyped 3rd party decorator + def add_data_context_id_to_url( + self, jinja_context: Any, url: str, add_datetime: bool = True + ) -> str: data_context_id = jinja_context.get("data_context_id") if add_datetime: datetime_iso_string = datetime.datetime.now(datetime.timezone.utc).strftime( @@ -131,15 +134,15 @@ def add_data_context_id_to_url(self, jinja_context, url, add_datetime=True): url += data_context_id return url - @contextfilter + @contextfilter # type: ignore[misc] # untyped 3rd party decorator def render_content_block( # noqa: PLR0911, PLR0913, PLR0912 self, - jinja_context, - content_block, - index=None, - content_block_id=None, + jinja_context: Any, + content_block: str | list | dict | RenderedComponentContent, + index: Any = None, + content_block_id: Any = None, render_to_markdown: bool = False, - ): + ) -> RenderedComponentContent | dict | str: """ :param jinja_context: @@ -187,7 +190,7 @@ def render_content_block( # noqa: PLR0911, PLR0913, PLR0912 template_filename = f"markdown_{content_block_type}.j2" else: template_filename = f"{content_block_type}.j2" - template = self._get_template(template=template_filename) + template = self._get_template(template_str=template_filename) if content_block_id: return template.render( jinja_context, @@ -201,7 +204,7 @@ def render_content_block( # noqa: PLR0911, PLR0913, PLR0912 ) def render_dict_values( - self, context, dict_, index=None, content_block_id=None + self, context: Any, dict_: dict, index: Any = None, content_block_id: Any = None ) -> None: for key, val in dict_.items(): if key.startswith("_"): @@ -210,18 +213,22 @@ def render_dict_values( context, val, index, content_block_id ) - @contextfilter + @contextfilter # type: ignore[misc] # untyped 3rd party decorator def render_bootstrap_table_data( - self, context, table_data, index=None, content_block_id=None + self, + context: Any, + table_data: Iterable[dict], + index: Any | None = None, + content_block_id: str | None = None, ): for table_data_dict in table_data: self.render_dict_values(context, table_data_dict, index, content_block_id) return table_data - def get_html_escaped_json_string_from_dict(self, source_dict): + def get_html_escaped_json_string_from_dict(self, source_dict: dict) -> str: return json.dumps(source_dict).replace('"', '\\"').replace('"', """) - def attributes_dict_to_html_string(self, attributes_dict, prefix=""): + def attributes_dict_to_html_string(self, attributes_dict: dict, prefix=""): attributes_string = "" if prefix: prefix += "-" @@ -229,7 +236,7 @@ def attributes_dict_to_html_string(self, attributes_dict, prefix=""): attributes_string += f'{prefix}{attribute}="{value}" ' return attributes_string - def render_styling(self, styling): + def render_styling(self, styling: Mapping) -> str: """Adds styling information suitable for an html tag. Example styling block:: @@ -290,7 +297,7 @@ def render_styling(self, styling): return styling_string - def render_styling_from_string_template(self, template): + def render_styling_from_string_template(self, template: dict | OrderedDict) -> str: # NOTE: We should add some kind of type-checking to template """This method is a thin wrapper use to call `render_styling` from within jinja templates.""" if not isinstance(template, (dict, OrderedDict)): @@ -435,7 +442,8 @@ def _validate_document(self, document) -> None: class DefaultJinjaPageView(DefaultJinjaView): _template = "page.j2" - def _validate_document(self, document) -> None: + @override + def _validate_document(self, document: RenderedDocumentContent) -> None: assert isinstance(document, RenderedDocumentContent) @@ -446,7 +454,8 @@ class DefaultJinjaIndexPageView(DefaultJinjaPageView): class DefaultJinjaSectionView(DefaultJinjaView): _template = "section.j2" - def _validate_document(self, document) -> None: + @override + def _validate_document(self, document: Mapping) -> None: assert isinstance( document["section"], dict ) # For now low-level views take dicts @@ -455,7 +464,8 @@ def _validate_document(self, document) -> None: class DefaultJinjaComponentView(DefaultJinjaView): _template = "component.j2" - def _validate_document(self, document) -> None: + @override + def _validate_document(self, document: Mapping) -> None: assert isinstance( document["content_block"], dict ) # For now low-level views take dicts @@ -466,7 +476,8 @@ class DefaultMarkdownPageView(DefaultJinjaView): Convert a document to markdown format. """ - def _validate_document(self, document: RenderedDocumentContent) -> bool: + @override + def _validate_document(self, document: RenderedDocumentContent) -> None: """ Validate that the document is of the appropriate type at runtime. """ @@ -474,7 +485,8 @@ def _validate_document(self, document: RenderedDocumentContent) -> bool: _template = "markdown_validation_results_page.j2" - def render(self, document, template=None, **kwargs): + @override + def render(self, document: list, template=None, **kwargs) -> list[str] | str: """ Handle list as well as single document """ @@ -490,7 +502,10 @@ def render(self, document, template=None, **kwargs): else: return super().render(document=document, template=template, **kwargs) - def render_string_template(self, template: pTemplate) -> pTemplate: + @override + def render_string_template( + self, template: pTemplate | dict | OrderedDict + ) -> pTemplate | str: """ Render string for markdown rendering. Bold all parameters and perform substitution. Args: @@ -547,19 +562,20 @@ def render_string_template(self, template: pTemplate) -> pTemplate: "$PARAMETER", "$$PARAMETER" ) - return pTemplate(template.get("template")).safe_substitute( + return pTemplate(template["template"]).safe_substitute( template.get("params", {}) ) - @contextfilter + @override + @contextfilter # type: ignore[misc] # untyped 3rd party decorator def render_content_block( # noqa: PLR0913 self, - jinja_context, - content_block, - index=None, - content_block_id=None, + jinja_context: Any, + content_block: Any, + index: Any | None = None, + content_block_id: str | None = None, render_to_markdown: bool = True, - ): + ) -> RenderedComponentContent | dict | str: """ Render a content block to markdown using jinja templates. Args: diff --git a/pyproject.toml b/pyproject.toml index 0ac35fdc6a4f..f1c5f6f59412 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -73,18 +73,24 @@ exclude = [ 'cli/upgrade_helpers/upgrade_helper_v13\.py', # 17 - This is legacy code and will not be typed. 'dataset/sparkdf_dataset\.py', # 3 - This is legacy code and will not be typed. 'dataset/sqlalchemy_dataset\.py', # 16 - This is legacy code and will not be typed. + 'core/usage_statistics/anonymizers/batch_anonymizer\.py', # 10 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/batch_request_anonymizer\.py', # 16 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/checkpoint_anonymizer\.py', # 16 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/data_docs_anonymizer\.py', # 5 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/datasource_anonymizer\.py', # 9 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/expectation_anonymizer\.py', # 6 - This code will be removed in 1.0 + 'core/usage_statistics/anonymizers/validation_operator_anonymizer\.py', # 5 - This code will be removed in 1.0 + 'render/renderer/v3/suite_edit_notebook_renderer\.py', # 11 - This is legacy code and will not be typed. + 'render/renderer/v3/suite_profile_notebook_renderer\.py', # 4 - This is legacy code and will not be typed. + 'render/renderer/suite_edit_notebook_renderer\.py', # 7 - This is legacy code and will not be typed. + 'render/renderer/suite_scaffold_notebook_renderer\.py', # 7 - This is legacy code and will not be typed. + 'render/renderer/datasource_new_notebook_renderer\.py', # 4 - This is legacy code and will not be typed. + 'render/renderer/checkpoint_new_notebook_renderer\.py', # 9 - This is legacy code and will not be typed. # END ALWAYS EXCLUDE SECTION ###################################################### # # ################################################################################# # TODO: complete typing for the following modules and remove from exclude list # number is the current number of typing errors for the excluded pattern - 'core/usage_statistics/anonymizers/batch_anonymizer\.py', # 10 - 'core/usage_statistics/anonymizers/batch_request_anonymizer\.py', # 16 - 'core/usage_statistics/anonymizers/checkpoint_anonymizer\.py', # 16 - 'core/usage_statistics/anonymizers/data_docs_anonymizer\.py', # 5 - 'core/usage_statistics/anonymizers/datasource_anonymizer\.py', # 9 - 'core/usage_statistics/anonymizers/expectation_anonymizer\.py', # 6 - 'core/usage_statistics/anonymizers/validation_operator_anonymizer\.py', # 5 'expectations/core/expect_column_values_to_be_of_type\.py', # 12 'expectations/core/expect_column_values_to_not_match_regex_list\.py', # 2 'expectations/core/expect_column_values_to_not_match_regex\.py', # 2 @@ -133,20 +139,12 @@ exclude = [ 'expectations/regex_based_column_map_expectation\.py', # 3 'expectations/row_conditions\.py', # 4 'expectations/set_based_column_map_expectation\.py', # 3 - 'render/renderer/checkpoint_new_notebook_renderer\.py', # 9 'render/renderer/content_block/content_block\.py', # 5 'render/renderer/content_block/exception_list_content_block\.py', # 4 - 'render/renderer/datasource_new_notebook_renderer\.py', # 4 - 'render/renderer/notebook_renderer\.py', # 2 'render/renderer/page_renderer\.py', # 10 'render/renderer/profiling_results_overview_section_renderer\.py', # 2 'render/renderer/site_builder\.py', # 3 'render/renderer/slack_renderer\.py', # 9 - 'render/renderer/suite_edit_notebook_renderer\.py', # 7 - 'render/renderer/suite_scaffold_notebook_renderer\.py', # 7 - 'render/renderer/v3/suite_edit_notebook_renderer\.py', # 11 - 'render/renderer/v3/suite_profile_notebook_renderer\.py', # 4 - 'render/view/view\.py', # 11 'rule_based_profiler/domain_builder/map_metric_column_domain_builder\.py', # 8 'rule_based_profiler/estimators/bootstrap_numeric_range_estimator\.py', # 8 'rule_based_profiler/estimators/kde_numeric_range_estimator\.py', # 7