diff --git a/.github/workflows/triage-labels.yml b/.github/workflows/triage-labels.yml new file mode 100644 index 00000000..c693eb48 --- /dev/null +++ b/.github/workflows/triage-labels.yml @@ -0,0 +1,31 @@ +# **what?** +# When the maintenance team triages, we sometimes need more information from the issue creator. In +# those cases we remove the `triage` label and add the `awaiting_response` label. Once we +# recieve a response in the form of a comment, we want the `awaiting_response` label removed +# in favor of the `triage` label so we are aware that the issue needs action. + +# **why?** +# To help with out team triage issue tracking + +# **when?** +# This will run when a comment is added to an issue and that issue has the `awaiting_response` label. + +name: Update Triage Label + +on: issue_comment + +defaults: + run: + shell: bash + +permissions: + issues: write + +jobs: + triage_label: + if: contains(github.event.issue.labels.*.name, 'awaiting_response') + uses: dbt-labs/actions/.github/workflows/swap-labels.yml@main + with: + add_label: "triage" + remove_label: "awaiting_response" + secrets: inherit diff --git a/tests/functional/test_experimental_parser.py b/tests/functional/test_experimental_parser.py deleted file mode 100644 index 9064ee4f..00000000 --- a/tests/functional/test_experimental_parser.py +++ /dev/null @@ -1,304 +0,0 @@ -import os - -from dbt.context.providers import RefArgs -from dbt.contracts.graph.manifest import Manifest -import pytest - -from tests.functional.utils import run_dbt, run_dbt_and_capture - - -def get_manifest(): - path = "./target/partial_parse.msgpack" - if os.path.exists(path): - with open(path, "rb") as fp: - manifest_mp = fp.read() - manifest: Manifest = Manifest.from_msgpack(manifest_mp) - return manifest - else: - return None - - -basic__schema_yml = """ -version: 2 - -sources: - - name: my_src - schema: "{{ target.schema }}" - tables: - - name: my_tbl - -models: - - name: model_a - columns: - - name: fun - -""" - -basic__model_a_sql = """ -{{ config(tags='hello', x=False) }} -{{ config(tags='world', x=True) }} - -select * from {{ ref('model_b') }} -cross join {{ source('my_src', 'my_tbl') }} -where false as boop - -""" - -basic__model_b_sql = """ -select 1 as fun -""" - - -ref_macro__schema_yml = """ -version: 2 - -""" - -ref_macro__models__model_a_sql = """ -select 1 as id - -""" - -source_macro__macros__source_sql = """ -{% macro source(source_name, table_name) %} - -{% endmacro %} -""" - -source_macro__schema_yml = """ -version: 2 - -""" - -source_macro__models__model_a_sql = """ -select 1 as id - -""" - -config_macro__macros__config_sql = """ -{% macro config() %} - -{% endmacro %} -""" - -config_macro__schema_yml = """ -version: 2 - -""" - -config_macro__models__model_a_sql = """ -select 1 as id - -""" - - -class BasicExperimentalParser: - @pytest.fixture(scope="class") - def models(self): - return { - "model_a.sql": basic__model_a_sql, - "model_b.sql": basic__model_b_sql, - "schema.yml": basic__schema_yml, - } - - -class TestBasicExperimentalParserFlag(BasicExperimentalParser): - @pytest.fixture(scope="class", autouse=True) - def setup(self, project): - os.environ["DBT_USE_EXPERIMENTAL_PARSER"] = "true" - yield - del os.environ["DBT_USE_EXPERIMENTAL_PARSER"] - - def test_env_use_experimental_parser(self, project): - _, log_output = run_dbt_and_capture(["--debug", "parse"]) - - # successful stable static parsing - assert not ("1699: " in log_output) - # successful experimental static parsing - assert "1698: " in log_output - # experimental parser failed - assert not ("1604: " in log_output) - # static parser failed - assert not ("1603: " in log_output) - # jinja rendering - assert not ("1602: " in log_output) - - -class TestBasicStaticParserFlag(BasicExperimentalParser): - @pytest.fixture(scope="class", autouse=True) - def setup(self, project): - os.environ["DBT_STATIC_PARSER"] = "false" - yield - del os.environ["DBT_STATIC_PARSER"] - - def test_env_static_parser(self, project): - _, log_output = run_dbt_and_capture(["--debug", "parse"]) - - print(log_output) - - # jinja rendering because of --no-static-parser - assert "1605: " in log_output - # successful stable static parsing - assert not ("1699: " in log_output) - # successful experimental static parsing - assert not ("1698: " in log_output) - # experimental parser failed - assert not ("1604: " in log_output) - # static parser failed - assert not ("1603: " in log_output) - # fallback jinja rendering - assert not ("1602: " in log_output) - - -class TestBasicExperimentalParser(BasicExperimentalParser): - # test that the experimental parser extracts some basic ref, source, and config calls. - def test_experimental_parser_basic( - self, - project, - ): - run_dbt(["--use-experimental-parser", "parse"]) - manifest = get_manifest() - node = manifest.nodes["model.test.model_a"] - assert node.refs == [RefArgs(name="model_b")] - assert node.sources == [["my_src", "my_tbl"]] - assert node.config._extra == {"x": True} - assert node.config.tags == ["hello", "world"] - - -class TestBasicStaticParser(BasicExperimentalParser): - # test that the static parser extracts some basic ref, source, and config calls by default - # without the experimental flag and without rendering jinja - def test_static_parser_basic(self, project): - _, log_output = run_dbt_and_capture(["--debug", "parse"]) - - # successful stable static parsing - assert "1699: " in log_output - # successful experimental static parsing - assert not ("1698: " in log_output) - # experimental parser failed - assert not ("1604: " in log_output) - # static parser failed - assert not ("1603: " in log_output) - # jinja rendering - assert not ("1602: " in log_output) - - manifest = get_manifest() - node = manifest.nodes["model.test.model_a"] - assert node.refs == [RefArgs(name="model_b")] - assert node.sources == [["my_src", "my_tbl"]] - assert node.config._extra == {"x": True} - assert node.config.tags == ["hello", "world"] - - -class TestBasicNoStaticParser(BasicExperimentalParser): - # test that the static parser doesn't run when the flag is set - def test_static_parser_is_disabled(self, project): - _, log_output = run_dbt_and_capture(["--debug", "--no-static-parser", "parse"]) - - # jinja rendering because of --no-static-parser - assert "1605: " in log_output - # successful stable static parsing - assert not ("1699: " in log_output) - # successful experimental static parsing - assert not ("1698: " in log_output) - # experimental parser failed - assert not ("1604: " in log_output) - # static parser failed - assert not ("1603: " in log_output) - # fallback jinja rendering - assert not ("1602: " in log_output) - - -class TestRefOverrideExperimentalParser: - @pytest.fixture(scope="class") - def models(self): - return { - "model_a.sql": ref_macro__models__model_a_sql, - "schema.yml": ref_macro__schema_yml, - } - - @pytest.fixture(scope="class") - def macros(self): - return { - "source.sql": source_macro__macros__source_sql, - } - - # test that the experimental parser doesn't run if the ref built-in is overriden with a macro - def test_experimental_parser_ref_override( - self, - project, - ): - _, log_output = run_dbt_and_capture(["--debug", "--use-experimental-parser", "parse"]) - - print(log_output) - - # successful experimental static parsing - assert not ("1698: " in log_output) - # fallback to jinja rendering - assert "1602: " in log_output - # experimental parser failed - assert not ("1604: " in log_output) - # didn't run static parser because dbt detected a built-in macro override - assert "1601: " in log_output - - -class TestSourceOverrideExperimentalParser: - @pytest.fixture(scope="class") - def models(self): - return { - "model_a.sql": source_macro__models__model_a_sql, - "schema.yml": source_macro__schema_yml, - } - - @pytest.fixture(scope="class") - def macros(self): - return { - "source.sql": source_macro__macros__source_sql, - } - - # test that the experimental parser doesn't run if the source built-in is overriden with a macro - def test_experimental_parser_source_override( - self, - project, - ): - _, log_output = run_dbt_and_capture(["--debug", "--use-experimental-parser", "parse"]) - - # successful experimental static parsing - assert not ("1698: " in log_output) - # fallback to jinja rendering - assert "1602: " in log_output - # experimental parser failed - assert not ("1604: " in log_output) - # didn't run static parser because dbt detected a built-in macro override - assert "1601: " in log_output - - -class TestConfigOverrideExperimentalParser: - @pytest.fixture(scope="class") - def models(self): - return { - "model_a.sql": config_macro__models__model_a_sql, - "schema.yml": config_macro__schema_yml, - } - - @pytest.fixture(scope="class") - def macros(self): - return { - "config.sql": config_macro__macros__config_sql, - } - - # test that the experimental parser doesn't run if the config built-in is overriden with a macro - def test_experimental_parser_config_override( - self, - project, - ): - _, log_output = run_dbt_and_capture(["--debug", "--use-experimental-parser", "parse"]) - - # successful experimental static parsing - assert not ("1698: " in log_output) - # fallback to jinja rendering - assert "1602: " in log_output - # experimental parser failed - assert not ("1604: " in log_output) - # didn't run static parser because dbt detected a built-in macro override - assert "1601: " in log_output diff --git a/tests/unit/test_adapter.py b/tests/unit/test_adapter.py index be2eef14..d73ed54c 100644 --- a/tests/unit/test_adapter.py +++ b/tests/unit/test_adapter.py @@ -1,10 +1,6 @@ -import dataclasses from multiprocessing import get_context from unittest import TestCase, mock -import agate -from dbt.adapters.base import BaseRelation -from dbt.adapters.contracts.relation import Path from dbt_common.exceptions import DbtValidationError from dbt.adapters.postgres import Plugin as PostgresPlugin, PostgresAdapter @@ -306,45 +302,3 @@ def test_set_zero_keepalive(self, psycopg2): connect_timeout=10, application_name="dbt", ) - - @mock.patch.object(PostgresAdapter, "execute_macro") - @mock.patch.object(PostgresAdapter, "_get_catalog_relations") - def test_get_catalog_various_schemas(self, mock_get_relations, mock_execute): - self.catalog_test(mock_get_relations, mock_execute, False) - - @mock.patch.object(PostgresAdapter, "execute_macro") - @mock.patch.object(PostgresAdapter, "_get_catalog_relations") - def test_get_filtered_catalog(self, mock_get_relations, mock_execute): - self.catalog_test(mock_get_relations, mock_execute, True) - - def catalog_test(self, mock_get_relations, mock_execute, filtered=False): - column_names = ["table_database", "table_schema", "table_name"] - relations = [ - BaseRelation(path=Path(database="dbt", schema="foo", identifier="bar")), - BaseRelation(path=Path(database="dbt", schema="FOO", identifier="baz")), - BaseRelation(path=Path(database="dbt", schema=None, identifier="bar")), - BaseRelation(path=Path(database="dbt", schema="quux", identifier="bar")), - BaseRelation(path=Path(database="dbt", schema="skip", identifier="bar")), - ] - rows = list(map(lambda x: dataclasses.astuple(x.path), relations)) - mock_execute.return_value = agate.Table(rows=rows, column_names=column_names) - - mock_get_relations.return_value = relations - - relation_configs = [] - used_schemas = {("dbt", "foo"), ("dbt", "quux")} - - if filtered: - catalog, exceptions = self.adapter.get_filtered_catalog( - relation_configs, used_schemas, set([relations[0], relations[3]]) - ) - else: - catalog, exceptions = self.adapter.get_catalog(relation_configs, used_schemas) - - tupled_catalog = set(map(tuple, catalog)) - if filtered: - self.assertEqual(tupled_catalog, {rows[0], rows[3]}) - else: - self.assertEqual(tupled_catalog, {rows[0], rows[1], rows[3]}) - - self.assertEqual(exceptions, [])