From 327fd908cd48720a1db0a3804ec474840b88d571 Mon Sep 17 00:00:00 2001 From: Bruno Dufour Date: Thu, 5 Dec 2024 20:56:35 -0500 Subject: [PATCH 1/3] Introduce dynamic processor registration --- .../_plugins/nativeapp/codegen/compiler.py | 57 +++++++++++++------ .../setup/native_app_setup_processor.py | 7 +++ .../codegen/snowpark/python_processor.py | 2 + .../codegen/templates/templates_processor.py | 2 + .../cli/api/project/definition_conversion.py | 3 +- tests_integration/helpers/test_v1_to_v2.py | 10 ++-- 6 files changed, 59 insertions(+), 22 deletions(-) diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py index 07302e6356..2151c25794 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py @@ -33,7 +33,6 @@ from snowflake.cli._plugins.nativeapp.codegen.templates.templates_processor import ( TemplatesProcessor, ) -from snowflake.cli._plugins.nativeapp.feature_flags import FeatureFlag from snowflake.cli.api.cli_global_context import get_cli_context from snowflake.cli.api.console import cli_console as cc from snowflake.cli.api.metrics import CLICounterField @@ -41,15 +40,7 @@ ProcessorMapping, ) -SNOWPARK_PROCESSOR = "snowpark" -NA_SETUP_PROCESSOR = "native app setup" -TEMPLATES_PROCESSOR = "templates" - -_REGISTERED_PROCESSORS_BY_NAME = { - SNOWPARK_PROCESSOR: SnowparkAnnotationProcessor, - NA_SETUP_PROCESSOR: NativeAppSetupProcessor, - TEMPLATES_PROCESSOR: TemplatesProcessor, -} +ProcessorClassType = type[ArtifactProcessor] class NativeAppCompiler: @@ -66,10 +57,28 @@ def __init__( bundle_ctx: BundleContext, ): self._assert_absolute_paths(bundle_ctx) + self._processor_classes_by_name: Dict[str, ProcessorClassType] = {} self._bundle_ctx = bundle_ctx # dictionary of all processors created and shared between different artifact objects. self.cached_processors: Dict[str, ArtifactProcessor] = {} + self._register(SnowparkAnnotationProcessor) + self._register(NativeAppSetupProcessor) + self._register(TemplatesProcessor) + + def _register(self, processor_cls: ProcessorClassType): + """ + Registers a processor class to enable. + """ + + name = getattr(processor_cls, "NAME", None) + assert name is not None + + if name in self._processor_classes_by_name: + raise ValueError(f"Processor {name} is already registered") + + self._processor_classes_by_name[str(name)] = processor_cls + @staticmethod def _assert_absolute_paths(bundle_ctx: BundleContext): for name in ["Project", "Deploy", "Bundle", "Generated"]: @@ -128,8 +137,8 @@ def _try_create_processor( if current_processor is not None: return current_processor - processor_factory = _REGISTERED_PROCESSORS_BY_NAME.get(processor_name) - if processor_factory is None: + processor_cls = self._processor_classes_by_name.get(processor_name) + if processor_cls is None: # No registered processor with the specified name return None @@ -141,7 +150,7 @@ def _try_create_processor( processor_ctx.generated_root = ( self._bundle_ctx.generated_root / processor_subdirectory ) - current_processor = processor_factory(processor_ctx) + current_processor = processor_cls(processor_ctx) self.cached_processors[processor_name] = current_processor return current_processor @@ -154,6 +163,22 @@ def _should_invoke_processors(self): return False def _is_enabled(self, processor: ProcessorMapping) -> bool: - if processor.name.lower() == NA_SETUP_PROCESSOR: - return FeatureFlag.ENABLE_NATIVE_APP_PYTHON_SETUP.is_enabled() - return True + """ + Determines is a process is enabled. All processors are considered enabled + unless they are explicitly disabled, typically via a feature flag. + """ + processor_name = processor.name.lower() + processor_cls = self._processor_classes_by_name.get(processor_name) + if processor_cls is None: + # Unknown processor, consider it enabled, even though trying to + # invoke it later will raise an exception + return True + + # if the processor class defines a static method named "is_enabled", then + # call it. Otherwise, it's considered enabled by default. + is_enabled_fn = getattr(processor_cls, "is_enabled", None) + if is_enabled_fn is None: + # No custom static method provided, enabled by default + return True + + return is_enabled_fn() diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/setup/native_app_setup_processor.py b/src/snowflake/cli/_plugins/nativeapp/codegen/setup/native_app_setup_processor.py index 06be6f01e8..b643f66304 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/setup/native_app_setup_processor.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/setup/native_app_setup_processor.py @@ -36,6 +36,7 @@ SandboxEnvBuilder, execute_script_in_sandbox, ) +from snowflake.cli._plugins.nativeapp.feature_flags import FeatureFlag from snowflake.cli._plugins.stage.diff import to_stage_path from snowflake.cli.api.console import cli_console as cc from snowflake.cli.api.project.schemas.v1.native_app.path_mapping import ( @@ -74,9 +75,15 @@ def safe_set(d: dict, *keys: str, **kwargs) -> None: class NativeAppSetupProcessor(ArtifactProcessor): + NAME = "native app setup" + def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) + @staticmethod + def is_enabled() -> bool: + return FeatureFlag.ENABLE_NATIVE_APP_PYTHON_SETUP.is_enabled() + def process( self, artifact_to_process: PathMapping, diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/snowpark/python_processor.py b/src/snowflake/cli/_plugins/nativeapp/codegen/snowpark/python_processor.py index 0241899388..58a9eb2baa 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/snowpark/python_processor.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/snowpark/python_processor.py @@ -164,6 +164,8 @@ class SnowparkAnnotationProcessor(ArtifactProcessor): and generate SQL code for creation of extension functions based on those discovered objects. """ + NAME = "snowpark" + def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/templates/templates_processor.py b/src/snowflake/cli/_plugins/nativeapp/codegen/templates/templates_processor.py index a16e1dc933..b6984f67c2 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/templates/templates_processor.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/templates/templates_processor.py @@ -49,6 +49,8 @@ class TemplatesProcessor(ArtifactProcessor): Processor class to perform template expansion on all relevant artifacts (specified in the project definition file). """ + NAME = "templates" + def expand_templates_in_file( self, src: Path, dest: Path, template_context: dict[str, Any] | None = None ) -> None: diff --git a/src/snowflake/cli/api/project/definition_conversion.py b/src/snowflake/cli/api/project/definition_conversion.py index 1f76e998d2..7834586a07 100644 --- a/src/snowflake/cli/api/project/definition_conversion.py +++ b/src/snowflake/cli/api/project/definition_conversion.py @@ -13,7 +13,6 @@ bundle_artifacts, ) from snowflake.cli._plugins.nativeapp.bundle_context import BundleContext -from snowflake.cli._plugins.nativeapp.codegen.compiler import TEMPLATES_PROCESSOR from snowflake.cli._plugins.nativeapp.codegen.templates.templates_processor import ( TemplatesProcessor, ) @@ -457,7 +456,7 @@ def _convert_templates_in_files( artifact for artifact in pkg_model.artifacts for processor in artifact.processors - if processor.name == TEMPLATES_PROCESSOR + if processor.name.lower() == TemplatesProcessor.NAME ] if not in_memory and artifacts_to_template: metrics.set_counter(CLICounterField.TEMPLATES_PROCESSOR, 1) diff --git a/tests_integration/helpers/test_v1_to_v2.py b/tests_integration/helpers/test_v1_to_v2.py index bfc594ff0b..a4aae93244 100644 --- a/tests_integration/helpers/test_v1_to_v2.py +++ b/tests_integration/helpers/test_v1_to_v2.py @@ -2,7 +2,9 @@ import pytest -from snowflake.cli._plugins.nativeapp.codegen.compiler import TEMPLATES_PROCESSOR +from snowflake.cli._plugins.nativeapp.codegen.templates.templates_processor import ( + TemplatesProcessor, +) from tests.nativeapp.factories import ProjectV11Factory @@ -44,10 +46,10 @@ def test_v1_to_v2_converts_templates_in_files(temp_dir, runner): pdf__native_app__package__name="my_pkg", pdf__native_app__application__name="my_app", pdf__native_app__artifacts=[ - dict(src="templated.txt", processors=[TEMPLATES_PROCESSOR]), + dict(src="templated.txt", processors=[TemplatesProcessor.NAME]), dict(src="untemplated.txt"), - dict(src="app/*", processors=[TEMPLATES_PROCESSOR]), - dict(src="nested/*", processors=[TEMPLATES_PROCESSOR]), + dict(src="app/*", processors=[TemplatesProcessor.NAME]), + dict(src="nested/*", processors=[TemplatesProcessor.NAME]), ], files={ filename: source_contents From 5e6929e5d41b118ab9bad0e84f496387deb52c56 Mon Sep 17 00:00:00 2001 From: Bruno Dufour Date: Thu, 5 Dec 2024 21:52:05 -0500 Subject: [PATCH 2/3] Add test for disabled processors --- .../_plugins/nativeapp/codegen/compiler.py | 8 ++-- tests/nativeapp/codegen/test_compiler.py | 38 +++++++++++++++++++ 2 files changed, 42 insertions(+), 4 deletions(-) diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py index 2151c25794..0339ff9467 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py @@ -62,11 +62,11 @@ def __init__( # dictionary of all processors created and shared between different artifact objects. self.cached_processors: Dict[str, ArtifactProcessor] = {} - self._register(SnowparkAnnotationProcessor) - self._register(NativeAppSetupProcessor) - self._register(TemplatesProcessor) + self.register(SnowparkAnnotationProcessor) + self.register(NativeAppSetupProcessor) + self.register(TemplatesProcessor) - def _register(self, processor_cls: ProcessorClassType): + def register(self, processor_cls: ProcessorClassType): """ Registers a processor class to enable. """ diff --git a/tests/nativeapp/codegen/test_compiler.py b/tests/nativeapp/codegen/test_compiler.py index 4ec64ef983..7da3382588 100644 --- a/tests/nativeapp/codegen/test_compiler.py +++ b/tests/nativeapp/codegen/test_compiler.py @@ -13,10 +13,12 @@ # limitations under the License. import re from pathlib import Path +from typing import Optional import pytest from snowflake.cli._plugins.nativeapp.bundle_context import BundleContext from snowflake.cli._plugins.nativeapp.codegen.artifact_processor import ( + ArtifactProcessor, UnsupportedArtifactProcessorError, ) from snowflake.cli._plugins.nativeapp.codegen.compiler import NativeAppCompiler @@ -29,6 +31,10 @@ from snowflake.cli.api.project.schemas.project_definition import ( build_project_definition, ) +from snowflake.cli.api.project.schemas.v1.native_app.path_mapping import ( + PathMapping, + ProcessorMapping, +) @pytest.fixture() @@ -114,3 +120,35 @@ def test_find_and_execute_processors_exception(test_proj_def, test_compiler): with pytest.raises(UnsupportedArtifactProcessorError): test_compiler.compile_artifacts() + + +class TestProcessor(ArtifactProcessor): + NAME = "test_processor" + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + assert False # never invoked + + @staticmethod + def is_enabled(): + return False + + def process( + self, + artifact_to_process: PathMapping, + processor_mapping: Optional[ProcessorMapping], + **kwargs, + ) -> None: + assert False # never invoked + + +def test_skips_disabled_processors(test_proj_def, test_compiler): + pkg_model = test_proj_def.entities["pkg"] + pkg_model.artifacts = [ + {"dest": "./", "src": "app/*", "processors": ["test_processor"]} + ] + test_compiler = NativeAppCompiler(_get_bundle_context(pkg_model)) + test_compiler.register(TestProcessor) + + # TestProcessor is never invoked, otherwise calling its methods will make the test fail + test_compiler.compile_artifacts() From dceb994e45f58369aa753d17bd220a6acfbceef3 Mon Sep 17 00:00:00 2001 From: Bruno Dufour Date: Fri, 6 Dec 2024 12:01:45 -0500 Subject: [PATCH 3/3] Addressed PR comments --- src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py index 0339ff9467..7681f1fab3 100644 --- a/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py +++ b/src/snowflake/cli/_plugins/nativeapp/codegen/compiler.py @@ -176,9 +176,5 @@ def _is_enabled(self, processor: ProcessorMapping) -> bool: # if the processor class defines a static method named "is_enabled", then # call it. Otherwise, it's considered enabled by default. - is_enabled_fn = getattr(processor_cls, "is_enabled", None) - if is_enabled_fn is None: - # No custom static method provided, enabled by default - return True - + is_enabled_fn = getattr(processor_cls, "is_enabled", lambda: True) return is_enabled_fn()