From 48dce2b64aa4abc21030e8d2e77cb0c7bd394c5f Mon Sep 17 00:00:00 2001 From: Joppe Vos <44348300+joppevos@users.noreply.github.com> Date: Wed, 13 Dec 2023 18:54:31 +0100 Subject: [PATCH] Extend DbtDocsLocalOperator with static flag (#759) ## Description Extends the docsOperator to make use of dbt `--static` flag. The static flag is available from dbt 1.7> I decided to add the flag through `dbt_cmd_flags`. Other options could be a parameter or the `operator_args`. There is no official documentation page from DBT on the flag [yet](https://github.com/dbt-labs/docs.getdbt.com/issues/4599). When they do add it, I can trim down our documentation and link directly to dbt. I contribute to learn and am appreciative of any feedback. ## Related Issue(s) closes #746 ## Breaking Change? ## Checklist - [x] I have made corresponding changes to the documentation (if required) - [x] I have added tests that prove my fix is effective or that my feature works --------- Co-authored-by: Tatiana Al-Chueyr Co-authored-by: Justin Bandoro <79104794+jbandoro@users.noreply.github.com> --- cosmos/operators/local.py | 11 +++++++++- docs/configuration/generating-docs.rst | 28 ++++++++++++++++++++++++++ tests/operators/test_local.py | 11 ++++++++++ 3 files changed, 49 insertions(+), 1 deletion(-) diff --git a/cosmos/operators/local.py b/cosmos/operators/local.py index 3b1751cd9..ea7cb41ae 100644 --- a/cosmos/operators/local.py +++ b/cosmos/operators/local.py @@ -548,6 +548,15 @@ def __init__(self, **kwargs: Any) -> None: super().__init__(**kwargs) self.base_cmd = ["docs", "generate"] + self.check_static_flag() + + def check_static_flag(self) -> None: + flag = "--static" + if self.dbt_cmd_flags: + if flag in self.dbt_cmd_flags: + # For the --static flag we only upload the generated static_index.html file + self.required_files = ["static_index.html"] + class DbtDocsCloudLocalOperator(DbtDocsLocalOperator, ABC): """ @@ -578,7 +587,7 @@ def upload_to_cloud_storage(self, project_dir: str) -> None: class DbtDocsS3LocalOperator(DbtDocsCloudLocalOperator): """ - Executes `dbt docs generate` command and upload to S3 storage. Returns the S3 path to the generated documentation. + Executes `dbt docs generate` command and upload to S3 storage. :param connection_id: S3's Airflow connection ID :param bucket_name: S3's bucket name diff --git a/docs/configuration/generating-docs.rst b/docs/configuration/generating-docs.rst index 88459fd14..6112ebcee 100644 --- a/docs/configuration/generating-docs.rst +++ b/docs/configuration/generating-docs.rst @@ -83,6 +83,34 @@ You can use the :class:`~cosmos.operators.DbtDocsGCSOperator` to generate and up bucket_name="test_bucket", ) +Static Flag +~~~~~~~~~~~~~~~~~~~~~~~ + +All of the DbtDocsOperator accept the ``--static`` flag. To learn more about the static flag, check out the `original PR on dbt-core `_. +The static flag is used to generate a single doc file that can be hosted directly from cloud storage. +By having a single documentation file, you can make use of Access control can be configured through Identity-Aware Proxy (IAP), and making it easy to host. + +.. note:: + The static flag is only available from dbt-core >=1.7 + +The following code snippet shows how to provide this flag with the default jaffle_shop project: + + +.. code-block:: python + + from cosmos.operators import DbtDocsGCSOperator + + # then, in your DAG code: + generate_dbt_docs_aws = DbtDocsGCSOperator( + task_id="generate_dbt_docs_gcs", + project_dir="path/to/jaffle_shop", + profile_config=profile_config, + # docs-specific arguments + connection_id="test_gcs", + bucket_name="test_bucket", + dbt_cmd_flags=["--static"], + ) + Custom Callback ~~~~~~~~~~~~~~~~~~~~~~~ diff --git a/tests/operators/test_local.py b/tests/operators/test_local.py index b0a36b335..dd7d34a6d 100644 --- a/tests/operators/test_local.py +++ b/tests/operators/test_local.py @@ -488,3 +488,14 @@ def test_operator_execute_deps_parameters( mock_ensure_profile.return_value.__enter__.return_value = (Path("/path/to/profile"), {"ENV_VAR": "value"}) task.execute(context={"task_instance": MagicMock()}) assert mock_build_and_run_cmd.call_args_list[0].kwargs["command"] == expected_call_kwargs + + +def test_dbt_docs_local_operator_with_static_flag(): + # Check when static flag is passed, the required files are correctly adjusted to a single file + operator = DbtDocsLocalOperator( + task_id="fake-task", + project_dir="fake-dir", + profile_config=profile_config, + dbt_cmd_flags=["--static"], + ) + assert operator.required_files == ["static_index.html"]