From b7dafbd0750d506a7884c7b82fc2daa06ad58871 Mon Sep 17 00:00:00 2001 From: Maxime Armstrong <46797220+maximearmstrong@users.noreply.github.com> Date: Tue, 5 Nov 2024 18:44:00 -0500 Subject: [PATCH] [dagster-tableau] Update Tableau docs with custom materialization example (#25672) ## Summary & Motivation As title. --- docs/content/integrations/tableau.mdx | 66 +++++++++++++++++++ .../materialize-tableau-assets-advanced.py | 59 +++++++++++++++++ 2 files changed, 125 insertions(+) create mode 100644 examples/docs_snippets/docs_snippets/integrations/tableau/materialize-tableau-assets-advanced.py diff --git a/docs/content/integrations/tableau.mdx b/docs/content/integrations/tableau.mdx index d628fbdcfcc72..5c1ba6833e633 100644 --- a/docs/content/integrations/tableau.mdx +++ b/docs/content/integrations/tableau.mdx @@ -302,6 +302,72 @@ defs = dg.Definitions( ) ``` +### Customizing how Tableau assets are materialized + +Instead of using the out-of-the-box utility, you can build your own assets definition that trigger the refresh of your Tableau workbooks. This allows you to customize how the refresh is triggered or to run custom code before or after the refresh. + +```python file=/integrations/tableau/materialize-tableau-assets-advanced.py +from typing import Sequence + +from dagster_tableau import ( + TableauCloudWorkspace, + load_tableau_asset_specs, + parse_tableau_external_and_materializable_asset_specs, +) + +import dagster as dg + +tableau_workspace = TableauCloudWorkspace( + connected_app_client_id=dg.EnvVar("TABLEAU_CONNECTED_APP_CLIENT_ID"), + connected_app_secret_id=dg.EnvVar("TABLEAU_CONNECTED_APP_SECRET_ID"), + connected_app_secret_value=dg.EnvVar("TABLEAU_CONNECTED_APP_SECRET_VALUE"), + username=dg.EnvVar("TABLEAU_USERNAME"), + site_name=dg.EnvVar("TABLEAU_SITE_NAME"), + pod_name=dg.EnvVar("TABLEAU_POD_NAME"), +) + + +# Assets definition factory which triggers workbooks refresh and sends a notification once complete +def build_tableau_materialize_and_notify_asset_def( + specs: Sequence[dg.AssetSpec], refreshable_workbook_ids: Sequence[str] +) -> dg.AssetsDefinition: + @dg.multi_asset( + name="tableau_sync", + compute_kind="tableau", + specs=specs, + ) + def asset_fn(context: dg.AssetExecutionContext, tableau: TableauCloudWorkspace): + with tableau.get_client() as client: + yield from client.refresh_and_materialize_workbooks( + specs=specs, refreshable_workbook_ids=refreshable_workbook_ids + ) + # Do some custom work after refreshing here, such as sending an email notification + + return asset_fn + + +# Load Tableau asset specs +tableau_specs = load_tableau_asset_specs( + workspace=tableau_workspace, +) + +external_asset_specs, materializable_asset_specs = ( + parse_tableau_external_and_materializable_asset_specs(tableau_specs) +) + +# Use the asset definition builder to construct the definition for tableau materializable assets +defs = dg.Definitions( + assets=[ + build_tableau_materialize_and_notify_asset_def( + specs=materializable_asset_specs, + refreshable_workbook_ids=["b75fc023-a7ca-4115-857b-4342028640d0"], + ), + *external_asset_specs, + ], + resources={"tableau": tableau_workspace}, +) +``` + ### Related - [`dagster-tableau` API reference](/\_apidocs/libraries/dagster-tableau) diff --git a/examples/docs_snippets/docs_snippets/integrations/tableau/materialize-tableau-assets-advanced.py b/examples/docs_snippets/docs_snippets/integrations/tableau/materialize-tableau-assets-advanced.py new file mode 100644 index 0000000000000..0d5b26ea11b77 --- /dev/null +++ b/examples/docs_snippets/docs_snippets/integrations/tableau/materialize-tableau-assets-advanced.py @@ -0,0 +1,59 @@ +from typing import Sequence + +from dagster_tableau import ( + TableauCloudWorkspace, + load_tableau_asset_specs, + parse_tableau_external_and_materializable_asset_specs, +) + +import dagster as dg + +tableau_workspace = TableauCloudWorkspace( + connected_app_client_id=dg.EnvVar("TABLEAU_CONNECTED_APP_CLIENT_ID"), + connected_app_secret_id=dg.EnvVar("TABLEAU_CONNECTED_APP_SECRET_ID"), + connected_app_secret_value=dg.EnvVar("TABLEAU_CONNECTED_APP_SECRET_VALUE"), + username=dg.EnvVar("TABLEAU_USERNAME"), + site_name=dg.EnvVar("TABLEAU_SITE_NAME"), + pod_name=dg.EnvVar("TABLEAU_POD_NAME"), +) + + +# Assets definition factory which triggers workbooks refresh and sends a notification once complete +def build_tableau_materialize_and_notify_asset_def( + specs: Sequence[dg.AssetSpec], refreshable_workbook_ids: Sequence[str] +) -> dg.AssetsDefinition: + @dg.multi_asset( + name="tableau_sync", + compute_kind="tableau", + specs=specs, + ) + def asset_fn(context: dg.AssetExecutionContext, tableau: TableauCloudWorkspace): + with tableau.get_client() as client: + yield from client.refresh_and_materialize_workbooks( + specs=specs, refreshable_workbook_ids=refreshable_workbook_ids + ) + # Do some custom work after refreshing here, such as sending an email notification + + return asset_fn + + +# Load Tableau asset specs +tableau_specs = load_tableau_asset_specs( + workspace=tableau_workspace, +) + +external_asset_specs, materializable_asset_specs = ( + parse_tableau_external_and_materializable_asset_specs(tableau_specs) +) + +# Use the asset definition builder to construct the definition for tableau materializable assets +defs = dg.Definitions( + assets=[ + build_tableau_materialize_and_notify_asset_def( + specs=materializable_asset_specs, + refreshable_workbook_ids=["b75fc023-a7ca-4115-857b-4342028640d0"], + ), + *external_asset_specs, + ], + resources={"tableau": tableau_workspace}, +)