From aca1cd7fe41627e31b1e5ce9df3f1b1474b5c35b Mon Sep 17 00:00:00 2001 From: david-leifker <114954101+david-leifker@users.noreply.github.com> Date: Tue, 3 Dec 2024 09:25:24 -0600 Subject: [PATCH 1/9] fix(test): fix metadata-io tests (#12006) --- .../search/fixtures/SampleDataFixtureTestBase.java | 8 ++++---- .../src/test/resources/search_config_fixture_test.yml | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/metadata-io/src/test/java/com/linkedin/metadata/search/fixtures/SampleDataFixtureTestBase.java b/metadata-io/src/test/java/com/linkedin/metadata/search/fixtures/SampleDataFixtureTestBase.java index 504eb5f5fc13d..fd663de40e005 100644 --- a/metadata-io/src/test/java/com/linkedin/metadata/search/fixtures/SampleDataFixtureTestBase.java +++ b/metadata-io/src/test/java/com/linkedin/metadata/search/fixtures/SampleDataFixtureTestBase.java @@ -1367,8 +1367,8 @@ public void testScrollAcrossEntities() throws IOException { resultUrns.addAll(result.getEntities().stream().map(SearchEntity::getEntity).toList()); scrollId = result.getScrollId(); } while (scrollId != null); - // expect 2 total matching results - assertEquals(totalResults, 2, String.format("query `%s` Results: %s", query, resultUrns)); + // expect 8 total matching results + assertEquals(totalResults, 8, String.format("query `%s` Results: %s", query, resultUrns)); } @Test @@ -1745,7 +1745,7 @@ public void testOr() { String.format("%s - Expected search results to include matched fields", query)); assertEquals( result.getEntities().size(), - 2, + 8, String.format( "Query: `%s` Results: %s", query, @@ -1776,7 +1776,7 @@ public void testNegate() { String.format("%s - Expected search results to include matched fields", query)); assertEquals( result.getEntities().size(), - 2, + 8, String.format( "Query: `%s` Results: %s", query, diff --git a/metadata-io/src/test/resources/search_config_fixture_test.yml b/metadata-io/src/test/resources/search_config_fixture_test.yml index 08e713c6b1cd3..e3c97c267188f 100644 --- a/metadata-io/src/test/resources/search_config_fixture_test.yml +++ b/metadata-io/src/test/resources/search_config_fixture_test.yml @@ -57,9 +57,9 @@ queryConfigurations: boost_mode: replace # Criteria for exact-match only - # Contains quotes, is a single term with `_`, `.`, or `-` (normally consider for tokenization) then use exact match query + # Contains quotes - queryRegex: >- - ^["'].+["']$|^[a-zA-Z0-9]\S+[_.-]\S+[a-zA-Z0-9]$ + ^["'].+["']$ simpleQuery: false prefixMatchQuery: true exactMatchQuery: true From 230bd2674bb10ae004a1753a30e9e3d0e0a573e9 Mon Sep 17 00:00:00 2001 From: Tamas Nemeth Date: Tue, 3 Dec 2024 20:01:12 +0100 Subject: [PATCH 2/9] fix(ingest/looker): Don't fail on unknown liquid filters (#12014) --- .../datahub/ingestion/source/looker/looker_liquid_tag.py | 9 ++++++++- .../tests/integration/lookml/test_lookml.py | 6 +++--- 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_liquid_tag.py b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_liquid_tag.py index 7d4ebf00cc06e..f48ba6758564b 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/looker/looker_liquid_tag.py +++ b/metadata-ingestion/src/datahub/ingestion/source/looker/looker_liquid_tag.py @@ -4,6 +4,7 @@ from liquid import Environment from liquid.ast import Node from liquid.context import Context +from liquid.filter import string_filter from liquid.parse import expect, get_parser from liquid.stream import TokenStream from liquid.tag import Tag @@ -81,12 +82,18 @@ def parse(self, stream: TokenStream) -> Node: custom_tags = [ConditionTag] +@string_filter +def sql_quote_filter(variable: str) -> str: + return f"'{variable}'" + + @lru_cache(maxsize=1) def _create_env() -> Environment: - env: Environment = Environment() + env: Environment = Environment(strict_filters=False) # register tag. One time activity for custom_tag in custom_tags: env.add_tag(custom_tag) + env.add_filter("sql_quote", sql_quote_filter) return env diff --git a/metadata-ingestion/tests/integration/lookml/test_lookml.py b/metadata-ingestion/tests/integration/lookml/test_lookml.py index ab55321a4d734..4cd2777dc7dca 100644 --- a/metadata-ingestion/tests/integration/lookml/test_lookml.py +++ b/metadata-ingestion/tests/integration/lookml/test_lookml.py @@ -889,7 +889,7 @@ def test_view_to_view_lineage_and_liquid_template(pytestconfig, tmp_path, mock_t @freeze_time(FROZEN_TIME) def test_special_liquid_variables(): - text: str = """ + text: str = """{% assign source_table_variable = "source_table" | sql_quote | non_existing_filter_where_it_should_not_fail %} SELECT employee_id, employee_name, @@ -903,7 +903,7 @@ def test_special_liquid_variables(): 'default_table' as source {% endif %}, employee_income - FROM source_table + FROM {{ source_table_variable }} """ input_liquid_variable: dict = {} @@ -958,7 +958,7 @@ def test_special_liquid_variables(): expected_text: str = ( "\n SELECT\n employee_id,\n employee_name,\n \n " "prod_core.data.r_metric_summary_v2\n ,\n employee_income\n FROM " - "source_table\n " + "'source_table'\n " ) assert actual_text == expected_text From b31d849b9f9f6a57fc73d58b881e6ae955c9aeb1 Mon Sep 17 00:00:00 2001 From: Jay <159848059+jayacryl@users.noreply.github.com> Date: Tue, 3 Dec 2024 14:04:54 -0500 Subject: [PATCH 3/9] feat(docs-website) fix links (#12019) --- docs-website/src/pages/cloud/UnifiedTabs/index.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/docs-website/src/pages/cloud/UnifiedTabs/index.js b/docs-website/src/pages/cloud/UnifiedTabs/index.js index c0fbc25a8de6b..d17138fcee629 100644 --- a/docs-website/src/pages/cloud/UnifiedTabs/index.js +++ b/docs-website/src/pages/cloud/UnifiedTabs/index.js @@ -11,21 +11,21 @@ const TabbedComponent = () => { title: 'Discovery', description: 'All the search and discovery features of DataHub Core you already love, enhanced.', icon: "/img/assets/data-discovery.svg", - link: "https://www.acryldata.io/acryl-datahub", + link: "https://datahubproject.io/solutions/discovery", image: 'https://raw.githubusercontent.com/datahub-project/static-assets/main/imgs/saas/demo/discovery.webm', }, { title: 'Observability', description: 'Detect, resolve, and prevent data quality issues before they impact your business. Unify data health signals from all your data quality tools, including dbt tests and more.', icon: "/img/assets/data-ob.svg", - link: "https://www.acryldata.io/observe", + link: "https://datahubproject.io/solutions/observability", image: 'https://raw.githubusercontent.com/datahub-project/static-assets/main/imgs/saas/demo/observe.webm', }, { title: 'Governance', description: 'Powerful Automation, Reporting and Organizational tools to help you govern effectively.', icon: "/img/assets/data-governance.svg", - link: "https://www.acryldata.io/acryl-datahub#governance", + link: "https://datahubproject.io/solutions/governance", image: 'https://raw.githubusercontent.com/datahub-project/static-assets/main/imgs/saas/demo/governance.webm', }, ]; From a004c9293d68b4e04e6d1024b74ffc27d4546336 Mon Sep 17 00:00:00 2001 From: david-leifker <114954101+david-leifker@users.noreply.github.com> Date: Tue, 3 Dec 2024 15:47:08 -0600 Subject: [PATCH 4/9] fix(ci): fix datahub-client validatePythonEnv (#12023) --- metadata-integration/java/datahub-client/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metadata-integration/java/datahub-client/build.gradle b/metadata-integration/java/datahub-client/build.gradle index af71227809d2a..2535d091f6ce5 100644 --- a/metadata-integration/java/datahub-client/build.gradle +++ b/metadata-integration/java/datahub-client/build.gradle @@ -62,7 +62,7 @@ compileJava.dependsOn copyAvroSchemas // Add Python environment validation task -task validatePythonEnv { +task validatePythonEnv(dependsOn: [":metadata-ingestion:installDev"]) { doFirst { def venvPath = System.getProperty('python.venv.path', '../../../metadata-ingestion/venv') def isWindows = System.getProperty('os.name').toLowerCase().contains('windows') From 82774bb65eb53d4b2126b08ff343d8410226038e Mon Sep 17 00:00:00 2001 From: david-leifker <114954101+david-leifker@users.noreply.github.com> Date: Tue, 3 Dec 2024 15:48:46 -0600 Subject: [PATCH 5/9] test(urn-validation): additional test case (#12001) --- .../entity/validation/ValidationApiUtilsTest.java | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/metadata-io/metadata-io-api/src/test/java/com/linkedin/metadata/entity/validation/ValidationApiUtilsTest.java b/metadata-io/metadata-io-api/src/test/java/com/linkedin/metadata/entity/validation/ValidationApiUtilsTest.java index a2c9a15d92f90..2ab6a50945ba3 100644 --- a/metadata-io/metadata-io-api/src/test/java/com/linkedin/metadata/entity/validation/ValidationApiUtilsTest.java +++ b/metadata-io/metadata-io-api/src/test/java/com/linkedin/metadata/entity/validation/ValidationApiUtilsTest.java @@ -79,11 +79,19 @@ public void testUrnWithIllegalDelimiter() { } @Test(expectedExceptions = IllegalArgumentException.class) - public void testComplexUrnWithParens() { + public void testComplexUrnWithParens1() { Urn invalidUrn = UrnUtils.getUrn("urn:li:dataset:(urn:li:dataPlatform:hdfs,(illegal),PROD)"); ValidationApiUtils.validateUrn(entityRegistry, invalidUrn, true); } + @Test(expectedExceptions = IllegalArgumentException.class) + public void testComplexUrnWithParens2() { + Urn invalidUrn = + UrnUtils.getUrn( + "urn:li:dataJob:(urn:li:dataFlow:(mssql,1/2/3/4.c_n on %28LOCAL%29,PROD),1/2/3/4.c_n on (LOCAL))"); + ValidationApiUtils.validateUrn(entityRegistry, invalidUrn, true); + } + @Test(expectedExceptions = IllegalArgumentException.class) public void testSimpleUrnWithParens() { Urn invalidUrn = UrnUtils.getUrn("urn:li:corpuser:(foo)123"); From eef9759f880b74369567ba7f297ebb0406345f6f Mon Sep 17 00:00:00 2001 From: Shirshanka Das Date: Tue, 3 Dec 2024 14:59:34 -0800 Subject: [PATCH 6/9] feat(hudi): add hudi platform to the list of default platforms (#11993) Co-authored-by: Chris Collins --- datahub-web-react/src/images/hudilogo.png | Bin 0 -> 75205 bytes .../src/main/resources/bootstrap_mcps.yaml | 2 +- .../bootstrap_mcps/data-platforms.yaml | 10 ++++++++++ 3 files changed, 11 insertions(+), 1 deletion(-) create mode 100644 datahub-web-react/src/images/hudilogo.png diff --git a/datahub-web-react/src/images/hudilogo.png b/datahub-web-react/src/images/hudilogo.png new file mode 100644 index 0000000000000000000000000000000000000000..4b58cc5a34826df31e370f810a93116f033c5c4d GIT binary patch literal 75205 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

QC6MHl-Ni#XX0jMZKr+v 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 W?qhOY}MbX_iQ8K%>M&w(q3%< literal 0 HcmV?d00001 diff --git a/metadata-service/configuration/src/main/resources/bootstrap_mcps.yaml b/metadata-service/configuration/src/main/resources/bootstrap_mcps.yaml index 0e283dfdfc93c..a81cf39ce386f 100644 --- a/metadata-service/configuration/src/main/resources/bootstrap_mcps.yaml +++ b/metadata-service/configuration/src/main/resources/bootstrap_mcps.yaml @@ -13,7 +13,7 @@ bootstrap: mcps_location: "bootstrap_mcps/root-user.yaml" - name: data-platforms - version: v1 + version: v2 blocking: true async: false mcps_location: "bootstrap_mcps/data-platforms.yaml" diff --git a/metadata-service/configuration/src/main/resources/bootstrap_mcps/data-platforms.yaml b/metadata-service/configuration/src/main/resources/bootstrap_mcps/data-platforms.yaml index 0b3d815c71098..2230d552ed4c0 100644 --- a/metadata-service/configuration/src/main/resources/bootstrap_mcps/data-platforms.yaml +++ b/metadata-service/configuration/src/main/resources/bootstrap_mcps/data-platforms.yaml @@ -119,6 +119,16 @@ displayName: Hive type: FILE_SYSTEM logoUrl: "/assets/platforms/hivelogo.png" +- entityUrn: urn:li:dataPlatform:hudi + entityType: dataPlatform + aspectName: dataPlatformInfo + changeType: UPSERT + aspect: + datasetNameDelimiter: "." + name: hudi + displayName: Hudi + type: FILE_SYSTEM + logoUrl: "/assets/platforms/hudilogo.png" - entityUrn: urn:li:dataPlatform:iceberg entityType: dataPlatform aspectName: dataPlatformInfo From 49b6284ebfa6fae65bf463e0eb3218b9793bb1f2 Mon Sep 17 00:00:00 2001 From: Steffen Grohsschmiedt Date: Wed, 4 Dec 2024 01:16:44 +0100 Subject: [PATCH 7/9] fix(airflow): fix AthenaOperator extraction (#11857) Co-authored-by: Harshal Sheth --- .../airflow-plugin/setup.py | 2 +- .../src/datahub_airflow_plugin/_extractors.py | 24 +- .../tests/integration/dags/athena_operator.py | 43 ++ .../goldens/v2_athena_operator.json | 672 ++++++++++++++++++ .../v2_athena_operator_no_dag_listener.json | 672 ++++++++++++++++++ .../tests/integration/test_plugin.py | 29 + 6 files changed, 1440 insertions(+), 2 deletions(-) create mode 100644 metadata-ingestion-modules/airflow-plugin/tests/integration/dags/athena_operator.py create mode 100644 metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator.json create mode 100644 metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator_no_dag_listener.json diff --git a/metadata-ingestion-modules/airflow-plugin/setup.py b/metadata-ingestion-modules/airflow-plugin/setup.py index 0d5ceefd989dc..02a0bbb6022e0 100644 --- a/metadata-ingestion-modules/airflow-plugin/setup.py +++ b/metadata-ingestion-modules/airflow-plugin/setup.py @@ -96,7 +96,7 @@ def get_long_description(): *plugins["datahub-kafka"], f"acryl-datahub[testing-utils]{_self_pin}", # Extra requirements for loading our test dags. - "apache-airflow[snowflake]>=2.0.2", + "apache-airflow[snowflake,amazon]>=2.0.2", # A collection of issues we've encountered: # - Connexion's new version breaks Airflow: # See https://github.com/apache/airflow/issues/35234. diff --git a/metadata-ingestion-modules/airflow-plugin/src/datahub_airflow_plugin/_extractors.py b/metadata-ingestion-modules/airflow-plugin/src/datahub_airflow_plugin/_extractors.py index de0d4f8711f53..28d5775f61f54 100644 --- a/metadata-ingestion-modules/airflow-plugin/src/datahub_airflow_plugin/_extractors.py +++ b/metadata-ingestion-modules/airflow-plugin/src/datahub_airflow_plugin/_extractors.py @@ -50,7 +50,6 @@ def __init__(self): "BigQueryOperator", "BigQueryExecuteQueryOperator", # Athena also does something similar. - "AthenaOperator", "AWSAthenaOperator", # Additional types that OL doesn't support. This is only necessary because # on older versions of Airflow, these operators don't inherit from SQLExecuteQueryOperator. @@ -59,6 +58,8 @@ def __init__(self): for operator in _sql_operator_overrides: self.task_to_extractor.extractors[operator] = GenericSqlExtractor + self.task_to_extractor.extractors["AthenaOperator"] = AthenaOperatorExtractor + self.task_to_extractor.extractors[ "BigQueryInsertJobOperator" ] = BigQueryInsertJobOperatorExtractor @@ -276,6 +277,27 @@ def extract(self) -> Optional[TaskMetadata]: ) +class AthenaOperatorExtractor(BaseExtractor): + def extract(self) -> Optional[TaskMetadata]: + from airflow.providers.amazon.aws.operators.athena import ( + AthenaOperator, # type: ignore + ) + + operator: "AthenaOperator" = self.operator + sql = operator.query + if not sql: + self.log.warning("No query found in AthenaOperator") + return None + + return _parse_sql_into_task_metadata( + self, + sql, + platform="athena", + default_database=None, + default_schema=self.operator.database, + ) + + def _snowflake_default_schema(self: "SnowflakeExtractor") -> Optional[str]: if hasattr(self.operator, "schema") and self.operator.schema is not None: return self.operator.schema diff --git a/metadata-ingestion-modules/airflow-plugin/tests/integration/dags/athena_operator.py b/metadata-ingestion-modules/airflow-plugin/tests/integration/dags/athena_operator.py new file mode 100644 index 0000000000000..96cdacbbad37d --- /dev/null +++ b/metadata-ingestion-modules/airflow-plugin/tests/integration/dags/athena_operator.py @@ -0,0 +1,43 @@ +from datetime import datetime + +from airflow import DAG +from airflow.providers.amazon.aws.operators.athena import AthenaOperator + +ATHENA_COST_TABLE = "costs" +ATHENA_PROCESSED_TABLE = "processed_costs" + + +def _fake_athena_execute(*args, **kwargs): + pass + + +with DAG( + "athena_operator", + start_date=datetime(2023, 1, 1), + schedule_interval=None, + catchup=False, +) as dag: + # HACK: We don't want to send real requests to Athena. As a workaround, + # we can simply monkey-patch the operator. + AthenaOperator.execute = _fake_athena_execute # type: ignore + + transform_cost_table = AthenaOperator( + aws_conn_id="my_aws", + task_id="transform_cost_table", + database="athena_db", + query=""" + CREATE OR REPLACE TABLE {{ params.out_table_name }} AS + SELECT + id, + month, + total_cost, + area, + total_cost / area as cost_per_area + FROM {{ params.in_table_name }} + """, + params={ + "in_table_name": ATHENA_COST_TABLE, + "out_table_name": ATHENA_PROCESSED_TABLE, + }, + output_location="s3://athena-results-bucket/", + ) diff --git a/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator.json b/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator.json new file mode 100644 index 0000000000000..baa738fef7b5f --- /dev/null +++ b/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator.json @@ -0,0 +1,672 @@ +[ +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "dataFlowInfo", + "aspect": { + "json": { + "customProperties": { + "_access_control": "None", + "catchup": "False", + "description": "None", + "doc_md": "None", + "fileloc": "", + "is_paused_upon_creation": "None", + "start_date": "DateTime(2023, 1, 1, 0, 0, 0, tzinfo=Timezone('UTC'))", + "tags": "[]", + "timezone": "Timezone('UTC')" + }, + "externalUrl": "http://airflow.example.com/tree?dag_id=athena_operator", + "name": "athena_operator", + "env": "PROD" + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "browsePathsV2", + "aspect": { + "json": { + "path": [ + { + "id": "athena_operator" + } + ] + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInfo", + "aspect": { + "json": { + "customProperties": { + "depends_on_past": "False", + "email": "None", + "label": "'transform_cost_table'", + "execution_timeout": "None", + "sla": "None", + "task_id": "'transform_cost_table'", + "trigger_rule": "", + "wait_for_downstream": "False", + "downstream_task_ids": "[]", + "inlets": "[]", + "outlets": "[]", + "openlineage_job_facet_sql": "{\"_producer\": \"https://github.com/OpenLineage/OpenLineage/tree/1.22.0/integration/airflow\", \"_schemaURL\": \"https://raw.githubusercontent.com/OpenLineage/OpenLineage/main/spec/OpenLineage.json#/definitions/SqlJobFacet\", \"query\": \"\\n CREATE OR REPLACE TABLE processed_costs AS\\n SELECT\\n id,\\n month,\\n total_cost,\\n area,\\n total_cost / area as cost_per_area\\n FROM costs\\n \"}" + }, + "externalUrl": "http://airflow.example.com/taskinstance/list/?flt1_dag_id_equals=athena_operator&_flt_3_task_id=transform_cost_table", + "name": "transform_cost_table", + "type": { + "string": "COMMAND" + }, + "env": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInputOutput", + "aspect": { + "json": { + "inputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ], + "outputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ], + "inputDatajobs": [], + "fineGrainedLineages": [ + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + } + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceProperties", + "aspect": { + "json": { + "customProperties": { + "run_id": "manual_run_test", + "duration": "", + "start_date": "", + "end_date": "", + "execution_date": "2023-09-27 21:34:38+00:00", + "try_number": "0", + "max_tries": "0", + "external_executor_id": "None", + "state": "running", + "operator": "AthenaOperator", + "priority_weight": "1", + "log_url": "http://airflow.example.com/dags/athena_operator/grid?dag_run_id=manual_run_test&task_id=transform_cost_table&map_index=-1&tab=logs", + "orchestrator": "airflow", + "dag_id": "athena_operator", + "task_id": "transform_cost_table" + }, + "externalUrl": "http://airflow.example.com/dags/athena_operator/grid?dag_run_id=manual_run_test&task_id=transform_cost_table&map_index=-1&tab=logs", + "name": "athena_operator_transform_cost_table_manual_run_test", + "type": "BATCH_AD_HOC", + "created": { + "time": 1732719433576, + "actor": "urn:li:corpuser:datahub" + } + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRelationships", + "aspect": { + "json": { + "parentTemplate": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "upstreamInstances": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceInput", + "aspect": { + "json": { + "inputs": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceOutput", + "aspect": { + "json": { + "outputs": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRunEvent", + "aspect": { + "json": { + "timestampMillis": 1732719433576, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "status": "STARTED", + "attempt": 1 + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "operation", + "aspect": { + "json": { + "timestampMillis": 1732719433736, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "actor": "urn:li:corpuser:airflow", + "operationType": "CREATE", + "lastUpdatedTimestamp": 1732719433736 + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInfo", + "aspect": { + "json": { + "customProperties": { + "depends_on_past": "False", + "email": "None", + "label": "'transform_cost_table'", + "execution_timeout": "None", + "sla": "None", + "task_id": "'transform_cost_table'", + "trigger_rule": "", + "wait_for_downstream": "False", + "downstream_task_ids": "[]", + "inlets": "[]", + "outlets": "[]", + "openlineage_job_facet_sql": "{\"_producer\": \"https://github.com/OpenLineage/OpenLineage/tree/1.22.0/integration/airflow\", \"_schemaURL\": \"https://raw.githubusercontent.com/OpenLineage/OpenLineage/main/spec/OpenLineage.json#/definitions/SqlJobFacet\", \"query\": \"\\n CREATE OR REPLACE TABLE processed_costs AS\\n SELECT\\n id,\\n month,\\n total_cost,\\n area,\\n total_cost / area as cost_per_area\\n FROM costs\\n \"}" + }, + "externalUrl": "http://airflow.example.com/taskinstance/list/?flt1_dag_id_equals=athena_operator&_flt_3_task_id=transform_cost_table", + "name": "transform_cost_table", + "type": { + "string": "COMMAND" + }, + "env": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInputOutput", + "aspect": { + "json": { + "inputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ], + "outputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ], + "inputDatajobs": [], + "fineGrainedLineages": [ + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + } + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRunEvent", + "aspect": { + "json": { + "timestampMillis": 1732719433747, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "status": "COMPLETE", + "result": { + "type": "SUCCESS", + "nativeResultType": "airflow" + } + } + } +} +] \ No newline at end of file diff --git a/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator_no_dag_listener.json b/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator_no_dag_listener.json new file mode 100644 index 0000000000000..c53825a9979e3 --- /dev/null +++ b/metadata-ingestion-modules/airflow-plugin/tests/integration/goldens/v2_athena_operator_no_dag_listener.json @@ -0,0 +1,672 @@ +[ +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "dataFlowInfo", + "aspect": { + "json": { + "customProperties": { + "_access_control": "None", + "catchup": "False", + "description": "None", + "doc_md": "None", + "fileloc": "", + "is_paused_upon_creation": "None", + "start_date": "DateTime(2023, 1, 1, 0, 0, 0, tzinfo=Timezone('UTC'))", + "tags": "[]", + "timezone": "Timezone('UTC')" + }, + "externalUrl": "http://airflow.example.com/tree?dag_id=athena_operator", + "name": "athena_operator", + "env": "PROD" + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataFlow", + "entityUrn": "urn:li:dataFlow:(airflow,athena_operator,prod)", + "changeType": "UPSERT", + "aspectName": "browsePathsV2", + "aspect": { + "json": { + "path": [ + { + "id": "athena_operator" + } + ] + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInfo", + "aspect": { + "json": { + "customProperties": { + "depends_on_past": "False", + "email": "None", + "label": "'transform_cost_table'", + "execution_timeout": "None", + "sla": "None", + "task_id": "'transform_cost_table'", + "trigger_rule": "", + "wait_for_downstream": "False", + "downstream_task_ids": "[]", + "inlets": "[]", + "outlets": "[]", + "openlineage_job_facet_sql": "{\"_producer\": \"https://github.com/OpenLineage/OpenLineage/tree/1.22.0/integration/airflow\", \"_schemaURL\": \"https://raw.githubusercontent.com/OpenLineage/OpenLineage/main/spec/OpenLineage.json#/definitions/SqlJobFacet\", \"query\": \"\\n CREATE OR REPLACE TABLE processed_costs AS\\n SELECT\\n id,\\n month,\\n total_cost,\\n area,\\n total_cost / area as cost_per_area\\n FROM costs\\n \"}" + }, + "externalUrl": "http://airflow.example.com/taskinstance/list/?flt1_dag_id_equals=athena_operator&_flt_3_task_id=transform_cost_table", + "name": "transform_cost_table", + "type": { + "string": "COMMAND" + }, + "env": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInputOutput", + "aspect": { + "json": { + "inputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ], + "outputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ], + "inputDatajobs": [], + "fineGrainedLineages": [ + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + } + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceProperties", + "aspect": { + "json": { + "customProperties": { + "run_id": "manual_run_test", + "duration": "", + "start_date": "", + "end_date": "", + "execution_date": "2023-09-27 21:34:38+00:00", + "try_number": "0", + "max_tries": "0", + "external_executor_id": "None", + "state": "running", + "operator": "AthenaOperator", + "priority_weight": "1", + "log_url": "http://airflow.example.com/log?execution_date=2023-09-27T21%3A34%3A38%2B00%3A00&task_id=transform_cost_table&dag_id=athena_operator&map_index=-1", + "orchestrator": "airflow", + "dag_id": "athena_operator", + "task_id": "transform_cost_table" + }, + "externalUrl": "http://airflow.example.com/log?execution_date=2023-09-27T21%3A34%3A38%2B00%3A00&task_id=transform_cost_table&dag_id=athena_operator&map_index=-1", + "name": "athena_operator_transform_cost_table_manual_run_test", + "type": "BATCH_AD_HOC", + "created": { + "time": 1733121901482, + "actor": "urn:li:corpuser:datahub" + } + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRelationships", + "aspect": { + "json": { + "parentTemplate": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "upstreamInstances": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceInput", + "aspect": { + "json": { + "inputs": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceOutput", + "aspect": { + "json": { + "outputs": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRunEvent", + "aspect": { + "json": { + "timestampMillis": 1733121901482, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "status": "STARTED", + "attempt": 1 + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "operation", + "aspect": { + "json": { + "timestampMillis": 1733121901625, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "actor": "urn:li:corpuser:airflow", + "operationType": "CREATE", + "lastUpdatedTimestamp": 1733121901625 + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInfo", + "aspect": { + "json": { + "customProperties": { + "depends_on_past": "False", + "email": "None", + "label": "'transform_cost_table'", + "execution_timeout": "None", + "sla": "None", + "task_id": "'transform_cost_table'", + "trigger_rule": "", + "wait_for_downstream": "False", + "downstream_task_ids": "[]", + "inlets": "[]", + "outlets": "[]", + "openlineage_job_facet_sql": "{\"_producer\": \"https://github.com/OpenLineage/OpenLineage/tree/1.22.0/integration/airflow\", \"_schemaURL\": \"https://raw.githubusercontent.com/OpenLineage/OpenLineage/main/spec/OpenLineage.json#/definitions/SqlJobFacet\", \"query\": \"\\n CREATE OR REPLACE TABLE processed_costs AS\\n SELECT\\n id,\\n month,\\n total_cost,\\n area,\\n total_cost / area as cost_per_area\\n FROM costs\\n \"}" + }, + "externalUrl": "http://airflow.example.com/taskinstance/list/?flt1_dag_id_equals=athena_operator&_flt_3_task_id=transform_cost_table", + "name": "transform_cost_table", + "type": { + "string": "COMMAND" + }, + "env": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "status", + "aspect": { + "json": { + "removed": false + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "dataJobInputOutput", + "aspect": { + "json": { + "inputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)" + ], + "outputDatasets": [ + "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)" + ], + "inputDatajobs": [], + "fineGrainedLineages": [ + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),id)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),id)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),month)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),month)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),total_cost)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),area)" + ], + "confidenceScore": 1.0 + }, + { + "upstreamType": "FIELD_SET", + "upstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),area)", + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD),total_cost)" + ], + "downstreamType": "FIELD", + "downstreams": [ + "urn:li:schemaField:(urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD),cost_per_area)" + ], + "confidenceScore": 1.0 + } + ] + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataset", + "entityUrn": "urn:li:dataset:(urn:li:dataPlatform:athena,athena_db.processed_costs,PROD)", + "changeType": "UPSERT", + "aspectName": "datasetKey", + "aspect": { + "json": { + "platform": "urn:li:dataPlatform:athena", + "name": "athena_db.processed_costs", + "origin": "PROD" + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "ownership", + "aspect": { + "json": { + "owners": [ + { + "owner": "urn:li:corpuser:airflow", + "type": "DEVELOPER", + "source": { + "type": "SERVICE" + } + } + ], + "ownerTypes": {}, + "lastModified": { + "time": 0, + "actor": "urn:li:corpuser:airflow" + } + } + } +}, +{ + "entityType": "dataJob", + "entityUrn": "urn:li:dataJob:(urn:li:dataFlow:(airflow,athena_operator,prod),transform_cost_table)", + "changeType": "UPSERT", + "aspectName": "globalTags", + "aspect": { + "json": { + "tags": [] + } + } +}, +{ + "entityType": "dataProcessInstance", + "entityUrn": "urn:li:dataProcessInstance:9cd4fbcec3a50a4988ffc5841beaf0ad", + "changeType": "UPSERT", + "aspectName": "dataProcessInstanceRunEvent", + "aspect": { + "json": { + "timestampMillis": 1733121901675, + "partitionSpec": { + "partition": "FULL_TABLE_SNAPSHOT", + "type": "FULL_TABLE" + }, + "status": "COMPLETE", + "result": { + "type": "SUCCESS", + "nativeResultType": "airflow" + } + } + } +} +] \ No newline at end of file diff --git a/metadata-ingestion-modules/airflow-plugin/tests/integration/test_plugin.py b/metadata-ingestion-modules/airflow-plugin/tests/integration/test_plugin.py index 3becf10703df6..75bb43af1a43d 100644 --- a/metadata-ingestion-modules/airflow-plugin/tests/integration/test_plugin.py +++ b/metadata-ingestion-modules/airflow-plugin/tests/integration/test_plugin.py @@ -111,6 +111,24 @@ def _wait_for_dag_finish( raise NotReadyError(f"DAG has not finished yet: {dag_run['state']}") +@tenacity.retry( + reraise=True, + wait=tenacity.wait_fixed(1), + stop=tenacity.stop_after_delay(90), + retry=tenacity.retry_if_exception_type(NotReadyError), +) +def _wait_for_dag_to_load(airflow_instance: AirflowInstance, dag_id: str) -> None: + print("Checking if DAG was loaded") + res = airflow_instance.session.get( + url=f"{airflow_instance.airflow_url}/api/v1/dags", + timeout=5, + ) + res.raise_for_status() + + if len(list(filter(lambda x: x["dag_id"] == dag_id, res.json()["dags"]))) == 0: + raise NotReadyError("DAG was not loaded yet") + + def _dump_dag_logs(airflow_instance: AirflowInstance, dag_id: str) -> None: # Get the dag run info res = airflow_instance.session.get( @@ -206,6 +224,15 @@ def _run_airflow( "insecure_mode": "true", }, ).get_uri(), + "AIRFLOW_CONN_MY_AWS": Connection( + conn_id="my_aws", + conn_type="aws", + extra={ + "region_name": "us-east-1", + "aws_access_key_id": "AKIAIOSFODNN7EXAMPLE", + "aws_secret_access_key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY", + }, + ).get_uri(), "AIRFLOW_CONN_MY_SQLITE": Connection( conn_id="my_sqlite", conn_type="sqlite", @@ -327,6 +354,7 @@ class DagTestCase: DagTestCase("sqlite_operator", v2_only=True), DagTestCase("custom_operator_dag", v2_only=True), DagTestCase("datahub_emitter_operator_jinja_template_dag", v2_only=True), + DagTestCase("athena_operator", v2_only=True), ] @@ -398,6 +426,7 @@ def test_airflow_plugin( tmp_path, dags_folder=DAGS_FOLDER, is_v1=is_v1 ) as airflow_instance: print(f"Running DAG {dag_id}...") + _wait_for_dag_to_load(airflow_instance, dag_id) subprocess.check_call( [ "airflow", From df9755c9483d9d46603c82b122bbece71dad89be Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sergio=20G=C3=B3mez=20Villamor?= Date: Wed, 4 Dec 2024 10:06:25 +0100 Subject: [PATCH 8/9] feat(tableau): review reporting and debug traces (#12015) Co-authored-by: Harshal Sheth --- .../ingestion/source/tableau/tableau.py | 46 +++++++++++-------- 1 file changed, 27 insertions(+), 19 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py b/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py index 0eafdb4ad23ba..f3ad5ea706f7c 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py +++ b/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py @@ -289,16 +289,12 @@ def make_tableau_client(self, site: str) -> Server: server.auth.sign_in(authentication) return server except ServerResponseError as e: + message = f"Unable to login (invalid/expired credentials or missing permissions): {str(e)}" if isinstance(authentication, PersonalAccessTokenAuth): # Docs on token expiry in Tableau: # https://help.tableau.com/current/server/en-us/security_personal_access_tokens.htm#token-expiry - logger.info( - "Error authenticating with Tableau. Note that Tableau personal access tokens " - "expire if not used for 15 days or if over 1 year old" - ) - raise ValueError( - f"Unable to login (invalid/expired credentials or missing permissions): {str(e)}" - ) from e + message = f"Error authenticating with Tableau. Note that Tableau personal access tokens expire if not used for 15 days or if over 1 year old: {str(e)}" + raise ValueError(message) from e except Exception as e: raise ValueError( f"Unable to login (check your Tableau connection and credentials): {str(e)}" @@ -722,6 +718,7 @@ def get_workunits_internal(self) -> Iterable[MetadataWorkUnit]: title="Failed to Retrieve Tableau Metadata", message="Unable to retrieve metadata from tableau.", context=str(md_exception), + exc=md_exception, ) def close(self) -> None: @@ -826,6 +823,7 @@ def _populate_usage_stat_registry(self) -> None: if not view.id: continue self.tableau_stat_registry[view.id] = UsageStat(view_count=view.total_views) + logger.info(f"Got Tableau stats for {len(self.tableau_stat_registry)} assets") logger.debug("Tableau stats %s", self.tableau_stat_registry) def _populate_database_server_hostname_map(self) -> None: @@ -876,7 +874,7 @@ def form_path(project_id: str) -> List[str]: ancestors = [cur_proj.name] while cur_proj.parent_id is not None: if cur_proj.parent_id not in all_project_map: - self.report.report_warning( + self.report.warning( "project-issue", f"Parent project {cur_proj.parent_id} not found. We need Site Administrator Explorer permissions.", ) @@ -974,8 +972,11 @@ def _init_datasource_registry(self) -> None: self.datasource_project_map[ds.id] = ds.project_id except Exception as e: self.report.get_all_datasources_query_failed = True - logger.info(f"Get all datasources query failed due to error {e}") - logger.debug("Error stack trace", exc_info=True) + self.report.warning( + title="Unexpected Query Error", + message="Get all datasources query failed due to error", + exc=e, + ) def _init_workbook_registry(self) -> None: if self.server is None: @@ -1141,7 +1142,6 @@ def get_connection_object_page( ) if node_limit_errors: - logger.debug(f"Node Limit Error. query_data {query_data}") self.report.warning( title="Tableau Data Exceed Predefined Limit", message="The numbers of record in result set exceeds a predefined limit. Increase the tableau " @@ -1257,9 +1257,10 @@ def emit_workbooks(self) -> Iterable[MetadataWorkUnit]: wrk_id: Optional[str] = workbook.get(c.ID) prj_name: Optional[str] = workbook.get(c.PROJECT_NAME) - logger.debug( - f"Skipping workbook {wrk_name}({wrk_id}) as it is project {prj_name}({project_luid}) not " - f"present in project registry" + self.report.warning( + title="Skipping Missing Workbook", + message="Skipping workbook as its project is not present in project registry", + context=f"workbook={wrk_name}({wrk_id}), project={prj_name}({project_luid})", ) continue @@ -1453,7 +1454,7 @@ def get_upstream_tables( c.COLUMNS_CONNECTION ].get("totalCount") if not is_custom_sql and not num_tbl_cols: - logger.debug( + logger.warning( f"Skipping upstream table with id {table[c.ID]}, no columns: {table}" ) continue @@ -1469,7 +1470,12 @@ def get_upstream_tables( table, default_schema_map=self.config.default_schema_map ) except Exception as e: - logger.info(f"Failed to generate upstream reference for {table}: {e}") + self.report.warning( + title="Potentially Missing Lineage Issue", + message="Failed to generate upstream reference", + exc=e, + context=f"table={table}", + ) continue table_urn = ref.make_dataset_urn( @@ -1917,10 +1923,12 @@ def _query_published_datasource_for_project_luid(self, ds_luid: str) -> None: self.datasource_project_map[ds_result.id] = ds_result.project_id except Exception as e: self.report.num_get_datasource_query_failures += 1 - logger.warning( - f"Failed to get datasource project_luid for {ds_luid} due to error {e}" + self.report.warning( + title="Unexpected Query Error", + message="Failed to get datasource details", + exc=e, + context=f"ds_luid={ds_luid}", ) - logger.debug("Error stack trace", exc_info=True) def _get_workbook_project_luid(self, wb: dict) -> Optional[str]: if wb.get(c.LUID) and self.workbook_project_map.get(wb[c.LUID]): From 2b42b29d2fbfb12cfb68a0578b63993bcd182c07 Mon Sep 17 00:00:00 2001 From: Harshal Sheth Date: Wed, 4 Dec 2024 04:07:09 -0500 Subject: [PATCH 9/9] fix(ingest/tableau): make `sites.get_by_id` call optional (#12024) --- .../ingestion/source/tableau/tableau.py | 34 ++++++++++++++----- .../tableau/test_tableau_ingest.py | 2 ++ 2 files changed, 28 insertions(+), 8 deletions(-) diff --git a/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py b/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py index f3ad5ea706f7c..197e73dca7141 100644 --- a/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py +++ b/metadata-ingestion/src/datahub/ingestion/source/tableau/tableau.py @@ -68,6 +68,7 @@ CapabilityReport, MetadataWorkUnitProcessor, Source, + StructuredLogLevel, TestableSource, TestConnectionReport, ) @@ -696,6 +697,7 @@ def get_workunits_internal(self) -> Iterable[MetadataWorkUnit]: config=self.config, ctx=self.ctx, site=site, + site_id=site.id, report=self.report, server=self.server, platform=self.platform, @@ -703,11 +705,19 @@ def get_workunits_internal(self) -> Iterable[MetadataWorkUnit]: logger.info(f"Ingesting assets of site '{site.content_url}'.") yield from site_source.ingest_tableau_site() else: - site = self.server.sites.get_by_id(self.server.site_id) + site = None + with self.report.report_exc( + title="Unable to fetch site details. Site hierarchy may be incomplete and external urls may be missing.", + message="This usually indicates missing permissions. Ensure that you have all necessary permissions.", + level=StructuredLogLevel.WARN, + ): + site = self.server.sites.get_by_id(self.server.site_id) + site_source = TableauSiteSource( config=self.config, ctx=self.ctx, site=site, + site_id=self.server.site_id, report=self.report, server=self.server, platform=self.platform, @@ -740,7 +750,8 @@ def __init__( self, config: TableauConfig, ctx: PipelineContext, - site: SiteItem, + site: Optional[SiteItem], + site_id: Optional[str], report: TableauSourceReport, server: Server, platform: str, @@ -749,9 +760,16 @@ def __init__( self.report = report self.server: Server = server self.ctx: PipelineContext = ctx - self.site: SiteItem = site self.platform = platform + self.site: Optional[SiteItem] = site + if site_id is not None: + self.site_id: str = site_id + else: + assert self.site is not None, "site or site_id is required" + assert self.site.id is not None, "site_id is required when site is provided" + self.site_id = self.site.id + self.database_tables: Dict[str, DatabaseTable] = {} self.tableau_stat_registry: Dict[str, UsageStat] = {} self.tableau_project_registry: Dict[str, TableauProject] = {} @@ -805,7 +823,7 @@ def dataset_browse_prefix(self) -> str: def _re_authenticate(self): tableau_auth: Union[ TableauAuth, PersonalAccessTokenAuth - ] = self.config.get_tableau_auth(self.site.content_url) + ] = self.config.get_tableau_auth(self.site_id) self.server.auth.sign_in(tableau_auth) @property @@ -3189,10 +3207,10 @@ def emit_project_in_topological_order( else: # This is a root Tableau project since the parent_project_id is None. # For a root project, either the site is the parent, or the platform is the default parent. - if self.config.add_site_container and self.site and self.site.id: + if self.config.add_site_container: # The site containers have already been generated by emit_site_container, so we # don't need to emit them again here. - parent_project_key = self.gen_site_key(self.site.id) + parent_project_key = self.gen_site_key(self.site_id) yield from gen_containers( container_key=project_key, @@ -3209,12 +3227,12 @@ def emit_project_in_topological_order( yield from emit_project_in_topological_order(project) def emit_site_container(self): - if not self.site or not self.site.id: + if not self.site: logger.warning("Can not ingest site container. No site information found.") return yield from gen_containers( - container_key=self.gen_site_key(self.site.id), + container_key=self.gen_site_key(self.site_id), name=self.site.name or "Default", sub_types=[c.SITE], ) diff --git a/metadata-ingestion/tests/integration/tableau/test_tableau_ingest.py b/metadata-ingestion/tests/integration/tableau/test_tableau_ingest.py index 6c45b8a47de41..38a53b323876d 100644 --- a/metadata-ingestion/tests/integration/tableau/test_tableau_ingest.py +++ b/metadata-ingestion/tests/integration/tableau/test_tableau_ingest.py @@ -1028,6 +1028,7 @@ def check_lineage_metadata( ctx=context, platform="tableau", site=SiteItem(name="Site 1", content_url="site1"), + site_id="site1", report=TableauSourceReport(), server=Server("https://test-tableau-server.com"), ) @@ -1248,6 +1249,7 @@ def test_permission_mode_switched_error(pytestconfig, tmp_path, mock_datahub_gra config=mock.MagicMock(), ctx=mock.MagicMock(), site=mock.MagicMock(), + site_id=None, server=mock_sdk.return_value, report=reporter, )