diff --git a/.changes/unreleased/Fixes-20250110-155824.yaml b/.changes/unreleased/Fixes-20250110-155824.yaml new file mode 100644 index 00000000000..2a7fcbcbf7d --- /dev/null +++ b/.changes/unreleased/Fixes-20250110-155824.yaml @@ -0,0 +1,6 @@ +kind: Fixes +body: Fix for custom fields in generic test config for not_null and unique tests +time: 2025-01-10T15:58:24.479245-05:00 +custom: + Author: gshank + Issue: "11208" diff --git a/core/dbt/parser/generic_test_builders.py b/core/dbt/parser/generic_test_builders.py index 1c9478734bf..bbd17603d04 100644 --- a/core/dbt/parser/generic_test_builders.py +++ b/core/dbt/parser/generic_test_builders.py @@ -229,92 +229,6 @@ def extract_test_args(data_test, name=None) -> Tuple[str, Dict[str, Any]]: test_args["column_name"] = name return test_name, test_args - @property - def enabled(self) -> Optional[bool]: - return self.config.get("enabled") - - @property - def alias(self) -> Optional[str]: - return self.config.get("alias") - - @property - def severity(self) -> Optional[str]: - sev = self.config.get("severity") - if sev: - return sev.upper() - else: - return None - - @property - def store_failures(self) -> Optional[bool]: - return self.config.get("store_failures") - - @property - def store_failures_as(self) -> Optional[bool]: - return self.config.get("store_failures_as") - - @property - def where(self) -> Optional[str]: - return self.config.get("where") - - @property - def limit(self) -> Optional[int]: - return self.config.get("limit") - - @property - def warn_if(self) -> Optional[str]: - return self.config.get("warn_if") - - @property - def error_if(self) -> Optional[str]: - return self.config.get("error_if") - - @property - def fail_calc(self) -> Optional[str]: - return self.config.get("fail_calc") - - @property - def meta(self) -> Optional[dict]: - return self.config.get("meta") - - @property - def database(self) -> Optional[str]: - return self.config.get("database") - - @property - def schema(self) -> Optional[str]: - return self.config.get("schema") - - def get_static_config(self): - config = {} - if self.alias is not None: - config["alias"] = self.alias - if self.severity is not None: - config["severity"] = self.severity - if self.enabled is not None: - config["enabled"] = self.enabled - if self.where is not None: - config["where"] = self.where - if self.limit is not None: - config["limit"] = self.limit - if self.warn_if is not None: - config["warn_if"] = self.warn_if - if self.error_if is not None: - config["error_if"] = self.error_if - if self.fail_calc is not None: - config["fail_calc"] = self.fail_calc - if self.store_failures is not None: - config["store_failures"] = self.store_failures - if self.store_failures_as is not None: - config["store_failures_as"] = self.store_failures_as - if self.meta is not None: - config["meta"] = self.meta - if self.database is not None: - config["database"] = self.database - if self.schema is not None: - config["schema"] = self.schema - return config - def tags(self) -> List[str]: tags = self.config.get("tags", []) if isinstance(tags, str): diff --git a/core/dbt/parser/schema_generic_tests.py b/core/dbt/parser/schema_generic_tests.py index 58be6dc94be..5eaf4aa948e 100644 --- a/core/dbt/parser/schema_generic_tests.py +++ b/core/dbt/parser/schema_generic_tests.py @@ -286,7 +286,7 @@ def render_test_update(self, node, config, builder, schema_file_id): # to the context in rendering processing node.depends_on.add_macro(macro_unique_id) if macro_unique_id in ["macro.dbt.test_not_null", "macro.dbt.test_unique"]: - config_call_dict = builder.get_static_config() + config_call_dict = builder.config config._config_call_dict = config_call_dict # This sets the config from dbt_project self.update_parsed_node_config(node, config) diff --git a/tests/functional/partial_parsing/test_pp_vars.py b/tests/functional/partial_parsing/test_pp_vars.py index a2e915dcb34..650b4df41da 100644 --- a/tests/functional/partial_parsing/test_pp_vars.py +++ b/tests/functional/partial_parsing/test_pp_vars.py @@ -104,7 +104,7 @@ def test_env_vars_models(self, project): schema_file = manifest.files[source.file_id] test_id = "test.test.source_not_null_seed_sources_raw_customers_id.e39ee7bf0d" test_node = manifest.nodes[test_id] - assert test_node.config.severity == "WARN" + assert test_node.config.severity == "warn" # Change severity env var os.environ["ENV_VAR_SEVERITY"] = "error" @@ -125,7 +125,7 @@ def test_env_vars_models(self, project): } assert expected_schema_file_env_vars == schema_file.env_vars test_node = manifest.nodes[test_id] - assert test_node.config.severity == "ERROR" + assert test_node.config.severity == "error" # Change database env var os.environ["ENV_VAR_DATABASE"] = "test_dbt" diff --git a/tests/functional/schema_tests/fixtures.py b/tests/functional/schema_tests/fixtures.py index ac266545712..51ae067bd84 100644 --- a/tests/functional/schema_tests/fixtures.py +++ b/tests/functional/schema_tests/fixtures.py @@ -1273,80 +1273,3 @@ data_tests: - my_custom_test """ - -custom_config_yml = """ -version: 2 -models: - - name: table - columns: - - name: color - data_tests: - - accepted_values: - values: ['blue', 'red'] - config: - custom_config_key: some_value - - custom_color_from_config: - severity: error - config: - test_color: orange - store_failures: true - unlogged: True -""" - -mixed_config_yml = """ -version: 2 -models: - - name: table - columns: - - name: color - data_tests: - - accepted_values: - values: ['blue', 'red'] - config: - custom_config_key: some_value - severity: warn - - custom_color_from_config: - severity: error - config: - test_color: blue -""" - -same_key_error_yml = """ -version: 2 -models: - - name: table - columns: - - name: color - data_tests: - - accepted_values: - values: ['blue', 'red'] - severity: warn - config: - severity: error -""" - -seed_csv = """ -id,color,value -1,blue,10 -2,red,20 -3,green,30 -4,yellow,40 -5,blue,50 -6,red,60 -7,blue,70 -8,green,80 -9,yellow,90 -10,blue,100 -""".strip() - -table_sql = """ --- content of the table.sql -select * from {{ ref('seed') }} -""" - -test_custom_color_from_config = """ -{% test custom_color_from_config(model, column_name) %} - select * from {{ model }} - where color = '{{ config.get('test_color') }}' -{% endtest %} -""" diff --git a/tests/functional/schema_tests/data_test_config.py b/tests/functional/schema_tests/test_custom_test_config.py similarity index 54% rename from tests/functional/schema_tests/data_test_config.py rename to tests/functional/schema_tests/test_custom_test_config.py index a6a16818000..bafa5427e42 100644 --- a/tests/functional/schema_tests/data_test_config.py +++ b/tests/functional/schema_tests/test_custom_test_config.py @@ -5,15 +5,84 @@ from dbt.contracts.graph.manifest import Manifest from dbt.contracts.graph.nodes import TestNode from dbt.exceptions import CompilationError -from dbt.tests.util import get_manifest, run_dbt -from tests.functional.schema_tests.fixtures import ( - custom_config_yml, - mixed_config_yml, - same_key_error_yml, - seed_csv, - table_sql, - test_custom_color_from_config, -) +from dbt.tests.util import get_manifest, run_dbt, update_config_file + +custom_config_yml = """ +models: + - name: table + columns: + - name: color + data_tests: + - accepted_values: + values: ['blue', 'red'] + config: + custom_config_key: some_value + - custom_color_from_config: + severity: error + config: + test_color: orange + store_failures: true + unlogged: True + - not_null: + config: + not_null_key: abc +""" + +mixed_config_yml = """ +models: + - name: table + columns: + - name: color + data_tests: + - accepted_values: + values: ['blue', 'red'] + config: + custom_config_key: some_value + severity: warn + - custom_color_from_config: + severity: error + config: + test_color: blue +""" + +same_key_error_yml = """ +models: + - name: table + columns: + - name: color + data_tests: + - accepted_values: + values: ['blue', 'red'] + severity: warn + config: + severity: error +""" + +seed_csv = """ +id,color,value +1,blue,10 +2,red,20 +3,green,30 +4,yellow,40 +5,blue,50 +6,red,60 +7,blue,70 +8,green,80 +9,yellow,90 +10,blue,100 +""".strip() + +table_sql = """ +-- content of the table.sql +select * from {{ ref('seed') }} +""" + +test_custom_color_from_config = """ +{% test custom_color_from_config(model, column_name) %} + select * from {{ model }} + where color = '{{ config.get('test_color') }}' +{% endtest %} +""" def _select_test_node(manifest: Manifest, pattern: re.Pattern[str]): @@ -50,12 +119,6 @@ def seeds(self): def macros(self): return {"custom_color_from_config.sql": test_custom_color_from_config} - @pytest.fixture(scope="class") - def project_config_update(self): - return { - "config-version": 2, - } - @pytest.fixture(scope="class", autouse=True) def setUp(self, project): run_dbt(["seed"]) @@ -68,7 +131,7 @@ def models(self): def test_custom_config(self, project): run_dbt(["run"]) - run_dbt(["test", "--log-level", "debug"], expect_pass=False) + run_dbt(["test"], expect_pass=False) manifest = get_manifest(project.project_root) # Pattern to match the test_id without the specific suffix @@ -79,12 +142,29 @@ def test_custom_config(self, project): assert "custom_config_key" in test_node.config assert test_node.config["custom_config_key"] == "some_value" - # pattern = re.compile(r"test\.test\.custom_color_from_config.*") - # test_node = _select_test_node(manifest, pattern) + custom_color_pattern = re.compile(r"test\.test\.custom_color_from_config.*") + custom_color_test_node = _select_test_node(manifest, custom_color_pattern) + assert custom_color_test_node.config.get("test_color") == "orange" + assert custom_color_test_node.config.get("unlogged") is True persistence = get_table_persistence(project, "custom_color_from_config_table_color") - assert persistence == "u" + not_null_pattern = re.compile(r"test\.test\.not_null.*") + not_null_test_node = _select_test_node(manifest, not_null_pattern) + assert not_null_test_node.config.get("not_null_key") == "abc" + + # set dbt_project.yml config and ensure that schema configs override project configs + config_patch = { + "data_tests": {"test_color": "blue", "some_key": "strange", "not_null_key": "def"} + } + update_config_file(config_patch, project.project_root, "dbt_project.yml") + manifest = run_dbt(["parse"]) + custom_color_test_node = _select_test_node(manifest, custom_color_pattern) + assert custom_color_test_node.config.get("test_color") == "orange" + assert custom_color_test_node.config.get("some_key") == "strange" + not_null_test_node = _select_test_node(manifest, not_null_pattern) + assert not_null_test_node.config.get("not_null_key") == "abc" + class TestMixedDataTestConfig(BaseDataTestsConfig): @pytest.fixture(scope="class") @@ -97,21 +177,8 @@ def test_mixed_config(self, project): # Pattern to match the test_id without the specific suffix pattern = re.compile(r"test\.test\.accepted_values_table_color__blue__red\.\d+") + test_node = _select_test_node(manifest, pattern) - # Find the test_id dynamically - test_id = None - for node_id in manifest.nodes: - if pattern.match(node_id): - test_id = node_id - break - - # Ensure the test_id was found - assert ( - test_id is not None - ), "Test ID matching the pattern was not found in the manifest nodes" - - # Proceed with the assertions - test_node = manifest.nodes[test_id] assert "custom_config_key" in test_node.config assert test_node.config["custom_config_key"] == "some_value" assert "severity" in test_node.config @@ -137,6 +204,3 @@ def test_same_key_error(self, project): # Assert that the error message contains the expected text assert "Test cannot have the same key at the top-level and in config" in exception_message - - # Assert that the error message contains the context of the error - assert "models/same_key_error.yml" in exception_message