diff --git a/doc/user/content/manage/dbt/_index.md b/doc/user/content/manage/dbt/_index.md index e9fc56e26da01..ca459e225f12b 100644 --- a/doc/user/content/manage/dbt/_index.md +++ b/doc/user/content/manage/dbt/_index.md @@ -571,22 +571,18 @@ and correct results** with millisecond latency whenever you query your views. [//]: # "TODO(morsapaes) Call out the cluster configuration for tests and store_failures_as once this page is rehashed." -[//]: # "TODO(morsapaes) Add instructions for unit testing after the upcoming -v1.8 release of dbt Core." - ### Configure continuous testing Using dbt in a streaming context means that you're able to run data quality and integrity [tests](https://docs.getdbt.com/docs/building-a-dbt-project/tests) -non-stop, and monitor failures as soon as they happen. This is useful for unit -testing during the development of your dbt models, and later in production to +non-stop. This is useful to monitor failures as soon as they happen, and trigger **real-time alerts** downstream. -1. To configure your project for continuous testing, add a `tests` property to +1. To configure your project for continuous testing, add a `data_tests` property to `dbt_project.yml` with the `store_failures` configuration: ```yaml - tests: + data_tests: dbt_project.name: models: +store_failures: true @@ -601,7 +597,7 @@ trigger **real-time alerts** downstream. **Note:** As an alternative, you can specify the `--store-failures` flag when running `dbt test`. -1. Add tests to your models using the `tests` property in the model +1. Add tests to your models using the `data_tests` property in the model configuration `.yml` files: ```yaml @@ -611,7 +607,7 @@ trigger **real-time alerts** downstream. columns: - name: col_a description: 'column a description' - tests: + data_tests: - not_null - unique ``` @@ -623,7 +619,7 @@ trigger **real-time alerts** downstream. 1. Run the tests: ```bash - dbt test + dbt test # use --select test_type:data to only run data tests! ``` When configured to `store_failures`, this command will create a materialized diff --git a/doc/user/content/manage/dbt/development-workflows.md b/doc/user/content/manage/dbt/development-workflows.md index 6303f3acf4b62..3ffb20e285765 100644 --- a/doc/user/content/manage/dbt/development-workflows.md +++ b/doc/user/content/manage/dbt/development-workflows.md @@ -19,7 +19,7 @@ using [dbt](/manage/dbt/) as your deployment tool. When you're prototyping your use case and fine-tuning the underlying data model, your priority is **iteration speed**. dbt has many features that can help speed -up development, like [node selection](#node-selection) and [model preview](#preview-model-results). +up development, like [node selection](#node-selection) and [model preview](#model-results-preview). Before you start, we recommend getting familiar with how these features work with the `dbt-materialize` adapter to make the most of your development time. @@ -97,7 +97,7 @@ dbt run --select "path/to/my_model.sql" # runs a specific model by its path For a full rundown of selection logic options, check the [dbt documentation](https://docs.getdbt.com/reference/node-selection/syntax). -### Preview model results +### Model results preview {{< note >}} The `dbt show` command uses a `LIMIT` clause under the hood, which has @@ -134,6 +134,145 @@ It's important to note that previewing results compiles the model and runs the compiled SQL against Materialize; it doesn't query the already-materialized database relation (see [`dbt-core` #7391](https://github.com/dbt-labs/dbt-core/issues/7391)). +### Unit tests + +**Minimum requirements:** `dbt-materialize` v1.8.0+ + +{{< note >}} +Complex types like [`map`](/sql/types/map/) and [`list`](/sql/types/list/) are +not supported in unit tests yet (see [`dbt-adapters` #113](https://github.com/dbt-labs/dbt-adapters/issues/113)). +For an overview of other known limitations, check the [dbt documentation](https://docs.getdbt.com/docs/build/unit-tests#before-you-begin). +{{}} + +To validate your SQL logic without fully materializing a model, as well as +future-proof it against edge cases, you can use [unit tests](https://docs.getdbt.com/docs/build/unit-tests). +Unit tests can be a **quicker way to iterate on model development** in +comparison to re-running the models, since you don't need to wait for a model +to hydrate before you can validate that it produces the expected results. + +1. As an example, imagine your dbt project includes the following models: + + **Filename:** _models/my_model_a.sql_ + ```sql + SELECT + 1 AS a, + 1 AS id, + 2 AS not_testing, + 'a' AS string_a, + DATE '2020-01-02' AS date_a + ``` + + **Filename:** _models/my_model_b.sql_ + ```sql + SELECT + 2 as b, + 1 as id, + 2 as c, + 'b' as string_b + ``` + + **Filename:** models/my_model.sql + ```sql + SELECT + a+b AS c, + CONCAT(string_a, string_b) AS string_c, + not_testing, + date_a + FROM {{ ref('my_model_a')}} my_model_a + JOIN {{ ref('my_model_b' )}} my_model_b + ON my_model_a.id = my_model_b.id + ``` + +1. To add a unit test to `my_model`, create a `.yml` file under the `/models` + directory, and use the [`unit_tests`](https://docs.getdbt.com/reference/resource-properties/unit-tests) + property: + + **Filename:** _models/unit_tests.yml_ + ```yaml + unit_tests: + - name: test_my_model + model: my_model + given: + - input: ref('my_model_a') + rows: + - {id: 1, a: 1} + - input: ref('my_model_b') + rows: + - {id: 1, b: 2} + - {id: 2, b: 2} + expect: + rows: + - {c: 2} + ``` + + For simplicity, this example provides mock data using inline dictionary + values, but other formats are supported. Check the [dbt documentation](https://docs.getdbt.com/reference/resource-properties/data-formats) + for a full rundown of the available options. + +1. Run the unit tests using `dbt test`: + + ```bash + dbt test --select test_type:unit + + 12:30:14 Running with dbt=1.8.0 + 12:30:14 Registered adapter: materialize=1.8.0 + 12:30:14 Found 6 models, 1 test, 4 seeds, 1 source, 471 macros, 1 unit test + 12:30:14 + 12:30:16 Concurrency: 1 threads (target='dev') + 12:30:16 + 12:30:16 1 of 1 START unit_test my_model::test_my_model ................................. [RUN] + 12:30:17 1 of 1 FAIL 1 my_model::test_my_model .......................................... [FAIL 1 in 1.51s] + 12:30:17 + 12:30:17 Finished running 1 unit test in 0 hours 0 minutes and 2.77 seconds (2.77s). + 12:30:17 + 12:30:17 Completed with 1 error and 0 warnings: + 12:30:17 + 12:30:17 Failure in unit_test test_my_model (models/models/unit_tests.yml) + 12:30:17 + + actual differs from expected: + + @@ ,c + +++,3 + ---,2 + ``` + + It's important to note that the **direct upstream dependencies** of the + model that you're unit testing **must exist** in Materialize before you can + execute the unit test via `dbt test`. To ensure these dependencies exist, + you can use the `--empty` flag to build an empty version of the models: + + ```bash + dbt run --select "my_model_a.sql" "my_model_b.sql" --empty + ``` + + Alternatively, you can execute unit tests as part of the `dbt build` + command, which will ensure the upstream depdendencies are created before + any unit tests are executed: + + ```bash + dbt build --select "+my_model.sql" + + 11:53:30 Running with dbt=1.8.0 + 11:53:30 Registered adapter: materialize=1.8.0 + ... + 11:53:33 2 of 12 START sql view model public.my_model_a ................................. [RUN] + 11:53:34 2 of 12 OK created sql view model public.my_model_a ............................ [CREATE VIEW in 0.49s] + 11:53:34 3 of 12 START sql view model public.my_model_b ................................. [RUN] + 11:53:34 3 of 12 OK created sql view model public.my_model_b ............................ [CREATE VIEW in 0.45s] + ... + 11:53:35 11 of 12 START unit_test my_model::test_my_model ............................... [RUN] + 11:53:36 11 of 12 FAIL 1 my_model::test_my_model ........................................ [FAIL 1 in 0.84s] + 11:53:36 Failure in unit_test test_my_model (models/models/unit_tests.yml) + 11:53:36 + + actual differs from expected: + + @@ ,c + +++,3 + ---,2 + ``` + ## Deployment Once your dbt project is ready to move out of development, or as soon as you diff --git a/misc/dbt-materialize/CHANGELOG.md b/misc/dbt-materialize/CHANGELOG.md index af6de1a1b0e2a..8be78624515c8 100644 --- a/misc/dbt-materialize/CHANGELOG.md +++ b/misc/dbt-materialize/CHANGELOG.md @@ -1,5 +1,16 @@ # dbt-materialize Changelog +## Unreleased + +* Update base adapter references as part of + [decoupling migration from dbt-core](https://github.com/dbt-labs/dbt-adapters/discussions/87) + * Migrate to dbt-common and dbt-adapters packages. + * Add tests for `--empty` flag as part of [dbt-labs/dbt-core#8971](https://github.com/dbt-labs/dbt-core/pull/8971) + * Add functional tests for unit testing. +* Support enforcing model contracts for the [`map`](https://materialize.com/docs/sql/types/map/), + [`list`](https://materialize.com/docs/sql/types/list/), + and [`record`](https://materialize.com/docs/sql/types/record/) pseudo-types. + ## 1.7.8 - 2024-05-06 * Fix permission management in blue/green automation macros for non-admin users @@ -7,7 +18,7 @@ ## 1.7.7 - 2024-04-19 -* Tweak [`deploy_permission_validation]`](https://github.com/MaterializeInc/materialize/blob/main/misc/dbt-materialize/dbt/include/materialize/macros/deploy/deploy_permission_validation.sql) +* Tweak [`deploy_permission_validation`](https://github.com/MaterializeInc/materialize/blob/main/misc/dbt-materialize/dbt/include/materialize/macros/deploy/deploy_permission_validation.sql) macro to work around [#26738](https://github.com/MaterializeInc/materialize/issues/26738). ## 1.7.6 - 2024-04-18 diff --git a/misc/dbt-materialize/Dockerfile b/misc/dbt-materialize/Dockerfile index 1d9f139c79b25..b0a6b230412f2 100644 --- a/misc/dbt-materialize/Dockerfile +++ b/misc/dbt-materialize/Dockerfile @@ -11,4 +11,5 @@ FROM python:3.8.6 COPY . dbt-materialize/ +RUN pip install pytest RUN pip install ./dbt-materialize[dev] diff --git a/misc/dbt-materialize/dbt/adapters/materialize/__version__.py b/misc/dbt-materialize/dbt/adapters/materialize/__version__.py index e6cc5a28a2c30..3332b02968283 100644 --- a/misc/dbt-materialize/dbt/adapters/materialize/__version__.py +++ b/misc/dbt-materialize/dbt/adapters/materialize/__version__.py @@ -15,4 +15,4 @@ # limitations under the License. # If you bump this version, bump it in setup.py too. -version = "1.7.8" +version = "1.8.0" diff --git a/misc/dbt-materialize/dbt/adapters/materialize/connections.py b/misc/dbt-materialize/dbt/adapters/materialize/connections.py index f89786069d92f..7e9a02890b51a 100644 --- a/misc/dbt-materialize/dbt/adapters/materialize/connections.py +++ b/misc/dbt-materialize/dbt/adapters/materialize/connections.py @@ -17,13 +17,12 @@ from dataclasses import dataclass from typing import Optional +import dbt_common.exceptions import psycopg2 +from dbt_common.semver import versions_compatible -import dbt.adapters.postgres.connections -import dbt.exceptions +from dbt.adapters.events.logging import AdapterLogger from dbt.adapters.postgres import PostgresConnectionManager, PostgresCredentials -from dbt.events import AdapterLogger -from dbt.semver import versions_compatible # If you bump this version, bump it in README.md too. SUPPORTED_MATERIALIZE_VERSIONS = ">=0.68.0" @@ -97,7 +96,7 @@ def open(cls, connection): mz_version = mz_version.split()[0] # e.g. v0.79.0-dev mz_version = mz_version[1:] # e.g. 0.79.0-dev if not versions_compatible(mz_version, SUPPORTED_MATERIALIZE_VERSIONS): - raise dbt.exceptions.DbtRuntimeError( + raise dbt_common.exceptions.DbtRuntimeError( f"Detected unsupported Materialize version {mz_version}\n" f" Supported versions: {SUPPORTED_MATERIALIZE_VERSIONS}" ) diff --git a/misc/dbt-materialize/dbt/adapters/materialize/exceptions.py b/misc/dbt-materialize/dbt/adapters/materialize/exceptions.py index d40491ef1ba57..a024e0a2ad1c9 100644 --- a/misc/dbt-materialize/dbt/adapters/materialize/exceptions.py +++ b/misc/dbt-materialize/dbt/adapters/materialize/exceptions.py @@ -15,7 +15,7 @@ from typing import Any -from dbt.exceptions import CompilationError +from dbt_common.exceptions import CompilationError class RefreshIntervalConfigNotDictError(CompilationError): diff --git a/misc/dbt-materialize/dbt/adapters/materialize/impl.py b/misc/dbt-materialize/dbt/adapters/materialize/impl.py index 7a54dc47d786d..53228150f0131 100644 --- a/misc/dbt-materialize/dbt/adapters/materialize/impl.py +++ b/misc/dbt-materialize/dbt/adapters/materialize/impl.py @@ -17,7 +17,13 @@ from dataclasses import dataclass from typing import Any, Dict, List, Optional -import dbt.exceptions +import dbt_common.exceptions +from dbt_common.contracts.constraints import ( + ColumnLevelConstraint, + ConstraintType, +) +from dbt_common.dataclass_schema import ValidationError, dbtClassMixin + from dbt.adapters.base.impl import AdapterConfig, ConstraintSupport from dbt.adapters.base.meta import available from dbt.adapters.capability import ( @@ -32,14 +38,9 @@ RefreshIntervalConfigNotDictError, ) from dbt.adapters.materialize.relation import MaterializeRelation -from dbt.adapters.postgres import PostgresAdapter from dbt.adapters.postgres.column import PostgresColumn +from dbt.adapters.postgres.impl import PostgresAdapter from dbt.adapters.sql.impl import LIST_RELATIONS_MACRO_NAME -from dbt.contracts.graph.nodes import ( - ColumnLevelConstraint, - ConstraintType, -) -from dbt.dataclass_schema import ValidationError, dbtClassMixin # types in ./misc/dbt-materialize need to import generic types from typing @@ -58,10 +59,12 @@ def parse(cls, raw_index) -> Optional["MaterializeIndexConfig"]: cls.validate(raw_index) return cls.from_dict(raw_index) except ValidationError as exc: - msg = dbt.exceptions.validator_error_message(exc) - dbt.exceptions.CompilationError(f"Could not parse index config: {msg}") + msg = dbt_common.exceptions.validator_error_message(exc) + dbt_common.exceptions.CompilationError( + f"Could not parse index config: {msg}" + ) except TypeError: - dbt.exceptions.CompilationError( + dbt_common.exceptions.CompilationError( "Invalid index config:\n" f" Got: {raw_index}\n" ' Expected a dictionary with at minimum a "columns" key' diff --git a/misc/dbt-materialize/dbt/adapters/materialize/relation.py b/misc/dbt-materialize/dbt/adapters/materialize/relation.py index 9db6e59768c9c..6dd57e3bbb9ba 100644 --- a/misc/dbt-materialize/dbt/adapters/materialize/relation.py +++ b/misc/dbt-materialize/dbt/adapters/materialize/relation.py @@ -17,9 +17,10 @@ from dataclasses import dataclass from typing import Optional, Type -from dbt.adapters.postgres import PostgresRelation -from dbt.dataclass_schema import StrEnum -from dbt.utils import classproperty +from dbt_common.dataclass_schema import StrEnum + +from dbt.adapters.postgres.relation import PostgresRelation +from dbt.adapters.utils import classproperty # types in ./misc/dbt-materialize need to import generic types from typing diff --git a/misc/dbt-materialize/dbt/include/materialize/macros/materializations/unit.sql b/misc/dbt-materialize/dbt/include/materialize/macros/materializations/unit.sql new file mode 100644 index 0000000000000..e1084f565adb8 --- /dev/null +++ b/misc/dbt-materialize/dbt/include/materialize/macros/materializations/unit.sql @@ -0,0 +1,51 @@ +-- Copyright Materialize, Inc. and contributors. All rights reserved. +-- +-- Licensed under the Apache License, Version 2.0 (the "License"); +-- you may not use this file except in compliance with the License. +-- You may obtain a copy of the License in the LICENSE file at the +-- root of this repository, or online at +-- +-- http://www.apache.org/licenses/LICENSE-2.0 +-- +-- Unless required by applicable law or agreed to in writing, software +-- distributed under the License is distributed on an "AS IS" BASIS, +-- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +-- See the License for the specific language governing permissions and +-- limitations under the License. + +{%- materialization unit, adapter='materialize' -%} + + {% set relations = [] %} + + {% set expected_rows = config.get('expected_rows') %} + {% set expected_sql = config.get('expected_sql') %} + {% set tested_expected_column_names = expected_rows[0].keys() if (expected_rows | length) > 0 else get_columns_in_query(sql) %} + + {%- set target_relation = this.incorporate(type='view') -%} + {%- set temp_relation = make_temp_relation(target_relation) -%} + + {%- call statement(auto_begin=True) -%} + {{ materialize__create_view_as(temp_relation, get_empty_subquery_sql(sql)) }} + {%- endcall -%} + + {%- set columns_in_relation = adapter.get_columns_in_relation(temp_relation) -%} + {%- set column_name_to_data_types = {} -%} + {%- for column in columns_in_relation -%} + {%- do column_name_to_data_types.update({column.name|lower: column.data_type}) -%} + {%- endfor -%} + + {% if not expected_sql %} + {% set expected_sql = get_expected_sql(expected_rows, column_name_to_data_types) %} + {% endif %} + + {% set unit_test_sql = get_unit_test_sql(sql, expected_sql, tested_expected_column_names) %} + + {% call statement('main', fetch_result=True) -%} + {{ unit_test_sql }} + {%- endcall %} + + {% do adapter.drop_relation(temp_relation) %} + + {{ return({'relations': relations}) }} + +{%- endmaterialization -%} diff --git a/misc/dbt-materialize/dbt/include/materialize/macros/tests/helpers.sql b/misc/dbt-materialize/dbt/include/materialize/macros/tests/helpers.sql index 794fd837e0024..37780aef5e70b 100644 --- a/misc/dbt-materialize/dbt/include/materialize/macros/tests/helpers.sql +++ b/misc/dbt-materialize/dbt/include/materialize/macros/tests/helpers.sql @@ -22,3 +22,13 @@ {{ adapter.dispatch('get_test_sql', 'dbt')(main_sql, fail_calc, warn_if, error_if, limit) }} {%- endmacro %} + +{% macro get_unit_test_sql(main_sql, expected_fixture_sql, expected_column_names, cluster) -%} + {% if cluster %} + {% call statement(auto_begin=True) %} + set cluster = {{ cluster }} + {% endcall %} + {% endif %} + + {{ adapter.dispatch('get_unit_test_sql', 'dbt')(main_sql, expected_fixture_sql, expected_column_names) }} +{%- endmacro %} diff --git a/misc/dbt-materialize/dbt/include/materialize/macros/utils/cast.sql b/misc/dbt-materialize/dbt/include/materialize/macros/utils/cast.sql new file mode 100644 index 0000000000000..ba0093b8304f1 --- /dev/null +++ b/misc/dbt-materialize/dbt/include/materialize/macros/utils/cast.sql @@ -0,0 +1,27 @@ +-- Copyright Materialize, Inc. and contributors. All rights reserved. +-- +-- Licensed under the Apache License, Version 2.0 (the "License"); +-- you may not use this file except in compliance with the License. +-- You may obtain a copy of the License in the LICENSE file at the +-- root of this repository, or online at +-- +-- http://www.apache.org/licenses/LICENSE-2.0 +-- +-- Unless required by applicable law or agreed to in writing, software +-- distributed under the License is distributed on an "AS IS" BASIS, +-- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +-- See the License for the specific language governing permissions and +-- limitations under the License. + +{% macro materialize__cast(expression, data_type) -%} + {#-- Handle types that don't support cast(NULL as type) --#} + {%- if expression.strip().lower() == "null" and data_type.strip().lower() == "map" -%} + NULL::map[text => text] + {%- elif expression.strip().lower() == "null" and data_type.strip().lower() == "list" -%} + NULL::text list + {%- elif expression.strip().lower() == "null" and data_type.strip().lower() == "record" -%} + (SELECT row() WHERE false) + {%- else -%} + cast({{ expression }} as {{ data_type }}) + {%- endif -%} +{%- endmacro %} diff --git a/misc/dbt-materialize/dbt/include/starter_project/dbt_project.yml b/misc/dbt-materialize/dbt/include/starter_project/dbt_project.yml index 6fb8ee8fd4e1f..36a6c2a3b0000 100644 --- a/misc/dbt-materialize/dbt/include/starter_project/dbt_project.yml +++ b/misc/dbt-materialize/dbt/include/starter_project/dbt_project.yml @@ -25,7 +25,7 @@ clean-targets: - "target" - "dbt_packages" -tests: +data_tests: {project_name}: +store_failures: true +schema: 'etl_failure' diff --git a/misc/dbt-materialize/dbt/include/starter_project/models/example/schema.yml b/misc/dbt-materialize/dbt/include/starter_project/models/example/schema.yml index ee42e4568b8e4..6f287367095b8 100644 --- a/misc/dbt-materialize/dbt/include/starter_project/models/example/schema.yml +++ b/misc/dbt-materialize/dbt/include/starter_project/models/example/schema.yml @@ -23,7 +23,7 @@ models: columns: - name: seller description: "The seller for an auction" - tests: + data_tests: - unique - not_null - name: seller_item @@ -40,7 +40,7 @@ models: columns: - name: id description: "The id of the buyer or seller" - tests: + data_tests: - not_null - name: credits description: "Credit from an auction" @@ -52,7 +52,7 @@ models: columns: - name: id description: "The primary key of the auction" - tests: + data_tests: - unique - not_null - name: buyer @@ -73,7 +73,7 @@ models: columns: - name: id description: "The primary key for this table" - tests: + data_tests: - unique - not_null @@ -82,6 +82,6 @@ models: columns: - name: id description: "The primary key for this table" - tests: + data_tests: - unique - not_null diff --git a/misc/dbt-materialize/setup.py b/misc/dbt-materialize/setup.py index 06f382865cf30..9097c08ddcfd6 100644 --- a/misc/dbt-materialize/setup.py +++ b/misc/dbt-materialize/setup.py @@ -26,7 +26,7 @@ # This adapter's minor version should match the required dbt-postgres version, # but patch versions may differ. # If you bump this version, bump it in __version__.py too. - version="1.7.8", + version="1.8.0", description="The Materialize adapter plugin for dbt.", long_description=(Path(__file__).parent / "README.md").open().read(), long_description_content_type="text/markdown", @@ -41,8 +41,16 @@ "include/materialize/macros/**/*.sql", ] }, - install_requires=["dbt-postgres~=1.7.0"], + install_requires=[ + "dbt-common>=0.1.0a1,<2.0", + "dbt-adapters>=0.1.0a1,<2.0", + # add dbt-core to ensure backwards compatibility of installation, this is not a functional dependency + "dbt-core>=1.8.0", + "dbt-postgres~=1.8.0", + ], extras_require={ - "dev": ["dbt-tests-adapter~=1.7.0"], + "dev": [ + "dbt-tests-adapter @ git+https://github.com/dbt-labs/dbt-adapters.git#egg=dbt-tests-adapter&subdirectory=dbt-tests-adapter" + ], }, ) diff --git a/misc/dbt-materialize/tests/adapter/fixtures.py b/misc/dbt-materialize/tests/adapter/fixtures.py index 55c31afa99300..dc2fa6de11dd7 100644 --- a/misc/dbt-materialize/tests/adapter/fixtures.py +++ b/misc/dbt-materialize/tests/adapter/fixtures.py @@ -299,3 +299,25 @@ - name: c data_type: string """ + +contract_pseudo_types_yml = """ +version: 2 +models: + - name: test_pseudo_types + config: + contract: + enforced: true + columns: + - name: a + data_type: map + - name: b + data_type: record + - name: c + data_type: list +""" + +test_pseudo_types = """ +{{ config(materialized='view') }} + + SELECT '{a=>1, b=>2}'::map[text=>int] AS a, ROW(1, 2) AS b, LIST[[1,2],[3]] AS c +""" diff --git a/misc/dbt-materialize/tests/adapter/test_constraints.py b/misc/dbt-materialize/tests/adapter/test_constraints.py index 4807555a836b4..b721e4a9e1b15 100644 --- a/misc/dbt-materialize/tests/adapter/test_constraints.py +++ b/misc/dbt-materialize/tests/adapter/test_constraints.py @@ -32,8 +32,10 @@ from dbt.tests.util import run_dbt, run_sql_with_adapter from fixtures import ( contract_invalid_cluster_schema_yml, + contract_pseudo_types_yml, nullability_assertions_schema_yml, test_materialized_view, + test_pseudo_types, test_view, ) @@ -162,3 +164,18 @@ def test_materialize_drop_quickstart(self, project): run_dbt(["run", "--models", "contract_invalid_cluster"], expect_pass=True) project.run_sql("CREATE CLUSTER quickstart SIZE = '1'") + + +class TestContractPseudoTypes: + @pytest.fixture(scope="class") + def models(self): + return { + "contract_pseudo_types.yml": contract_pseudo_types_yml, + "contract_pseudo_types.sql": test_pseudo_types, + } + + # Pseudo-types in Materialize cannot be cast using the cast() function, so we + # special-handle their NULL casting for contract validation. + # See #17870: https://github.com/MaterializeInc/materialize/issues/17870 + def test_pseudo_types(self, project): + run_dbt(["run", "--models", "contract_pseudo_types"], expect_pass=True) diff --git a/misc/dbt-materialize/tests/adapter/test_dbt_clone.py b/misc/dbt-materialize/tests/adapter/test_dbt_clone.py index f74f6c97b4a33..a65892a03d539 100644 --- a/misc/dbt-materialize/tests/adapter/test_dbt_clone.py +++ b/misc/dbt-materialize/tests/adapter/test_dbt_clone.py @@ -32,10 +32,8 @@ def snapshots(self): def run_and_save_state(self, project_root, with_snapshot=False): results = run_dbt(["seed"]) assert len(results) == 1 - assert not any(r.node.deferred for r in results) results = run_dbt(["run"]) assert len(results) == 2 - assert not any(r.node.deferred for r in results) results = run_dbt(["test"]) assert len(results) == 2 @@ -59,6 +57,7 @@ def test_can_clone_false(self, project, unique_schema, other_schema): clone_args = [ "clone", + "--defer", "--state", "state", "--target", diff --git a/misc/dbt-materialize/tests/adapter/test_empty.py b/misc/dbt-materialize/tests/adapter/test_empty.py new file mode 100644 index 0000000000000..f2a80f0401ca5 --- /dev/null +++ b/misc/dbt-materialize/tests/adapter/test_empty.py @@ -0,0 +1,20 @@ +# Copyright Materialize, Inc. and contributors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License in the LICENSE file at the +# root of this repository, or online at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +from dbt.tests.adapter.empty.test_empty import BaseTestEmpty + + +class TestMaterializeEmpty(BaseTestEmpty): + pass diff --git a/misc/dbt-materialize/tests/adapter/test_unit_testing.py b/misc/dbt-materialize/tests/adapter/test_unit_testing.py new file mode 100644 index 0000000000000..d6927fe497ce1 --- /dev/null +++ b/misc/dbt-materialize/tests/adapter/test_unit_testing.py @@ -0,0 +1,59 @@ +# Copyright Materialize, Inc. and contributors. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License in the LICENSE file at the +# root of this repository, or online at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +import pytest +from dbt.tests.adapter.unit_testing.test_case_insensitivity import ( + BaseUnitTestCaseInsensivity, +) +from dbt.tests.adapter.unit_testing.test_invalid_input import BaseUnitTestInvalidInput +from dbt.tests.adapter.unit_testing.test_types import BaseUnitTestingTypes + + +class TestMaterializeUnitTestingTypes(BaseUnitTestingTypes): + @pytest.fixture + def data_types(self): + # sql_value, yaml_value + return [ + ["1", "1"], + ["'1'", "1"], + ["true", "true"], + ["DATE '2020-01-02'", "2020-01-02"], + ["TIMESTAMP '2013-11-03 00:00:00-0'", "2013-11-03 00:00:00-0"], + ["TIMESTAMPTZ '2013-11-03 00:00:00-0'", "2013-11-03 00:00:00-0"], + ["'1'::numeric", "1"], + [ + """'{"bar": "baz", "balance": 7.77, "active": false}'::json""", + """'{"bar": "baz", "balance": 7.77, "active": false}'""", + ], + ["MZ_TIMESTAMP '2023-05-21 12:34:56'", "2023-05-21 12:34:56"], + # TODO: array types + # ["LIST[1, 2, 3]", """'[1, 2, 3]'"""], + # ["LIST['a', 'b', 'c']", """'["a", "b", "c"]'"""], + # ["ARRAY[1,2,3]", """'{1, 2, 3}'"""], + # ["ARRAY[1.0,2.0,3.0]", """'{1.0, 2.0, 3.0}'"""], + # ["ARRAY[1::numeric,2::numeric,3::numeric]", """'{1.0, 2.0, 3.0}'"""], + # ["ARRAY['a','b','c']", """'{"a", "b", "c"}'"""], + # ["ARRAY[true,true,false]", """'{true, true, false}'"""], + # ["ARRAY[DATE '2020-01-02']", """'{"2020-01-02"}'"""], + # ["ARRAY[TIMESTAMP '2013-11-03 00:00:00-0']", """'{"2013-11-03 00:00:00-0"}'"""], + # ["ARRAY[TIMESTAMPTZ '2013-11-03 00:00:00-0']", """'{"2013-11-03 00:00:00-0"}'"""], + ] + + +class TestMaterializeUnitTestCaseInsensitivity(BaseUnitTestCaseInsensivity): + pass + + +class TestMaterializeUnitTestInvalidInput(BaseUnitTestInvalidInput): + pass diff --git a/misc/dbt-materialize/tests/adapter/test_utils.py b/misc/dbt-materialize/tests/adapter/test_utils.py index 22cd258df89d2..4417d4655bd97 100644 --- a/misc/dbt-materialize/tests/adapter/test_utils.py +++ b/misc/dbt-materialize/tests/adapter/test_utils.py @@ -23,6 +23,7 @@ from dbt.tests.adapter.utils.fixture_get_intervals_between import ( models__test_get_intervals_between_yml, ) +from dbt.tests.adapter.utils.test_cast import BaseCast from dbt.tests.adapter.utils.test_cast_bool_to_text import BaseCastBoolToText from dbt.tests.adapter.utils.test_current_timestamp import BaseCurrentTimestampAware from dbt.tests.adapter.utils.test_date_spine import BaseDateSpine @@ -96,6 +97,11 @@ 11 as expected """ + +class TestCast(BaseCast): + pass + + # The `cast_bool_to_text` macro works as expected, but we must alter the test case # because set operation type conversions do not work properly. # See https://github.com/MaterializeInc/materialize/issues/3331